msbuild를 사용하여 파일 시스템 게시 프로필 실행


86

VS2010으로 만든 AC # .Net 4.0 프로젝트가 있으며 이제 VS2012로 액세스됩니다.

이 웹 사이트에서 필요한 파일 만 대상 위치 (C : \ builds \ MyProject [Files]) 에 게시하려고합니다 .

내 파일 구조 : ./ProjectRoot/MyProject.csproj ./ProjectRoot/Properties/PublishProfiles/FileSystemDebug.pubxml

MSBuild를 통해 다음을 실행하고 있습니다.

C : \ Windows \ Microsoft.NET \ Framework \ v4.0.30319 \ MSBuild.exe ./ProjectRoot/MyProject.csproj / p : DeployOnBuild = true /p:PublishProfile=./ProjectRoot/Properties/PublishProfiles/FileSystemDebug.pubxml

다음은 FileSystemDebug.pubxml의 xml입니다.

<Project ToolsVersion="4.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
  <PropertyGroup>
    <WebPublishMethod>FileSystem</WebPublishMethod>
    <LastUsedBuildConfiguration>Release</LastUsedBuildConfiguration>
    <LastUsedPlatform>Any CPU</LastUsedPlatform>
    <SiteUrlToLaunchAfterPublish />
    <ExcludeApp_Data>False</ExcludeApp_Data>
    <publishUrl>C:\builds\MyProject\</publishUrl>
    <DeleteExistingFiles>True</DeleteExistingFiles>
  </PropertyGroup>
</Project>

결과 동작은 다음과 같습니다.

  • ./ProjectRoot/obj/Debug/Package/MyProject.zip에 zip 파일이 생성됩니다.
  • <publishUrl>C:\builds\MyProject\</publishUrl>WTF에 배포 된 항목이 없습니다.
  • 생성 된 zip 파일은 돼지 아침 식사이며 응용 프로그램에 필요하지 않은 파일로 가득합니다.

Visual Studio를 통해이 게시 프로필을 실행하면 * C : \ builds \ MyProject *에 폴더가 생성되고 원하는 정확한 아티팩트가 포함됩니다.

msbuild에서이 간단한 결과를 어떻게 얻습니까?

답변:


51

참고 : Visual Studio 2015에서 동일한 문제가 발생했습니다. 많은 시간을 시도한 후 이제 msbuild myproject.csproj /p:DeployOnBuild=true /p:PublishProfile=myprofile.

.csproj 파일을 편집하여 작동하도록해야했습니다. 다음과 같은 줄이 포함되어 있습니다.

<Import Project="$(MSBuildExtensionsPath32)\Microsoft\VisualStudio\v10.0\WebApplications\Microsoft.WebApplication.targets" 
  Condition="false" />

이 줄을 다음과 같이 변경했습니다.

<Import Project="$(MSBuildExtensionsPath32)\Microsoft\VisualStudio\v14.0\WebApplications\Microsoft.WebApplication.targets" />

(필자는 10.0을 14.0으로 변경했는데 이것이 필요한지 확실하지 않습니다.하지만 컨디션 부분은 확실히 제거해야했습니다.)


1
Condition="false"이전 버전과의 호환성을 위해 조건부 가져 오기가 존재합니다. VS2010에서는 잘못된 조건으로 인해 건너 뛴 경우에도이 가져 오기가 존재해야합니다. 다시 보면 csproj $(VSToolsPath)\WebApplications\Microsoft.WebApplication.targets에 현재 버전의 Visual Studio에 대한 대상 파일로 확인되는 다른 가져 오기가 포함되어 있음을 알 수 있습니다 .
스티븐 Liekens

3
$(MSBuildToolsVersion)적절한 VS 버전을 설명하는 경로에서 의 전략적인 사용에 유의하십시오 <Import Project="$(MSBuildExtensionsPath32)\Microsoft\VisualStudio\v$(MSBuildToolsVersion)\WebApplications\Microsoft.WebApplication.targets" Condition="false" />.. 이 VS2015 업데이트에 나를 위해 1 일
알 다스

42

여기에서 답을 찾았습니다. http://www.digitallycreated.net/Blog/59/locally-publishing-a-vs2010-asp.net-web-application-using-msbuild

