Microsoft MVP성태의 닷넷 이야기
개발 환경 구성: 379. Azure Web App 확장 예제 제작 [링크 복사], [링크+제목 복사]
조회: 733
글쓴 사람
정성태 (kevin13@chol.net)
홈페이지
첨부 파일
 

Azure Web App 확장 예제 제작

사실 지난번에 소개한 글만 봐도,

Azure Web App 확장 예제 - Simple WebSite Extension
; http://www.sysnet.pe.kr/2/0/11505

확장 프로그램을 만들 기본이 마련됩니다. 그래도 다시 정리하는 의미에서 바닥부터 만들어 보겠습니다.

우선 웹 사이트 하나를 만듭니다. (여기서는 ASP.NET WebForms 프로젝트로 시작합니다.)

유형: ASP.NET Web Application - Empty - Web Forms
이름: MyAppServiceExtension

그다음 default.aspx를 추가하고 다음과 같이 환경 변수를 출력하는 코드만 넣어두겠습니다. (따라서, 이 확장이 설치되면 App Service가 운영 중인 가상 머신의 환경 변수를 모두 확인할 수 있습니다. 참고로, AppService의 환경 변수는 Kudu에서 기본 출력해 줍니다.)

<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="Default.aspx.cs" Inherits="MyAppExtension.Default" %>

<!DOCTYPE html>

<html xmlns="http://www.w3.org/1999/xhtml">
<head runat="server">
    <title></title>
</head>
<body>
    Ver 1.0<br />
    <form id="form1" runat="server">
        <div>
            <table>
            <%
    foreach (System.Collections.DictionaryEntry entry in Environment.GetEnvironmentVariables())
    {
            %>
        <tr>
                <td><%=entry.Key.ToString()%></td>
                <td><%=entry.Value.ToString()%></td>
        </tr>
            <%
    }
             %>
                </table>
        </div>
    </form>
</body>
</html>

그다음, 웹 사이트 프로젝트의 루트에 applicationHost.xdt 파일을 다음의 내용으로 생성합니다.

<?xml version="1.0" encoding="utf-8" ?> 
<configuration xmlns:xdt="http://schemas.microsoft.com/XML-Document-Transform">
  <system.applicationHost>
    <sites>
      <site name="%XDT_SCMSITENAME%" xdt:Locator="Match(name)">
          <application path="/MyAppExtension" xdt:Locator="Match(path)" xdt:Transform="Remove" />
          <application path="/MyAppExtension" applicationPool="%XDT_APPPOOLNAME%" xdt:Transform="Insert">
          <virtualDirectory path="/" physicalPath="%XDT_EXTENSIONPATH%" />
        </application>
      </site>
    </sites>
  </system.applicationHost>
</configuration>

(위의 path 부분은 여러분의 상황에 맞는 이름으로 바꿔도 됩니다.)

이제 비주얼 스튜디오의 솔루션 탐색기에서 프로젝트 노드를 우 클릭해 "Publish" 메뉴로 "Folder"에 배포합니다. 이렇게 만들어진 확장을 배포하기 위해 폴더의 내용을 하나의 .zip 파일로 압축하고 이 파일을 (Windows 10 1803 빌드부터 기본 포함되어 있기도 한) curl.exe를 이용해 Azure Portal의 "Deployment Trigger Url(배포 트리거 URL)"로 배포하면 됩니다.

여기서 한 가지, 고려 사항이 있는데요. zip 파일은 반드시 내부에 "/SiteExtensions/...[extension_name]..."의 디렉터리 구조를 포함해야 합니다. 즉, 해당 zip 파일을 압축을 풀었을 때 다음과 같은 구조로 나와야 합니다.

/SiteExtensions
            /MyAppExtension
                    /app.config
                    /...[이하 생략]...

따라서 Publish 메뉴에서 Folder 설정을 "bin\Release\Publish"에서 "bin\Release\Publish\SiteExtensions\MyAppServiceExtension"으로 바꿔야 하고 zip 파일 생성은 ""bin\Release\Publish" 폴더를 기준으로 압축해야 합니다. 참고로, 다음의 글에 따라 FolderProfile.pubxml 파일을 변경하면 비주얼 스튜디오에서 배포(Publish) 시 압축파일까지 함께 생성할 수 있습니다.

MSBuild를 이용해 프로젝트 배포 후 결과물을 zip 파일로 압축하는 방법
; http://www.sysnet.pe.kr/2/0/11508

