Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 5개 있습니다.)
(시리즈 글이 10개 있습니다.)
VS.NET IDE: 60. Output 경로에 매크로 상수 사용하는 방법
; https://www.sysnet.pe.kr/2/0/688

개발 환경 구성: 91. MSBuild를 이용한 닷넷 응용프로그램의 플랫폼(x86/x64)별 빌드
; https://www.sysnet.pe.kr/2/0/963

개발 환경 구성: 93. MSBuild를 이용한 닷넷 응용프로그램의 다중 어셈블리 출력 빌드
; https://www.sysnet.pe.kr/2/0/965

개발 환경 구성: 102. MSBuild - DefineConstants에 다중 전처리 값 설정
; https://www.sysnet.pe.kr/2/0/988

개발 환경 구성: 115. MSBuild - x86/x64, .NET 2/4, debug/release 빌드에 대한 배치 처리
; https://www.sysnet.pe.kr/2/0/1017

개발 환경 구성: 372. MSBuild - 빌드 전/후, 배포 전/후 실행하고 싶은 Task 정의
; https://www.sysnet.pe.kr/2/0/11507

개발 환경 구성: 452. msbuild - csproj에 환경 변수 조건 사용
; https://www.sysnet.pe.kr/2/0/11985

개발 환경 구성: 580. msbuild의 Exec Task에 robocopy를 사용하는 방법
; https://www.sysnet.pe.kr/2/0/12716

개발 환경 구성: 693. msbuild - .NET Core/5+ 프로젝트에서 resgen을 이용한 리소스 파일 생성 방법
; https://www.sysnet.pe.kr/2/0/13481

닷넷: 2235. MSBuild - AccelerateBuildsInVisualStudio 옵션
; https://www.sysnet.pe.kr/2/0/13593




MSBuild - 빌드 전/후, 배포 전/후 실행하고 싶은 Task 정의

msbuild도 다른 빌드 도구들처럼 사용자가 임의로 Task를 작성해 추가할 수 있습니다. 예를 들면서 설명해 보면!

예제를 위해 Console Application 프로젝트를 하나 만들고 .csproj 파일의 내용에 다음과 같이 추가해 줍니다.

<?xml version="1.0" encoding="utf-8"?>
<Project ToolsVersion="15.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
  <Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
...[생략]...

  <ItemGroup>
    <None Include="App.config" />
  </ItemGroup>
  <Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />

  <Target Name="BeforeBuild">
       <Message Importance="high" Text="BeforeBuild(high) - Message Task $(MSBuildProjectFile)" /> 
       <Message Importance="normal" Text="BeforeBuild(normal) - Message Task $(MSBuildProjectFile)" /> 
  </Target>

  <Target Name="AfterBuild">
       <Message Importance="high" Text="AfterBuild - Message Task $(MSBuildProjectFile)" /> 
  </Target>

</Project>

이후 해당 프로젝트를 빌드하면 출력 창에 다음과 같은 내용을 확인할 수 있습니다.

1>------ Rebuild All started: Project: ConsoleApp1, Configuration: Debug Any CPU ------
1>  BeforeBuild(high) - Message Task ConsoleApp1.csproj
1>  ConsoleApp1 -> E:\ConsoleApp1\ConsoleApp1\bin\Debug\ConsoleApp1.exe
1>  AfterBuild - Message Task ConsoleApp1.csproj
========== Rebuild All: 1 succeeded, 0 failed, 0 skipped ==========

msbuild는 Microsoft.CSharp.targets 파일 내에서 빌드의 전/후에 각각 BeforeBuild와 AfterBuild에 대한 Target이 정의되어 있으면 이를 호출하도록 하고 있기 때문에 저렇게 Message Task가 동작하게 되는 것입니다.