Visual Studio 2010에는 단추 클릭만으로 웹앱 프로젝트를 쉽게 게시 할 수있는 새로운 웹 응용 프로그램 프로젝트 게시 기능이 있습니다. 백그라운드에서 Web.config 변환 및 패키지 빌드는 프로젝트 파일로 가져온 대규모 MSBuild 스크립트에 의해 수행됩니다 (C : \ Program Files (x86) \ MSBuild \ Microsoft \ VisualStudio \ v10.0 \ Web \ Microsoft .Web.Publishing.targets). 안타깝게도 스크립트는 매우 복잡하고 지저분하며 문서화되지 않았습니다 (그 외에는 철자가 잘못되어 파일에 대부분 쓸모없는 주석이 있습니다). 그 파일의 큰 순서도와 그것에 연결하는 방법에 대한 문서는 좋지만 슬프게도 부족한 것 같습니다 (또는 적어도 찾을 수 없습니다).

불행히도 이것은 명령 줄을 통해 게시를 수행하는 것이 필요한 것보다 훨씬 더 불투명하다는 것을 의미합니다. 요즘 많은 상점이 지속적 통합 서버를 사용하고 일부는 (VS2010 게시 기능이 많은 도움이 될 수있는) 자동화 된 배포를 수행하기 때문에이 영역의 문서가 부족하다는 사실에 놀랐습니다. 쉽게!) 기능의 주요 요구 사항이 될 것입니다.

어쨌든, 몇 시간 동안 Microsoft.Web.Publishing.targets 파일을 파헤 치고 시행 착오 벽에 머리를 부딪친 후, 저는 Visual Studio가 "파일 시스템에 게시"를 한 번의 클릭으로 수행하는 방법을 알아낼 수있었습니다. 및 "배포 패키지 빌드"기능. MSBuild 스크립팅에 대해 좀 더 알아 보겠습니다. MSBuild에 익숙하지 않다면이 크래시 코스 MSDN 페이지를 확인하는 것이 좋습니다.

파일 시스템에 게시

VS2010 파일 시스템에 게시 대화 상자 파일 시스템에 게시는 MSBuild의 현명한 사용이 발생할 것으로 예상했기 때문에 잠시 시간이 걸렸습니다. 대신 VS2010은 매우 이상한 작업을 수행합니다. 프로젝트의 obj 폴더에 웹 앱 파일을 준비하는 일종의 반 배포를 수행하기 위해 MSBuild를 호출 한 다음 해당 파일의 수동 복사를 수행하는 것 같습니다 (예 : MSBuild 외부). 대상 게시 폴더에. MSBuild는 파일 (및 기타 빌드 관련 항목)을 복사하도록 설계 되었기 때문에 이것은 정말 엉뚱한 동작입니다. 따라서 전체 프로세스가 VS2010이 호출 한 하나의 MSBuild 대상이고 대상이 아니라 수동 복사본이면 이해가 될 것입니다.

즉, 명령 줄에서 MSBuild를 통해이 작업을 수행하는 것은 특정 대상이있는 프로젝트 파일을 호출하고 일부 속성을 설정하는 것만 큼 간단하지 않습니다. VS2010에서 수행해야하는 작업을 수행해야합니다. 절반 배포를 수행하는 대상을 직접 만든 다음 결과를 대상 폴더에 복사합니다. 프로젝트 파일을 편집하려면 VS2010에서 프로젝트를 마우스 오른쪽 버튼으로 클릭하고 프로젝트 언로드를 클릭 한 다음 다시 마우스 오른쪽 버튼을 클릭하고 편집을 클릭합니다. 웹 응용 프로그램 대상 (Microsoft.WebApplication.targets,이 파일 자체가 앞서 언급 한 Microsoft.Web.Publishing.targets 파일을 가져옴)을 가져 오는 Import 요소를 찾을 때까지 아래로 스크롤합니다. 이 줄 아래에 PublishToFileSystem이라는 새 대상을 추가합니다.

<Target Name="PublishToFileSystem"
        DependsOnTargets="PipelinePreDeployCopyAllFilesToOneFolder">
    <Error Condition="'$(PublishDestination)'==''"
           Text="The PublishDestination property must be set to the intended publishing destination." />
    <MakeDir Condition="!Exists($(PublishDestination))"
             Directories="$(PublishDestination)" />

    <ItemGroup>
        <PublishFiles Include="$(_PackageTempDir)\**\*.*" />
    </ItemGroup>

    <Copy SourceFiles="@(PublishFiles)"
          DestinationFiles="@(PublishFiles->'$(PublishDestination)\%(RecursiveDir)%(Filename)%(Extension)')"
          SkipUnchangedFiles="True" />
</Target>