<?xml version="1.0" encoding="utf-8"?>
<!--
This file is used by the publish/package process of your Web project. You can customize the behavior of this process
by editing this MSBuild file. In order to learn more about this please visit https://go.microsoft.com/fwlink/?LinkID=208121. 
-->
<Project ToolsVersion="4.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
    <PropertyGroup>
        ...[생략]...
    </PropertyGroup>

    <PropertyGroup>
        <MyZipFilePath>$(publishUrl)\..\..\..\$(TargetName).zip</MyZipFilePath>
        <MyPublishFolderPath>$(publishUrl)\..\..</MyPublishFolderPath>
    </PropertyGroup>

    <Target Name="MyPublish" AfterTargets="GatherAllFilesToPublish">
        <Delete Files="$(MyZipFilePath)" />
        <Exec Command="powershell.exe -nologo -noprofile -command &quot;&amp; { Add-Type -A 'System.IO.Compression.FileSystem'; [IO.Compression.ZipFile]::CreateFromDirectory('$(MyPublishFolderPath)', '$(MyZipFilePath)'); }&quot;" />
    </Target>
</Project>

생성된 (이 글에서는 MyWebAppExtension.zip) zip 파일은 (Windows 10 1803 빌드부터 기본 포함되어 있기도 한) curl.exe를 이용해 Azure Portal의 "Deployment Trigger Url(배포 트리거 URL)"로 배포할 수 있습니다.

curl.exe -k -v -T "E:\...\MyAppExtension.zip" "https://....scm.azurewebsites.net/zip"

정상적으로 배포가 되었으면 다음의 URL을 통해 웹앱 확장이 잘 동작하는지 확인할 수 있습니다.

https://[app_service_name].scm.azurewebsites.net/MyAppExtension/

(일부 변경은 Azure Portal이나 Kudu Site Extensions 화면을 통해 재시작해줘야 반영됩니다.)




다음은 AppService 웹 사이트가 Azure VM 내 위치하는 폴더의 경로입니다.

D:\home\site\wwwroot\

그리고 그 AppService의 확장들은 다음의 폴더에 배포됩니다.

D:\home\SiteExtensions\...[your_extension]...

확장들이 포함한 applicationHost.xdt 파일은 대충 다음과 같은 식의 경로에 있는 applicationHost.config 파일과 병합됩니다.

C:\DWASFiles\Sites\...app_service...\Temp\applicationhost.config

위의 applicationHost.config 파일에 각각의 Extensions들에 의해 어떤 변경이 이뤄졌는지에 대한 로그를 다음의 경로에서 확인할 수 있습니다.

D:\home\LogFiles
                \*_SCM.log (예: RD0004FFE3003C_2018-05-28_06-53-21_SCM.log)

참고로, 위와 같은 동작을 하는데 꼭 확장을 만들 필요는 없습니다. 아래의 글에 보는 바와 같이,

Making changes to the applicationHost.config on Azure App Service
; https://blogs.msdn.microsoft.com/benjaminperkins/2015/03/03/making-changes-to-the-applicationhost-config-on-azure-websites/

d:/home/site 폴더에만 있으면 xdt 파일이 적용됩니다.




개발 팁을 하나 소개하면, App Service 확장은 로컬에서 개발하고 Azure로 올릴 텐데요, 이때 환경에 따른 코드 수행을 제어할 필요가 있습니다. 이에 대해 다음의 글에도 나오지만,

Writing a Site Extension for Azure Websites
; https://azure.microsoft.com/en-us/blog/writing-a-site-extension-for-azure-websites/

그럴 때 "home" 환경 변수 여부를 이용하면 됩니다.

if (Environment.GetEnvironmentVariable("home") != null)
{
    // in Azure
}
else
{
    // Allows us to run locally.
}

물론, 여러분의 로컬 PC에 'home' 환경 변수를 정의해 두었다면 다음의 변수들이 후보가 될 수 있습니다.

WEBSITE_HOME_STAMPNAME
WEBSITE_PROACTIVE_AUTOHEAL_ENABLED
ScmType
APPSETTING_WEBSITE_AUTH_ENABLED
WEBSITE_SCM_ALWAYS_ON_ENABLED
REGION_NAME
WEBSITE_SCM_SEPARATE_STATUS
WEBSITE_SITE_NAME
WEBSITE_AUTH_AUTO_AAD
WEBSITE_CURRENT_STAMPNAME
APPSETTING_ScmType
WEBSITE_COMPUTE_MODE
WEBSITE_HOSTNAME
WEBSITE_IIS_SITE_NAME
HOME_EXPANDED
WEBSITE_RESOURCE_GROUP
WEBSITE_AUTH_LOGOUT_PATH
WEBSITE_LOCALCACHE_ENABLED
WEBSITE_INSTANCE_ID
WEBSITE_SKU
WEBSITE_OWNER_NAME
WEBSITE_RELAYS




