You can control many aspects of the generated container through MSBuild properties. In general, if you could use a command in a Dockerfile to set some configuration, you can do the same via MSBuild.
Note The only exception to this is
RUN
commands - due to the way we build containers, those cannot be emulated. If you need this functionality, you will need to use a Dockerfile to build your container images.
This property controls the image used as the basis for your image. By default, we will infer the following values for you based on the properties of your project:
- if your project is self-contained, we use the
mcr.microsoft.com/dotnet/runtime-deps
image as the base image - if your project is an ASP.NET Core project, we use the
mcr.microsoft.com/dotnet/aspnet
image as the base image - otherwise we use the
mcr.microsoft.com/dotnet/runtime
image as the base image
We infer the tag of the image to be the numeric component of your chosen TargetFramework
- so a .net6.0
project will use the 6.0
tag of the inferred base image, a .net7.0-linux
project will use the 7.0
tag, and so on.
If you set a value here, you should set the fully-qualified name of the image to use as the base, including any tag you prefer:
<PropertyGroup>
<ContainerBaseImage>mcr.microsoft.com/dotnet/runtime:6.0</ContainerBaseImage>
<PropertyGroup>
This property controls the OS and platform used by your container if your ContainerBaseImage
is a 'Manifest List'. Manifest Lists are images that support more than one architecture behind a single, common, name. For example, the mcr.microsoft.com/dotnet/runtime
image is a manifest list that supports linux-x64
, linux-arm
, linux-arm64
and win10-x64
images.
When a Manifest List is your base image, we need to choose the most relevant image to use as the base. We do this by choosing the image that best matches the RuntimeIdentifier
of your project. If you set a value here, we will use that value to choose the best image to use as the base. Valid values for this property will vary based on the image you choose, but will always be in the form of a .NET SDK Runtime Identifier.
By default, if your project has a RuntimeIdentifier set, that value will be used. A RuntimeIdentifer is usually set via the -r
parameter to the dotnet publish
command, or by setting the RuntimeIdentifier
property in a PublishProfile used from Visual Studio.
<PropertyGroup>
<ContainerRuntimeIdentifier>linux-x64</ContainerRuntimeIdentifier>
</PropertyGroup>
Note If you'd like to publish to a musl-based OS like alpine (as opposed to a libc-based OS), you will need to specify the base image including architecture, instead of relying on any of the inference described above. For example, a
net7.0
-targeting application that wanted to run on alpine with the x64 architecture would use the7.0-alpine-amd64
tag of themcr.microsoft.com/dotnet/runtime
image (or another base image as appropriate for your project type):<PropertyGroup> <ContainerBaseImage>mcr.microsoft.com/dotnet/runtime:7.0-alpine-amd64</ContainerBaseImage> </PropertyGroup>
This property controls the destination registry - the place that the newly-created image will be pushed to.
Be default, we push to the local Docker daemon (annotated by docker://
), but you can also specify a remote registry. Interacting with that registry may require authentication, see Authenticating to container registries for more details.
<PropertyGroup>
<ContainerRegistry>registry.mycorp.com:1234</ContainerRegistry>
</PropertyGroup>
This property controls the name of the image itself, e.g dotnet/runtime
or my-awesome-app
.
By default, the value used will be the AssemblyName
of the project.
<PropertyGroup>
<ContainerImageName>my-super-awesome-app</ContainerImageName>
</PropertyGroup>
Note Image names consist of one or more slash-delimited segments, each of which can only contain lowercase alphanumeric characters, periods, underscores, and dashes, and must start with a letter or number - any other characters will result in an error being thrown.
This property controls the tag that is generated for the image. Tags are often used to refer to different versions of an application, but they can also refer to different operating system distributions, or even just different baked-in configuration. This property also can be used to push multiple tags - simply use a semicolon-delimited set of tags in the ContainerImageTags
property, similar to setting multiple TargetFrameworks
.
By default, the value used will be the Version
of the project.
<PropertyGroup>
<ContainerImageTag>1.2.3-alpha2</ContainerImageTag>
</PropertyGroup>
<PropertyGroup>
<ContainerImageTags>1.2.3-alpha2;latest</ContainerImageTags>
</PropertyGroup>
Note Tags can only contain up to 127 alphanumeric characters, periods, underscores, and dashes. They must start with an alphanumeric character or an underscore. Any other form will result in an error being thrown.
This property controls the working directory of the container - the directory that commands are executed within if not other command is run.
By default, we use the /app
directory as the working directory.
<PropertyGroup>
<ContainerWorkingDirectory>/bin</ContainerWorkingDirectory>
</PropertyGroup>
This item adds TCP or UDP ports to the list of known ports for the container. This enables container runtimes like Docker to map these ports to the host machine automatically. This is often used as documentation for the container, but can also be used to enable automatic port mapping.
ContainerPort items have two properties:
- Include
- The port number to expose
- Type
- One of
tcp
orudp
- the default istcp
- One of
<ItemGroup>
<ContainerPort Include="80" Type="tcp" />
</ItemGroup>
Note This item does nothing for the container by default and should be considered advisory at best.
This item adds a metadata label to the container. Labels have no impact on the container at runtime, but are often used to store version and authoring metadata for use by security scanners and other infrastructure tools.
ContainerLabel items have two properties:
- Include
- The key of the label
- Value
- The value of the label - this may be empty
See default container labels for a list of labels that are created by default.
<ItemGroup>
<ContainerLabel Include="org.contoso.businessunit" Value="contoso-university" />
</ItemGroup>
This item adds a new environment variable to the container. Environment variables will be accessible to the application running in the container immediately, and are often used to change the runtime behavior of the running application.
ContainerEnvironmentVariable items have two properties:
- Include
- The name of the environment variable
- Value
- The value of the environment variable
<ItemGroup>
<ContainerEnvironmentVariable Include="LOGGER_VERBOSITY" Value="Trace" />
</ItemGroup>
This item can be used to customize the entrypoint of the container - the binary that is run by default when the container is started.
By default, for builds that create an executable binary that binary is set as the ContainerEntrypoint. For builds that do not create an executable binary dotnet path/to/application.dll
is used as the ContainerEntrypoint.
ContainerEntrypoint items have one property:
- Include
- The command, option, or argument to use in the entrypoint command
<ItemGroup Label="Entrypoint Assignment">
<!-- This is how you would start the dotnet ef tool in your container -->
<ContainerEntrypoint Include="dotnet" />
<ContainerEntrypoint Include="ef" />
<!-- This shorthand syntax means the same thing - note the semicolon separating the tokens. -->
<ContainerEntrypoint Include="dotnet;ef" />
</ItemGroup>
This item controls the default arguments provided to the ContainerEntrypoint
. This should be used when the ContainerEntrypoint is a program that the user might want to use on its own.
By default, no ContainerEntrypointArgs are created on your behalf.
ContainerEntrypointArg items have one property:
- Include
- The option or argument to apply to the ContainerEntrypoint command
<ItemGroup>
<!-- Assuming the ContainerEntrypoint defined above, this would be the way to update the database by default, but let the user run a different EF command. -->
<ContainerEntrypointArgs Include="database" />
<ContainerEntrypointArgs Include="update" />
<!-- This is the shorthand syntax for the same idea -->
<ContainerEntrypointArgs Include="database;update" />
</ItemGroup>
Labels are often used to provide consistent metadata on container images. This package provides some default labels to encourage better maintainability of the generated images, drawn from the set defined as part of the OCI Image specification. Where possible, we use the values of common NuGet Project Properties as defaults for these annotations, though we also provide more specific properties for each of these labels.
Annotation | Default Value | Dedicated Property Name | Fallback Property Name | Enabled Property Name | Notes |
---|---|---|---|---|---|
org.opencontainers.image.created and org.opencontainers.artifact.created |
the RFC 3339 format of the current UTC DateTime | ContainerGenerateLabelsImageCreated |
|||
org.opencontainers.artifact.description and org.opencontainers.image.description |
ContainerDescription |
Description |
ContainerGenerateLabelsImageDescription |
||
org.opencontainers.image.authors |
ContainerAuthors |
Authors |
ContainerGenerateLabelsImageAuthors |
||
org.opencontainers.image.url |
ContainerInformationUrl |
PackageProjectUrl |
ContainerGenerateLabelsImageUrl |
||
org.opencontainers.image.documentation |
ContainerDocumentationUrl |
PackageProjectUrl |
ContainerGenerateLabelsImageDocumentation |
||
org.opencontainers.image.version |
ContainerVersion |
PackageVersion |
ContainerGenerateLabelsImageVersion |
||
org.opencontainers.image.vendor |
ContainerVendor |
ContainerGenerateLabelsImageVendor |
|||
org.opencontainers.image.licenses |
ContainerLicenseExpression |
PackageLicenseExpression |
ContainerGenerateLabelsImageLicenses |
||
org.opencontainers.image.title |
ContainerTitle |
Title |
ContainerGenerateLabelsImageTitle |
||
org.opencontainers.image.base.name |
ContainerBaseImage |
ContainerGenerateLabelsImageBaseName |
|||
org.opencontainers.image.source |
PrivateRepositoryUrl |
ContainerGenerateLabelsImageSource |
Only written if PublishRepositoryUrl is true . Also relies on Sourcelink infrastructure being part of the build. |
||
org.opencontainers.image.revision |
SourceRevisionId |
ContainerGenerateLabelsImageRevision |
Only written if PublishRepositoryUrl is true . Also relies on Sourcelink infrastructure being part of the build. |
Note You can disable all label generation by setting
ContainerGenerateLabels
tofalse
in your project file.