msbuild 스크립트 파일(C# 프로젝트인 경우 .csproj 파일)도 C++의 #include와 유사하게 다른 스크립트 파일을 import 노드를 이용해 포함할 수 있습니다. 일반적으로 비주얼 스튜디오에서 만든 C# 프로젝트는 다음의 스크립트 파일을 포함합니다.

<!-- %ProgramFiles(x86)%\Microsoft Visual Studio\2017\Enterprise\MSBuild\15.0\Bin\Microsoft.CSharp.targets --> 
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />

위의 Microsoft.CSharp.targets은 다시 .NET Framework이 설치된 폴더의 Microsoft.CSharp.targets을 import합니다.

<!-- C:\Windows\Microsoft.NET\Framework\v4.0.30319\Microsoft.CSharp.targets --> 
<CSharpTargetsPath>$(MSBuildFrameworkToolsPath)\Microsoft.CSharp.targets</CSharpTargetsPath>

.NET 설치 폴더에 있는 Microsoft.CSharp.targets은 동일 폴더의 Microsoft.Common.targets을 import하는데,

<!-- C:\Windows\Microsoft.NET\Framework\v4.0.30319\Microsoft.Common.targets --> 
<Import Project="Microsoft.Common.targets" />

(이 외에 프로젝트 유형에 따라 필요한 .targets 파일들이 import됩니다.)

바로 저 파일에 BeforeBuild와 AfterBuild가 이런 식으로 정의되어 있습니다.

  <PropertyGroup>
    <BuildDependsOn>
      BeforeBuild;
      CoreBuild;
      AfterBuild
    </BuildDependsOn>
  </PropertyGroup>

  <!--
    ============================================================
                                        BeforeBuild
    Redefine this target in your project in order to run tasks just before Build
    ============================================================
    -->
  <Target Name="BeforeBuild"/>

  <!--
    ============================================================
                                        AfterBuild
    Redefine this target in your project in order to run tasks just after Build
    ============================================================
    -->
  <Target Name="AfterBuild"/>

하지만, 저것은 로슬린 이전에 그랬고 Roslyn이 도입되면서 다음의 .targets 파일이 import 되고,

C:\Program Files (x86)\Microsoft Visual Studio\2017\Enterprise\MSBuild\15.0\Bin\Roslyn\Microsoft.CSharp.Core.targets

Microsoft.CSharp.Core.targets 파일에서 다시 import하는 아래의 파일에 BeforeBuild, AfterBuild가 정의되어 있습니다.

C:\Program Files (x86)\Microsoft Visual Studio\2017\Enterprise\MSBuild\15.0\Bin\Microsoft.Common.CurrentVersion.targets

로슬린 전/후에 상관없이 BeforeBuild/AfterBuild의 처리 방식은 본격적인 빌드 작업을 담당하는 CoreBuild의 전/후에 비어 있는 Target 작업을 하나씩 미리 정의해 둔 것으로 동일합니다. 그런데 여기서 주의할 점이 하나 있습니다. 주석에 써진 데로 이것은 "redefine"이기 때문에 동일한 Target Name을 갖는 작업이 2개 이상 정의되어 있으면 나중에 정의된 것이 덮어써서 동작하는 부작용이 있습니다. 가령, Microsoft.Common.CurrentVersion.targets 파일을 편집해 BeforeBuild를 다음과 같이 수정했다면,

<Target Name="BeforeBuild">
    <Message Importance="high" Text="===================" />
</Target>

이후 여러분들의 .csproj 파일에서 BeforeBuild Target을 재정의하지 않았다면 빌드 시 "===================" 메시지가 보이겠지만, 재정의하는 순간부터 저 메시지를 볼 수 없습니다.

이와 함께 또 다른 문제가 있는데요, msbuild의 모든 Target 작업들이 저런 식으로 Befoer, Core, After와 같은 식으로 정의된 것은 아니기 때문에 다른 작업의 전/후로 어떤 처리를 하고 싶다면 Core를 복사해서 Before/After를 처리하는 구조로 변경해야 하는 등의 번거로움이 있습니다.

그리고 바로 이런 2가지의 문제를 극복하기 위해 나온 것이 MSBuild 4.0 이후부터 제공되는 Target 노드의 BeforeTargets / AfterTargets 속성입니다. 이를 이용하면 기존 BeforeBuild와 AfterBuild를 다음과 같은 형식으로 정의할 수 있습니다.

<Target Name="MyBeforeBuild" BeforeTargets="CoreBuild">
    <Message Text="Project File Name: $(MSBuildProjectFile)" /> 
</Target>

<Target Name="MyAfterBuild" AfterTargets="CoreBuild">
    <Message Text="Project File Name: $(MSBuildProjectFile)" /> 
</Target>




CoreBuild가 BeforeBuild/AfterBuild를 제공하는 것처럼, Publish 작업 역시 BeforePublish, AfterPublish를 제공합니다. 따라서, 배포(Publish) 전/후에 Target Task를 추가하는 것은 .csproj 파일에 다음과 같이 추가만 해주면 됩니다.

<Target Name="BeforePublish">
    <Message Importance="high" Text="=BeforePublish=" />
</Target>

<Target Name="AfterPublish">
    <Message Importance="high" Text="=AfterPublish=" />
</Target>

그런데, BeforePublish, AfterPublish를 정의하기 보단 BeforeTargets/AfterTargets을 이용한 정의가 더 낫습니다. 왜냐하면 일부 (예: ASP.NET Core 웹 애플리케이션) 프로젝트의 경우 BeforePublish, AfterPublish를 정의하면 실행이 안되는 반면 BeforeTargets/AfterTargets으로 다음과 같이 정의하면 실행이 잘 됩니다.

<Target Name="MyBeforePublish" BeforeTargets="Publish">
    <Message Importance="high" Text="$(MSBuildProjectFile)" /> 
</Target>

<Target Name="MyAfterPublish" AfterTargets="Publish">
    <Message Importance="high" Text="$(MSBuildProjectFile)" /> 
</Target>

참고로, 여러분들의 빌드 작업에 어떤 target 작업들이 실행되고 있는지 보고 싶다면 비주얼 스튜디오의 "TOOLS" / "Options" 메뉴에서 "Projects and Solutions" / "Build and Run" 범주의 "MSBuild project build output verbosity" 항목을 "Minimal"에서 다른 세부 항목 값으로 설정하면 됩니다.

sysmon_test_2.png

또한, 이 글의 예제에서 Message Task의 Importance 설정을 high로 했는데, 기본 값인 normal(또는 low)로 설정한 경우 위의 verbosity 설정 수준에 따라 출력 유무가 결정됩니다. (즉, 기본 verbosity 설정인 Minimal에서는 high로 설정한 메시지들만 Output 창에 보입니다.)

그 외에, msbuild 스크립트에서 사용 가능한 변수들의 목록은 다음의 글을 참고할 수 있습니다.

MSBuildToolsPath: C:\Program Files (x86)\Microsoft Visual Studio\2017\Enterprise\MSBuild\15.0\Bin
MSBuildFrameworkToolsPath: C:\WINDOWS\Microsoft.NET\Framework\v4.0.30319\

Team Build에 사용되는 각종 Property 값
; https://www.sysnet.pe.kr/2/0/508




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 2/21/2022]

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