curl.exe로 배포 시 다음과 같은 오류가 발생한다면?

{"Message":"An error has occurred.","ExceptionMessage":"Access to the path 'D:\\home\\app.config' is denied.","ExceptionType":"System.UnauthorizedAccessException","StackTrace":" at System.IO.__Error.WinIOError(Int32 errorCode, String maybeFullPath)\r\n at System.IO.FileStream.Init(String path, FileMode mode, FileAccess access, Int32 rights, Boolean useRights, FileShare share, Int32 bufferSize, FileOptions options, SECURITY_ATTRIBUTES secAttrs, String msgPath, Boolean bFromProxy, Boolean useLongPath, Boolean checkHost)\r\n at System.IO.FileStream..ctor(String path, FileMode mode, FileAccess access, FileShare share)\r\n at System.IO.FileInfo.Open(FileMode mode, FileAccess access)\r\n at System.IO.Abstractions.FileInfoWrapper.Open(FileMode mode, FileAccess access)\r\n at Kudu.Core.Infrastructure.ZipArchiveExtensions.Extract(ZipArchive archive, String directoryName) in C:\\Kudu Files\\Private\\src\\master\\Kudu.Core\\Infrastructure\\ZipArchiveExtensions.cs:line 130\r\n
...[생략]...at System.Web.Http.Dispatcher.HttpControllerDispatcher.<SendAsync>d__1.MoveNext()"}* Connection #0 to host jpublic.scm.azurewebsites.net left intact


확장 모듈을 담은 .zip 파일에 "/SiteExtensions/...[extension name]..."과 같은 디렉터리 구조를 담지 않고 있기 때문입니다. 따라서 그 디렉터리를 고려해 다시 압축한 후 업로드하면 정상적으로 설치가 됩니다.




이 글을 실습하면서 zip 파일 압축 단계에서 다음과 같은 오류가 발생할 수 있습니다.

2>Transformed Web.config using E:\TestWebApp\Web.Release.config into obj\Release\TransformWebConfig\transformed\Web.config.
2>Copying all files to temporary location below for package/publish:
2>obj\Release\Package\PackageTmp.
2>7zipping files...
2>powershell.exe -nologo -noprofile -command "& { Add-Type -A 'System.IO.Compression.FileSystem'; [IO.Compression.ZipFile]::CreateFromDirectory('...', '...'); }"
2>Exception calling "CreateFromDirectory" with "2" argument(s): "The process cannot access the file 'E:\TestWebApp\bin\Release\Publish\TestWebApp.zip' because it is being used by another process."
2>At line:1 char:53
2>+ ... ileSystem'; [IO.Compression.ZipFile]::CreateFromDirectory('E:\...
2>+                 ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
2>    + CategoryInfo          : NotSpecified: (:) [], MethodInvocationException
2>    + FullyQualifiedErrorId : IOException
2> 
2>Publishing folder /...
2>Publishing folder bin...
2>Publishing folder bin/roslyn...
2>Publishing folder Content...
2>Publishing folder fonts...
2>Publishing folder Scripts...
2>Publishing folder Views...
2>Publishing folder Views/Home...
2>Publishing folder Views/Shared...
2>Web App was published successfully file:///E:/TestWebApp/bin/Release/Publish/SiteExtensions/TestWebApp
2>

이건 zip 파일이 생성되는 폴더가 압축을 하려는 폴더와 같은 경우에 발생합니다. 즉, 다음과 같은 식으로 설정하고 빌드하면,

<PropertyGroup>
    <MyZipFilePath>$(ProjectDir)$(PublishUrl)\..\..\$(TargetName).zip</MyZipFilePath>
    <MyPublishFolderPath>$(ProjectDir)$(PublishUrl)\..\..</MyPublishFolderPath>
</PropertyGroup>

<Target Name="MyPublish" AfterTargets="GatherAllFilesToPublish">
    
    <Delete Files="$(MyZipFilePath)" />
      
    <Exec Command="powershell.exe -nologo -noprofile -command &quot;&amp; { Add-Type -A 'System.IO.Compression.FileSystem'; [IO.Compression.ZipFile]::CreateFromDirectory('$(MyPublishFolderPath)', '$(MyZipFilePath)'); }&quot;"/>

</Target>

압축 대상 및 그것의 zip 파일이 "$(ProjectDir)$(PublishUrl)\..\.." 폴더를 대상으로 했기 때문에 "The process cannot access the file..." 오류가 발생하는 것입니다.




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]





