Understanding MSBuild
File Types
.*proj
: project identifier file for each project.props
: shared config to be imported at the beginning of the project file.targets
: shared config to be imported at the end of the project file.rsp
: msbuild response file, default options for msbuild cli so you don't have to repeat them on each build
File Structure
<Project>
: Top element<PropertyGroup>
: static values can be determined before build.- child tag can be arbitrary or pre-defined tag
- use
$(propName)
to retrieve value of a property in any expression-allowed attribute, returns empty string if undefined. - has some builtin properties preserved
<ItemGroup>
: items to be included on build, generally are config files- each child represents a type of items even when only one item was contained
- each item in the child has well-known metadata
- children with the same tag name are within a same source(they're just declared separately)
- use
@(itemType)
to retrieve a list of items from existing<ItemGroup>
- has some builtin item types preserved
- each child represents a type of items even when only one item was contained
<ItemDefinitionGroup>
: define a new shape of item with default metadata<Target>
: section to wrap sub-procedures and to be invoked during build- name it like a function
- use
Name
attribute to specify a name for it - use
DependsOnTargets
,BeforeTargets
,AfterTargets
attributes to hook to another target section
NOTE
<PropertyGroup>
, <ItemGroup>
and <Target>
could have multiple declarations for better organization or conditional declaration using Condition
attribute.
Project Attributes
SDK
: sepcify a sdk so that msbuild can prepare dedicated builtin targets and tasks for corresponding type of project.NOTE
- a project with specified
SDK
is referred as SDK-style project - a
<Project>
withSDK
would auto import standard*.targets
and*.props
- you could find standard config files from
${DOTNET_ROOT}/sdk/<dotnet_version>/Sdks/<sdk_name>/
- a project with specified
InitialTargets
: a list of targets should run first on buildDefaultTargets
: a list of targets should run afterInitialTargets
when no any target specified from msbuild cli
NOTE
Property Section
Two kinds of properties:
- reserved properties: pre-defined and cannot be overridden,
- well-known properties: pre-defined and can be overridden
- properties set by specified sdk, all other sdk inherits from
Microsoft.NET.Sdk
, see: .NET project SDKs
Property Functions
Merge Properties
<PropertyGroup>
<Foo>foo;bar</Foo>
<!-- override self with interpolating itself -->
<Foo>$(Foo);baz</Foo>
</PropertyGroup>
<Target>
<!-- foo;bar;baz -->
<Message Text="$(Foo)" Importance="high"/>
</Target>
2
3
4
5
6
7
8
9
10
Item Section
<ItemGroup>
section is generally for specifying files to be included on build. Children under <ItemGroup>
are not necessarily files, they can be any list with a name(the tag name) that has items represented as string. Attributes for child items are similar to parameters of item-cmdlet in PowerShell.
<ItemGroup>
<!-- represents a normal list -->
<!-- child tag with same name are included within same list -->
<FooList Include="foo;bar" />
<FooList Include="baz;qux" />
<MyFile Include="*.cs"/>
</ItemGroup>
2
3
4
5
6
7
8
Item Attributes
Item attributes are for controlling how items could be initialized, added and removed to a list
Include
: include items on declaration- use
KeepMetadata
orRemoveMetadata
to optionally include or exclude metadata from when creating items by transforming from another within a<Target>
xml<ItemGroup> <Old Include="*"> <Foo>foo</Foo> <Bar>bar</Bar> </Old> </ItemGroup> <Target> <ItemGroup> <New Include="@(Old)" RemoveMetadata="Foo"/> <!-- transform from Old --> </ItemGroup> <!-- Old.Foo was removed after transformation --> <Message Text="Old.Foo was removed after transformation" Condition="%(New.Foo) == ''" Importance="high"/> </Target>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16- use
KeepDuplicates
when adding new item within a<Target>
that you expect the new would be added when deplicates exist.
WARNING
The idendity of a item depends on all metadata within it, so as long as one metadata differ, it would not be considered as a duplicate.
xml<ItemGroup> <FooList Include="foo;bar" /> <FooList Include="foo;qux" /> </ItemGroup> <Target Name="Hello"> <ItemGroup> <!-- bar would not be added since it already exists in FooList --> <FooList Include="bar" KeepDuplicates="false" /> </ItemGroup> <!-- foo;bar;foo;qux --> <Message Text="@(FooList)" Importance="high"></Message> </Target>
1
2
3
4
5
6
7
8
9
10
11
12
13- use
Exclude
: exclude items on declarationxml<ItemGroup> <Compile Include="one.cs;three.cs" /> <Compile Exclude="*.fs" /> </ItemGroup>
1
2
3
4Remove
: remove items after declaration, typically inside a<Target>
section- optionally use
MatchOnMetadata
together to delete base on another type of items - optionally use
MatchOnMetadataOptions
to specify the comparing strategy on matchCaseInsensitive
: the default when unspecifiedCaseSensitive
PathLike
: match paths ignoring separator and trailing slash
xml<ItemGroup> <CSFile Include="Programs.cs"/> <CSFile Include="Foo.cs"/> <CSFile Include="Bar.cs"/> <Proj Include="Foo.csproj"/> <Proj Include="bar.csproj"/> <Proj Include="Baz.csproj"/> </ItemGroup> <Target Name="Hello"> <!-- Proj items are to be matched by metadata FileName --> <ItemGroup> <CSFile Remove="@(Proj)" MatchOnMetadata="FileName" MatchOnMetadataOptions="CaseSensitive" /> </ItemGroup> <!-- Remained cs items: Programs.cs --> <Message Text="Remained cs items: %(CSFile.Identity)" Importance="high"></Message> </Target>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18- optionally use
Update
: update metadata outside a<Target>
xml<ItemGroup> <FooList Include="foo;bar;baz"> <!-- assign FooMetaData to all items in Include --> <FooMetaData>this is a foo metadata</FooMetaData> </FooList> <!-- update FooMetaData for foo and bar --> <FooList Update="foo;bar" FooMetaData="this is a bar metadata now!"/> </ItemGroup> <Target Name="Hello"> <Message Text="%(FooList.Identity): %(FooList.FooMetaData)" Importance="high"/> <!-- foo: this is a bar metadata now! bar: this is a bar metadata now! baz: this is a foo metadata --> </Target>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
Item Functions
There's some intrinsic functions to be used to transform a item list to another or get properties like Count
of a item list.
NOTE
NOTE
If a expression includes item enumeration %(itemType)
, item functions that returns a statistics of list would return result for each distinct item
<ItemGroup>
<FooList Include="foo;bar" />
<FooList Include="foo;qux" />
</ItemGroup>
<Target Name="Hello">
<Message Text="%(FooList.Identity) @(FooList->Count())" Importance="high" />
<!-- foo 2
bar 1
qux 1 -->
</Target>
2
3
4
5
6
7
8
9
10
11
Default Filters
Builtin item tags like Compile
has some default wildcards when SDK
was sepcified.
Item Metadata
Each item has pre-defined metadata can be accessed.
NOTE
If a item does not represent a file, the most of intrinsic metadata would be empty.
Metadata Mutation
Metadata should be mutated by batching using a Condition
within a <Target>
<ItemGroup>
<Foo Include="2" Bar="foo" />
<Foo Include="1" Bar="bar" />
</ItemGroup>
<Target Name="Foo">
<ItemGroup>
<Foo Condition=" '%(Bar)' == 'blue' ">
<Bar>baz</Bar>
</Foo>
</ItemGroup>
</Target>
2
3
4
5
6
7
8
9
10
11
12
Common Items
Reference
: reference to dll filesPackageReference
: reference packages to be restored by nugetProjectReference
: reference project by project file, builds projects cascadingUsing
: extra global using by specifying namespaces
<ItemGroup>
<Using Include="System.IO.Pipes" />
</ItemGroup>
2
3
ItemDefinitionGroup Section
<ItemDefinitionGroup>
<Foo>
<!-- metadata Bar has default value bar -->
<Bar>bar</Bar>
</Foo>
</ItemDefinitionGroup>
<ItemGroup>
<Foo Include="foo"></Foo>
</ItemGroup>
<Target Name="Hello">
<!-- bar -->
<Message Text="%(Foo.Bar)" Importance="high"/>
</Target>
2
3
4
5
6
7
8
9
10
11
12
13
14
15
Expression Syntax
Expression syntax in msbuild has some flavor of Command Prompt and PowerShell.
- Operators
- metadata accessor
%(itemType.metadataName)
: iterate through the metadata of items for a same action applied on the expression. - reference accessor
$(propName)
: access properties or environment variable or (property is preferred when names conflict)- access a property with its name
- capture a value calculated from a method call in a powershell syntax
- access registry item with its path
- item accessor
@(itemType)
- use
@(itemType, '<sep>')
to concat items using separator - use
@(itemType->expr)
to collected transformed values generated byexpr
to a list,expr
can be a string interpolation or item function
xml<ItemGroup> <MyFile Include="Programs.cs"/> <MyFile Include="*.csproj"/> </ItemGroup> <Target Name="Hello"> <!-- Collected metadata: Program.cs; ConsoleApp.csproj --> <Message Text="Collected metadata: @(MyFile->'%(FileName)%(Extension)')" Importance="high" /> <!-- 5 --> <Message Text="Exists: @(MyFile->Count())" Importance="high" /> </Target>
1
2
3
4
5
6
7
8
9
10
11
12
13 - use
- metadata accessor
Escaping
For special characters, you may figure out its ASCII representation by:
[System.Uri]::EscapeDataString('"') # %22
TIP
Notation like "
is also viable
Conditional Component
Some elements supports conditional include using Condition
attribute. So one could include dedicated part of the config for different build scenarios.
<ItemGroup Condition="'$(DotNetBuildSourceOnly)' == 'true'">
<PackageVersion Include="Microsoft.Build" Version="17.3.4" />
<PackageVersion Include="Microsoft.Build.Framework" Version="17.3.4" />
<PackageVersion Include="Microsoft.Build.Tasks.Core" Version="17.3.4" />
<PackageVersion Include="Microsoft.Build.Utilities.Core" Version="17.3.4" />
</ItemGroup>
<ItemGroup Condition="'$(DotNetBuildSourceOnly)' != 'true' and '$(TargetFramework)' != 'net472'">
<PackageVersion Include="Microsoft.Build" Version="17.7.2" />
<PackageVersion Include="Microsoft.Build.Framework" Version="17.7.2" />
<PackageVersion Include="Microsoft.Build.Tasks.Core" Version="17.7.2" />
<PackageVersion Include="Microsoft.Build.Utilities.Core" Version="17.7.2" />
</ItemGroup>
2
3
4
5
6
7
8
9
10
11
12
13
Target Section
A task is a pre-defined procedure to be executed in <Target>
section in their declared order. The containing Target
is like a function which must have a name while a task is just named as its tag. MSBuild ships with some builtin task to be used out of box.
<Target>
<!-- Csc is a builtin task from sdk -->
<Csc
Sources="@(Compile)"
OutputAssembly="$(AppName).exe"
EmitDebugInformation="true" />
</Target>
2
3
4
5
6
7
NOTE
Builtin Targets
MSBuild has some pre-defined targets to perform common actions like Build
, Clean
... Targets starts with After
and Before
are preserved to be overridden as a hook on specific target, such as AfterBuild
and BeforeBuild
are hooks on Build
.
NOTE
Use dotnet msbuild -targets|-ts [proj]
to check existing targets(including builtin) from a project file.
# filter out internal targets starts with _
dotnet msbuild -targets | where { $_ -notmatch '^_' }
2
Target Hooks
DependsOnTargets
: this target must be executed after the specified targetBeforeTarget
Custom Task
MSBuild tasks are defined in certain language like csharp. You can implement one or some as nuget package. Or you can implement one using inline task directly within the config file.
NOTE
See: Task Writing
Update Metadata in Target
Such approach seems to only allow updating on all existing items since Include
could try to add new items and new metadata would result them to be duplicates.
<ItemGroup>
<FooList Include="foo;bar;baz">
<FooMetaData>this is a foo metadata</FooMetaData>
</FooList>
</ItemGroup>
<Target Name="Hello">
<ItemGroup>
<FooList> <!-- no Include here -->
<!-- update all existing items from FooList -->
<FooMetaData>this is a bar metadata now!</FooMetaData>
</FooList>
</ItemGroup>
<Message Text="%(FooList.Identity): %(FooList.FooMetaData)" Importance="high"/>
<!-- foo: this is a bar metadata now!
bar: this is a bar metadata now!
baz: this is a bar metadata now! -->
</Target>
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
Emit Property and Item from Task
- one task could emit multiple items and properties(use multiple
<Output>
) - the task supports output parameter(specific value for
TaskParameter
)- search
output parameter
to check availableTaskParameter
on documentation of a task
- search
- use
PropertyName
orItemName
attribute to specify the name to emit
<ItemGroup>
<NewDir Include="foo;bar" />
</ItemGroup>
<Target>
<MakeDir Directories="@(NewDir)">
emit DirCreated property with the value of DirectoriesCreated
<Output TaskParameter="DirectoriesCreated" ItemName="DirCreated"/>
</MakeDir>
<!-- log out new property -->
<Message Text="@(DirCreated)" Importance="high"/>
</Target>
2
3
4
5
6
7
8
9
10
11
12
Import & Evaluation Order
The only file msbuild cares for build is the *.*proj
file, all other files are just imports within it. The evaluation happens upon the project file, and splits the file into a process with priorities for different section.
SDK Style
Aforementioned SDK-style project has implicit <Import>
for standard props
and targets
.
- standard
props
were imported
Evaluation
NOTE
MSBuild files are order sensitive, order matters when properties and items may dependent on each other. MSBuild merges imported config files and proj file into one object, and then evaluate all values of different parts.
- environment variables: they're stored as properties
- imports & properties: evaluated by their declaration order
- when evaluating a import, the evaluation process applies to it recursively.
- the order between imports and properties matters if you inter-reference them on expression
- never reference items on properties not within a target
- definition of items: how should item were filtered out and captured.
- items: execute the process to fetch items and their metadata
- inline tasks(
<UsingTask>
) - targets: as well as items inside targets
IMPORTANT
Expressions are lazily evaluated until the identifier been referenced got evaluated. So any expression to be evaluated on evaluation time(not within a target) would be like a template. That is to say, the order doesn't matter when you inter-reference items and properties. See: Subtle effects of the evaluation order
<PropertyGroup>
<!-- reference before FooList were evaluated -->
<Foo>@(FooList)</Foo>
</PropertyGroup>
<ItemGroup>
<FooList Include="foo;bar"/>
</ItemGroup>
<Target Name="Hello">
<!-- foo;bar -->
<Message Text="$(Foo)" Importance="high"></Message>
</Target>
2
3
4
5
6
7
8
9
10
11
12
13
Folder-Specific Config
Folder-Specific config files includes:
Directory.Build.props
: content to be imported before standard config automatically for current or child foldersDirectory.Build.targets
: content to be imported after content of project file automatically for current or child foldersDirectory.Build.rsp
: default options for msbuild cli to be passed automatically on each build# inside rsp file -v:diag -t:Clean
1
2Directory.Packages.props
: for central package management, fixed version by<PackageVersion>
xml<PropertyGroup> <ManagePackageVersionsCentrally>true</ManagePackageVersionsCentrally> </PropertyGroup> <ItemGroup> <PackageVersion Include="Newtonsoft.Json" Version="13.0.1" /> </ItemGroup>
1
2
3
4
5
6
All these kind of config are special config files that would be auto-imported by msbuild at certain phase. MSBuild would search upward for each aforementioned config file until one was found, it doesn't stop on solution root until it reaches the drive root
<!-- Directory.Build.props at solution root -->
<Project>
<PropertyGroup>
<!-- MSBuildProjectName varies for each project -->
<!-- so it's not static even as a property -->
<OutDir>C:\output\$(MSBuildProjectName)</OutDir>
</PropertyGroup>
</Project>
2
3
4
5
6
7
8
Best Practice
Directory.Build.props
: imported before standard musbuild config- to set independent properties
- to set conditional items
Directory.Build.targets
: imported after standard- to override properties
- to set dependent properties
- to set targets
$(MSBuildProjectFullPath).user
: extra config specific to local machine, should not be included in source control
TIP
- use
dotnet new buildprops
to createDirectory.Build.props
- use
dotnet new buildtargets
to createDirectory.Build.targets
- use
dotnet new packagesprops
to createDirectory.Packages.props
MSBuild CLI
MSBuild cli was wrapped as dotnet msbuild [-options]
within dotnet
cli.
-p:<prop>=<value>
: override a property value(namely global property)-t:<target>[;..]
: trigger a target(and its dependent hooks)-r
:Restore
before building
NOTE
dotnet
cli essentially wrapped msbuild
cli and has some shorthand for common usage of msbuild
. All build and restore related command from dotnet
are wrapper of a msbuild
command. Such targets like Build
are pre-defined targets.
# they're equivalent
dotnet build
dotnet msbuild -t:Build
dotnet msbuild # Build would be the default target if no other targets were specified
dotnet pack
dotnet msbuild -t:Pack
dotnet restore -p:Foo=foo
dotnet msbuild -t:Restore -p:Foo=foo #
2
3
4
5
6
7
8
9
10
Target Execution Order
Targets would be executed by the order they were specified(each target triggers and waits its dependent targets)
dotnet msbuild -t:Clean,Build # Clean first then Build
NOTE