비밀번호

댓글 작성자
 




... 46  47  48  49  50  51  52  53  54  55  56  57  58  59  [60]  ...
NoWriterDateCnt.TitleFile(s)
12126정성태1/25/202010732.NET Framework: 880. C# - PE 파일로부터 IMAGE_COR20_HEADER 및 VTableFixups 테이블 분석파일 다운로드1
12125정성태1/24/20208601VS.NET IDE: 141. IDE0019 - Use pattern matching
12124정성태1/23/202010435VS.NET IDE: 140. IDE1006 - Naming rule violation: These words must begin with upper case characters: ...
12123정성태1/23/202011921웹: 39. Google Analytics - gtag 함수를 이용해 페이지 URL 수정 및 별도의 이벤트 생성 방법 [2]
12122정성태1/20/20208893.NET Framework: 879. C/C++의 UNREFERENCED_PARAMETER 매크로를 C#에서 우회하는 방법(IDE0060 - Remove unused parameter '...')파일 다운로드1
12121정성태1/20/20209427VS.NET IDE: 139. Visual Studio - Error List: "Could not find schema information for the ..."파일 다운로드1
12120정성태1/19/202010889.NET Framework: 878. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 네 번째 이야기(IL 코드로 직접 구현)파일 다운로드1
12119정성태1/17/202010920디버깅 기술: 160. Windbg 확장 DLL 만들기 (3) - C#으로 만드는 방법
12118정성태1/17/202011544개발 환경 구성: 466. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 세 번째 이야기 [1]
12117정성태1/15/202010576디버깅 기술: 159. C# - 디버깅 중인 프로세스를 강제로 다른 디버거에서 연결하는 방법파일 다운로드1
12116정성태1/15/202011047디버깅 기술: 158. Visual Studio로 디버깅 시 sos.dll 확장 명령어를 (비롯한 windbg의 다양한 기능을) 수행하는 방법
12115정성태1/14/202010837디버깅 기술: 157. C# - PEB.ProcessHeap을 이용해 디버깅 중인지 확인하는 방법파일 다운로드1
12114정성태1/13/202012687디버깅 기술: 156. C# - PDB 파일로부터 심벌(Symbol) 및 타입(Type) 정보 열거 [1]파일 다운로드3
12113정성태1/12/202013305오류 유형: 590. Visual C++ 빌드 오류 - fatal error LNK1104: cannot open file 'atls.lib' [1]
12112정성태1/12/20209911오류 유형: 589. PowerShell - 원격 Invoke-Command 실행 시 "WinRM cannot complete the operation" 오류 발생
12111정성태1/12/202013111디버깅 기술: 155. C# - KernelMemoryIO 드라이버를 이용해 실행 프로그램을 숨기는 방법(DKOM: Direct Kernel Object Modification) [16]파일 다운로드1
12110정성태1/11/202011706디버깅 기술: 154. Patch Guard로 인해 블루 스크린(BSOD)가 발생하는 사례 [5]파일 다운로드1
12109정성태1/10/20209630오류 유형: 588. Driver 프로젝트 빌드 오류 - Inf2Cat error -2: "Inf2Cat, signability test failed."
12108정성태1/10/20209680오류 유형: 587. Kernel Driver 시작 시 127(The specified procedure could not be found.) 오류 메시지 발생
12107정성태1/10/202010654.NET Framework: 877. C# - 프로세스의 모든 핸들을 열람 - 두 번째 이야기
12106정성태1/8/202012030VC++: 136. C++ - OSR Driver Loader와 같은 Legacy 커널 드라이버 설치 프로그램 제작 [1]
12105정성태1/8/202010704디버깅 기술: 153. C# - PEB를 조작해 로드된 DLL을 숨기는 방법
12104정성태1/7/202011435DDK: 9. 커널 메모리를 읽고 쓰는 NT Legacy driver와 C# 클라이언트 프로그램 [4]
12103정성태1/7/202014098DDK: 8. Visual Studio 2019 + WDK Legacy Driver 제작- Hello World 예제 [1]파일 다운로드2
12102정성태1/6/202011758디버깅 기술: 152. User 권한(Ring 3)의 프로그램에서 _ETHREAD 주소(및 커널 메모리를 읽을 수 있다면 _EPROCESS 주소) 구하는 방법
12101정성태1/5/202011106.NET Framework: 876. C# - PEB(Process Environment Block)를 통해 로드된 모듈 목록 열람
... 46  47  48  49  50  51  52  53  54  55  56  57  58  59  [60]  ...