[최초 등록일: ]
[최종 수정일: 6/9/2018 ]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:kevin13@chol.com

비밀번호

댓글 쓴 사람
 



2018-10-11 02시16분
[초보애저] 항상 감사드립니다.
*_SCM.log 에서 확인가능하다고 하셨는데요. *Main.log라는 것도 있던데 이것은 무엇을 의미하는지 알수 있을까요?
SCM log를 통해 적용이 확인되면, *Main.log가 없더라도 자동으로 실제 Main에도 적용이 되었다고보면 되는 것인가요?
[손님]
2018-10-11 05시02분
이 글에서 설명한 것은 *_SCM.log입니다. 해당 로그 파일이 생성되는 LogFiles 폴더는 다양한 로그들이 쌓이는 것이므로 *Main.log가 어떤 역할을 하는지는 그 내용을 보시고 판단하면 됩니다.
정성태

[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
11757정성태10/19/201820개발 환경 구성: 416. Visual Studio 2017을 이용한 아두이노 프로그램 개발(및 디버깅)
11756정성태10/19/201810오류 유형: 500. Visual Studio Code의 아두이노 프로그램 개발 시 인텔리센스가 안 된다면?
11755정성태10/19/201812오류 유형: 499. Visual Studio Code extension for Arduino - #include errors detected.
11754정성태10/19/201814개발 환경 구성: 415. Visual Studio Code를 이용한 아두이노 프로그램 개발 - 새 프로젝트
11753정성태10/19/201844개발 환경 구성: 414. Visual Studio Code를 이용한 아두이노 프로그램 개발
11752정성태10/18/201826오류 유형: 498. SQL 서버 - Database source is not a supported version of SQL Server
11751정성태10/18/201831오류 유형: 497. Visual Studio 실행 시 그래픽이 투명해진다거나, 깨진다면?
11750정성태10/18/201826오류 유형: 496. 비주얼 스튜디오 - One or more projects in the solution were not loaded correctly.
11749정성태10/18/201873개발 환경 구성: 413. 비주얼 스튜디오에서 작성한 프로그램을 빌드하는 가장 쉬운 방법
11748정성태10/18/201822개발 환경 구성: 412. Arduino IDE를 Store App으로 설치한 경우 컴파일만 되고 배포가 안 되는 문제
11747정성태10/17/201890.NET Framework: 799. C# - DLL에도 EXE처럼 Main 메서드를 넣어 실행할 수 있도록 만드는 방법파일 다운로드1
11746정성태10/15/201852개발 환경 구성: 411. Bitvise SSH Client의 인증서 모드에서 자동 로그인 방법파일 다운로드1
11745정성태10/15/201867오류 유형: 495. TFS 파일/폴더 삭제 - The item [...] could not be found in your workspace, or you do not have permission to access it.
11744정성태10/15/201844개발 환경 구성: 410. msbuild로 .pubxml 설정에 따른 배포 파일을 만드는 방법
11743정성태10/15/201855웹: 37. Bootstrap의 dl/dt/dd 조합에서 문자열이 잘리지 않도록 CSS 설정
11742정성태10/15/201853스크립트: 13. 윈도우 배치(Batch) 스크립트에서 날짜/시간 문자열을 구하는 방법
11741정성태10/15/201847Phone: 13. Android - LinearLayout 간략 설명
11740정성태10/15/201878사물인터넷: 51. Synology NAS(DS216+II)를 이용한 원격 컴퓨터의 전원 스위치 제어
11739정성태10/15/201853Windows: 151. 윈도우 10의 전원 관리가 "균형 조정(Balanced)"으로 바뀌는 문제
11738정성태10/15/201844오류 유형: 494. docker - 윈도우에서 실행 시 "unknown shorthand flag" 오류
11737정성태10/13/201864오류 유형: 493. Azure Kudu - There are 395 items in this directory, but maxViewItems is set to 299
11736정성태10/12/201875오류 유형: 492. Visual Studio 로딩 시 오류 - The 'Scc Display Information' package did not load correctly.
11735정성태10/12/201891VS.NET IDE: 129. Visual Studio - 특정 문자(열)를 개행 문자로 바꾸는 방법
11734정성태10/10/2018173Linux: 4. Synology NAS(DS216+II)에 FTDI 장치 연결 후 C#(.NET Core)으로 DTR 제어파일 다운로드1
11733정성태10/11/2018178Linux: 3. Synology NAS(DS216+II)에서 FTDI 장치를 C/C++로 제어
[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...