이 대상은 수동 복사를 수행하기 전에 VS2010이 호출하는 PipelinePreDeployCopyAllFilesToOneFolder 대상에 따라 다릅니다. Microsoft.Web.Publishing.targets를 살펴보면이 대상을 호출하면 프로젝트 파일이 _PackageTempDir 속성에 지정된 디렉터리에 배치된다는 것을 알 수 있습니다.

대상에서 호출하는 첫 번째 작업은 PublishDestination 속성이 설정되지 않은 경우에만 작업이 발생하도록 조건을 설정 한 오류 작업입니다. 이렇게하면 PublishDestination 속성을 지정하는 것을 잊은 경우 빌드 오류가 발생합니다. 그런 다음 MakeDir 작업을 호출하여 해당 PublishDestination 디렉터리가 아직없는 경우 만듭니다.

그런 다음 _PackageTempDir 폴더 아래에있는 모든 파일을 나타내는 PublishFiles라는 항목을 정의합니다. 그런 다음 모든 파일을 게시 대상 폴더로 복사하는 복사 작업이 호출됩니다. Copy 요소의 DestinationFiles 속성은 약간 복잡합니다. 항목의 변환을 수행하고 해당 경로를 PublishDestination 폴더에 루트가 지정된 새 경로로 변환합니다 (해당 % ()의 의미를 보려면 Well-Known Item Metadata를 확인하십시오).

명령 줄에서이 대상을 호출하기 위해 이제 다음 명령을 수행 할 수 있습니다 (분명히 프로젝트 파일 이름과 속성을 사용자에게 맞게 변경).

msbuild Website.csproj "/p:Platform=AnyCPU;Configuration=Release;PublishDestination=F:\Temp\Publish" /t:PublishToFileSystem

3
새 대상에 대한 코드 조각을 이해할 수 없습니다 (01 02 03 ...으로 표시됨). 편집 해 주시겠습니까?
fan711 2013-10-09

2
나는 fan711에 동의합니다. 그러나 솔루션은 링크에 설명되어 있습니다. 그러면 복사하는 것이 무엇입니까?
Антон Курьян dec.

4
@ АнтонКурьян : 링크는 시간이 지나면 죽는 경향이 있기 때문에 stackoverflow.com의 질문과 답변은 외부 리소스에 의존하지 않고 항상 독립적이어야합니다.
Oliver

결과는 게시 프로필이 MSBuild에서 전혀 사용되지 않는 것처럼 보이며 대신 패키지를 수행하는 것입니다 (아마도 기본값일까요?). 아래 솔루션은 프로파일이 수행하도록 설정된 작업을 복제하기 위해 수행하는 작업이며, Microsoft.Web.Publishing.targets는 배포 폴더 (파일 시스템 용)에서 올바른 유형을 선택하여 처리합니다. 이 문제를 해결하는 대신 여기에서 바퀴를 재발 명하는 것처럼 보입니다. 그러나 MSBuild 로그 없이는 확실히 말할 수 없습니다. 나는 일에, 내 대답에 세부 광산을 가지고
GregS

1
GregS의 솔루션이 작동하지 않습니다. 잘 빌드되지만 게시 디렉토리에 파일이 복사되지 않습니다
rushinge

19

위의 모든 답변을 시도한 후에도 여전히 문제가 발생했습니다 (Visual Studio 2013 사용). 게시 폴더에 복사 된 내용이 없습니다.

문제는 솔루션 대신 개별 프로젝트로 MSBuild를 실행하는 경우 Visual Studio 버전을 지정하는 추가 매개 변수를 입력해야한다는 것입니다.

/p:VisualStudioVersion=12.0

12.0VS2013 용이므로 사용하는 버전으로 교체하십시오. 이 매개 변수를 추가 한 후에는 작동했습니다.

전체 명령 줄은 다음과 같습니다.

MSBuild C:\PathToMyProject\MyProject.csproj /p:DeployOnBuild=true /p:PublishProfile=MyPublishProfile /p:VisualStudioVersion=12.0

여기에서 찾았습니다.

http://www.asp.net/mvc/overview/deployment/visual-studio-web-deployment/command-line-deployment

그들은 다음과 같이 진술합니다.

솔루션 대신 개별 프로젝트를 지정하는 경우 Visual Studio 버전을 지정하는 매개 변수를 추가해야합니다.


12

게시 프로필이 사용되지 않고 기본 패키징을 수행하는 것 같습니다. Microsoft Web Publish 대상은 위에서 수행 한 모든 작업을 수행하며 구성에 따라 올바른 대상을 선택합니다.

