MSBuild ArtifactsPath
Setting up an out-of-source build with MSBuild and the .NET SDK
A really cool but not well-known feature of the .NET SDK is the ArtifactsPath
property. It allows you to specify a single directory into which all build artifacts from all projects will be placed: Binaries, Intermediates, NuGet packages, etc.
This is especially useful in CI/CD environments, where you might want to keep all build artifacts in a dedicated temporary directory to make use of your CI runners built-in cleanup, caching, and publishing features.
It is also a nice feature for local development, as you can keep your project directories clean, remove local artifacts from all projects at once and easily find local build outputs while maintaining a short .gitignore
.
Usage
All you need to do to place all created artifacts into subfolders of ./artifacts
is to set the ArtifactsPath
MSBuild property:
<!-- Directory.Build.props -->
<Project>
<PropertyGroup>
<ArtifactsPath>$(MSBuildThisFileDirectory)artifacts</ArtifactsPath>
</PropertyGroup>
</Project>
It is a good idea to set this property in a Directory.Build.props
file in the root of your repository, so that it automatically applies to all projects when building locally.
In a CI/CD environment, you can then override this default with a command line argument. Example for an Azure DevOps pipeline:
- task: DotNetCoreCLI@2
inputs:
command: 'build'
projects: 'Solution.sln'
arguments: '--property:ArtifactsPath=$(Build.ArtifactStagingDirectory)\artifacts'
Or a simple command line call:
dotnet build Solution.sln --property:ArtifactsPath=artifacts
Default Paths
Setting a custom ArtifactsPath
sets off an override cascade by the .NET SDK regarding output paths:
$(OutputPath)
- Folder where the binaries for a project will be placed. Aka the
bin
directory.
Default:$(ArtifactsPath)\bin\$(MSBuildProjectName)\$(ArtifactsPivots)\
$(IntermediateOutputPath)
- Folder used to place intermediate artifacts, such as temporary
.props
generated by the restore process or the compilers direct outputs. Aka theobj
directory.
Default:$(ArtifactsPath)\obj\$(MSBuildProjectName)\$(ArtifactsPivots)\
$(PublishDir)
- Folder where a published artifact will be placed, generated by
dotnet publish
.
Default:$(ArtifactsPath)\publish\$(MSBuildProjectName)\$(ArtifactsPivots)\
$(PackageOutputPath)
- Folder where
.nupkg
files will be placed, generated bydotnet pack
.
Default:$(ArtifactsPath)\package\$(Configuration)\
The $(ArtifactsPivots)
variable is a placeholder for projects that feature multiple build variants. At the minimum, it will contain the $(Configuration)
, while the _$(RuntimeIdentifier)
and _$(TargetFramework)
suffixes will be added if necessary (for example when TargetFramework
s
is configured).
There are many more knobs that you can turn, such as omitting the project name from the paths by setting the IncludeProjectNameInArtifactsPaths
property. Most of these features can be found in the Microsoft.Common.CurrentVersion.targets
file provided by the .NET SDK.
A simpler way of adjusting the paths is by setting them directly in the project file or the Directory.Build.props
file: The .NET SDK will respect any value that you set.
However, note that overriding these properties might lead to unexpected consequences, so I don’t recommend diverging from these defaults unless necessary. I especially recommend against setting a single OutputPath
for multiple projects, as this will lead to sporadic conflicts without careful management.
Example
Let’s look at a simple example involving a C# class library, console project, and XUnit test project:
# Setup ArtifactsPath
echo "<Project>
<PropertyGroup>
<ArtifactsPath>\$(MSBuildThisFileDirectory)artifacts</ArtifactsPath>
</PropertyGroup>
</Project>" >> Directory.Build.props
# Create Projects
dotnet new classlib --name Sample.Library --output Sample.Library
dotnet new xunit --name Sample.Tests --output Sample.Tests
dotnet new console --name Sample.App --output Sample.App
# Add References
dotnet add Sample.App reference Sample.Library
dotnet add Sample.Tests reference Sample.Library
# Create Solution
dotnet new sln --name Sample
dotnet sln add Sample.Library/Sample.Library.csproj
dotnet sln add Sample.Tests/Sample.Tests.csproj
dotnet sln add Sample.App/Sample.App.csproj
# Build Solution
dotnet build Sample.sln
The resulting directory structure will look like this (files omitted for brevity):
.
├── Sample.App/
├── Sample.Library/
├── Sample.Tests/
└── artifacts/
├── bin/
│ ├── Sample.App/
│ │ └── debug/
│ ├── Sample.Library/
│ │ └── debug/
│ └── Sample.Tests/
│ └── debug/
└── obj/
├── Sample.App/
│ └── debug/
│ ├── ref/
│ └── refint/
├── Sample.Library/
│ └── debug/
│ ├── ref/
│ └── refint/
└── Sample.Tests/
└── debug/
├── ref/
└── refint/
The structure would, of course, look a bit different if we built the solution in release mode, published the projects, created NuGet packages or similar, as explained in Default Paths.
Also note that this sample was created with the .NET SDK version 9.0.100, the behavior might differ in prior or future versions.
More Information
If you would like to learn more about the ArtifactsPath
property, you can consult the official documentation on the MS Docs. A more powerful, albeit more technical, way to discover your possibilities is by analyzing a binary log of your build using the Structured Log Viewer.