TeamCity MSBuild 단계에서 문제가 발생하지 않았지만 프로필에 대한 명시 적 경로를 지정 했으므로 .pubxml (예 : FileSystemDebug)없이 이름으로 호출해야합니다. 귀하의 표준 폴더에있는 한 찾을 수 있습니다.

예:

C:\Windows\Microsoft.NET\Framework\v4.0.30319\MSBuild.exe ./ProjectRoot/MyProject.csproj /p:DeployOnBuild=true /p:PublishProfile=FileSystemDebug

이것은 일반적으로 "C : \ Program Files (x86) \ MSBuild \ Microsoft \ VisualStudio \ v11.0 \ Web"에있는 Visual Studio 2012 버전의 Microsoft Web Publish 대상을 사용하여 수행되었습니다. 사용되는 특정 배포 유형 대상에 대한 배포 폴더를 확인하십시오.


MS는 업데이트 된 프로그래밍 덕분에 몇 년 전에 게시 된 이후로 많은 개선을 이루었습니다.
P. Roe

3

참고 : 빌드 서버에서 실행하는 것과 동일한 문제 (.NET Core 2.1 웹 프로젝트의 VS 2017에서 구동되는 msbuild 15가 설치된 Jenkins).

제 경우에는 프로필을 무시한 msbuild와 함께 "publish"대상을 사용했습니다.

그래서 내 msbuild 명령은 다음과 같이 시작되었습니다.

msbuild /t:restore;build;publish

이로 인해 게시 프로세스가 올바르게 트리거되었지만 "/ p : PublishProfile = FolderProfile"의 조합이나 변형이 내가 사용하려는 프로필 ( "FolderProfile")을 선택하는 데 작동하지 않았습니다.

게시 대상 사용을 중지했을 때 :

msbuild /t:restore;build /p:DeployOnBuild=true /p:PublishProfile=FolderProfile

나는 (어리석게도) 차이가 없을 것이라고 생각했지만 DeployOnBuild 스위치를 사용하자마자 프로필을 올바르게 선택했습니다.


3

실제로 위의 문제를 해결하는 방법에 대한 모든 답변을 내 솔루션에 병합했습니다.

  1. 내 필요에 따라 pubxml 파일을 만듭니다.
  2. 그런 다음 pubxml 파일의 모든 매개 변수를 msbuild.exe의 "/ p : foo = bar"매개 변수 목록으로 복사합니다.
  3. pubxml 파일을 버립니다.

결과는 다음과 같습니다.

msbuild /t:restore /t:build /p:WebPublishMethod=FileSystem /p:publishUrl=C:\builds\MyProject\ /p:DeleteExistingFiles=True /p:LastUsedPlatform="Any CPU" /p:Configuration=Release


1

먼저 솔루션 (프로젝트)을 게시 할 수있는 개발자 PC의 Visual Studio 버전을 확인합니다. 표시된대로 VS 2013

 /p:VisualStudioVersion=12.0

위의 명령 줄을 추가하여 프로젝트를 빌드해야하는 Visual Studio 버전의 종류를 지정합니다. 이전 답변처럼 전체 솔루션이 아닌 하나의 프로젝트 만 게시하려고 할 때 발생할 수 있습니다.

따라서 완전한 코드는 다음과 같습니다.

"C : \ Windows \ Microsoft.NET \ Framework \ v4.0.30319 \ msbuild.exe" "C : \ Program Files (x86) \ Jenkins \ workspace \ Jenkinssecondsample \ MVCSampleJenkins \ MVCSampleJenkins.csproj"/ T : Build; Package / p : Configuration = DEBUG / p : OutputPath = "obj \ DEBUG"/ p : DeployIisAppPath = "기본 웹 사이트 / jenkinsdemoapp"/p:VisualStudioVersion=12.0


1
죄송합니다 shammakalubo 당신이 질문을 많이 잘못 해석했습니다.
P. Roe

1
@shammakalubo 대답은 옳지 만 완전히 언급되지 않았습니다. 이 매개 변수는 OP에서 언급 한 명령에 추가되어야합니다. MSBuild C:\PathToMyProject\MyProject.csproj /p:DeployOnBuild=true /p:PublishProfile=MyPublishProfile /p:VisualStudioVersion=12.0그러면 다음과 같이됩니다. 이 매개 변수는 내가 누락되었고 문제를 해결했습니다. 답을 완전히 언급하기 만하면됩니다!
에드 Waqas

@WaqasShah 감사합니다. 언급했듯이 내 대답을 편집하고 전체 코드를 게시했습니다.
Shammie
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.