Container registry configuration
Aspire 13.1 introduced explicit container registry configuration, giving developers control over where and when container images are pushed during deployment. This article explains how to configure container registries for your Aspire applications.
Container registry configuration
When deploying Aspire applications to production environments, your containerized services need to be pushed to a container registry. Prior to Aspire 13.1, registry configuration was often implicit, making it difficult to control and understand the deployment process. The new ContainerRegistryResource provides explicit configuration for:
- Generic container registries — DockerHub, GitHub Container Registry (GHCR), Harbor, or any Docker-compatible registry
- Azure Container Registry — First-class support with automatic credential management
- Pipeline integration — Control when images are built and pushed using
aspire do push - Authentication — Configure registry credentials securely
Generic container registries
Caution
This API is experimental and may change in future releases. Use diagnostic code ASPIRECOMPUTE003 to suppress the experimental warning. For more information, see ASPIRECOMPUTE003.
Use the AddContainerRegistry method to configure a generic container registry for your application. This works with any Docker-compatible registry including DockerHub, GitHub Container Registry, Harbor, and private registries.
Basic usage
The following example configures a container registry and associates it with a project resource:
C#
var builder = DistributedApplication.CreateBuilder(args);
// Add a container registry
var registry = builder.AddContainerRegistry("myregistry", "registry.example.com");
// Associate the registry with a project
var api = builder.AddProject<Projects.Api>("api")
.WithContainerRegistry(registry);
builder.Build().Run();
TypeScript
const builder = await createBuilder();
// Add a container registry
const registry = await builder.addContainerRegistry("myregistry", "registry.example.com");
// Associate the registry with a project
const api = await builder.addProject("api", "../Api/Api.csproj");
await api.withContainerRegistry(registry);
await builder.build().run();
The preceding code:
- Creates a container registry resource pointing to
registry.example.com. - Associates the registry with the
apiproject. - When deploying, the
apiproject will be built as a container image and pushed to the specified registry.
DockerHub example
To push images to DockerHub, specify docker.io as the registry endpoint:
C#
var builder = DistributedApplication.CreateBuilder(args);
var registry = builder.AddContainerRegistry("dockerhub", "docker.io");
var api = builder.AddProject<Projects.Api>("api")
.WithContainerRegistry(registry);
TypeScript
const builder = await createBuilder();
const registry = await builder.addContainerRegistry("dockerhub", "docker.io");
const api = await builder.addProject("api", "../Api/Api.csproj");
await api.withContainerRegistry(registry);
GitHub Container Registry example
To push images to GitHub Container Registry (GHCR):
C#
var builder = DistributedApplication.CreateBuilder(args);
var registry = builder.AddContainerRegistry("ghcr", "ghcr.io");
var api = builder.AddProject<Projects.Api>("api")
.WithContainerRegistry(registry);
TypeScript
const builder = await createBuilder();
const registry = await builder.addContainerRegistry("ghcr", "ghcr.io");
const api = await builder.addProject("api", "../Api/Api.csproj");
await api.withContainerRegistry(registry);
Private registry example
For private registries, provide the full registry URL:
C#
var builder = DistributedApplication.CreateBuilder(args);
var registry = builder.AddContainerRegistry(
"private-registry",
"registry.mycompany.com:5000");
var api = builder.AddProject<Projects.Api>("api")
.WithContainerRegistry(registry);
TypeScript
const builder = await createBuilder();
const registry = await builder.addContainerRegistry(
"private-registry",
"registry.mycompany.com:5000");
const api = await builder.addProject("api", "../Api/Api.csproj");
await api.withContainerRegistry(registry);
Authentication and credentials
Container registries typically require authentication for pushing images. You can configure credentials using parameters and secrets.
Using parameters for registry credentials
Parameters allow you to provide registry configuration dynamically:
C#
var builder = DistributedApplication.CreateBuilder(args);
var registryEndpoint = builder.AddParameter("registry-endpoint");
var registryRepository = builder.AddParameter("registry-repository");
var registry = builder.AddContainerRegistry(
"myregistry",
registryEndpoint,
registryRepository);
var api = builder.AddProject<Projects.Api>("api")
.WithContainerRegistry(registry);
TypeScript
const builder = await createBuilder();
const registryEndpoint = await builder.addParameter("registry-endpoint");
const registryRepository = await builder.addParameter("registry-repository");
const registry = await builder.addContainerRegistry(
"myregistry",
registryEndpoint,
registryRepository);
const api = await builder.addProject("api", "../Api/Api.csproj");
await api.withContainerRegistry(registry);
Configuring credentials
Registry credentials should be configured through your deployment environment:
Docker login
Before pushing images, ensure you're authenticated with the registry:
docker login registry.example.com
For DockerHub:
docker login docker.io -u username
For GitHub Container Registry:
echo $GITHUB_TOKEN | docker login ghcr.io -u USERNAME --password-stdin
CI/CD configuration
In CI/CD environments (GitHub Actions, Azure Pipelines, and so on), configure credentials using secrets:
- name: Login to GitHub Container Registry
uses: docker/login-action@v3
with:
registry: ghcr.io
username: ${{ github.actor }}
password: ${{ secrets.GITHUB_TOKEN }}
- name: Push images
run: aspire do push
Azure Container Registry
Azure Container Registry (ACR) provides first-class integration with Aspire, with automatic credential management and parallel provisioning.
Explicit ACR configuration
Aspire 13.1 introduces explicit container registry configuration for Azure Container Apps environments:
C#
var builder = DistributedApplication.CreateBuilder(args);
var environment = builder.AddAzureContainerAppEnvironment("myenv");
var api = builder.AddProject<Projects.Api>("api")
.WithContainerRegistry(environment);
builder.Build().Run();
TypeScript
const builder = await createBuilder();
const environment = await builder.addAzureContainerAppEnvironment("myenv");
const api = await builder.addProject("api", "../Api/Api.csproj");
await api.withContainerRegistry(environment);
await builder.build().run();
In the preceding example:
- The code creates an Azure Container Apps environment with an associated ACR.
- The ACR is provisioned in parallel with that environment.
- Images are pushed as soon as the registry is available.
- Credentials are automatically managed through Azure authentication.
Note
Prior to Aspire 13.1, ACR was provisioned implicitly as part of the Container Apps environment. The explicit configuration provides better control and visibility into the deployment process.
Using an existing ACR
To use an existing Azure Container Registry, call the PublishAsExisting method when you add the ACR:
C#
var builder = DistributedApplication.CreateBuilder(args);
var registryName = builder.AddParameter("registryName");
var rgName = builder.AddParameter("rgName");
var acr = builder.AddAzureContainerRegistry("my-acr")
.PublishAsExisting(registryName, rgName);
builder.AddAzureContainerAppEnvironment("env")
.WithAzureContainerRegistry(acr);
var api = builder.AddProject<Projects.Api>("api")
.WithContainerRegistry(acr);
TypeScript
const builder = await createBuilder();
const registryName = await builder.addParameter("registryName");
const rgName = await builder.addParameter("rgName");
const acr = await builder.addAzureContainerRegistry("my-acr");
await acr.publishAsExisting(registryName, rgName);
const environment = await builder.addAzureContainerAppEnvironment("env");
await environment.withAzureContainerRegistry(acr);
const api = await builder.addProject("api", "../Api/Api.csproj");
await api.withContainerRegistry(acr);
Pipeline integration
Aspire's deployment pipeline includes a dedicated push step for pushing container images to registries.
Using aspire do push
The aspire do push command builds container images and pushes them to configured registries:
aspire do push
This command:
- 1
Builds all container images for compute resources
- 2
Tags images with the appropriate registry and repository names
- 3
Pushes images to their configured registries
Example output:
16:03:38 (pipeline-execution) → Starting pipeline-execution...
16:03:38 (build-api) → Starting build-api...
16:03:43 (push-api) → Starting push-api...
16:03:43 (push-api) → Pushing api to container-registry
16:03:44 (push-api) i [INF] Docker tag for api -> docker.io/username/api:latest succeeded.
16:04:05 (push-api) i [INF] Docker push for docker.io/username/api:latest succeeded.
16:04:05 (push-api) ✓ Successfully pushed api to docker.io/username/api:latest (21.3s)
16:04:05 (push-api) ✓ push-api completed successfully
Pipeline step dependencies
The push step automatically handles dependencies:
build-prereq— Ensures prerequisites are met before buildingbuild-<resource>— Builds container images for each resourcepush-<resource>— Pushes images to registries
You can execute individual steps or the entire pipeline:
aspire do build
aspire do deploy
The deploy step includes building, pushing, and deploying all resources.
Complete examples
Multi-registry deployment
You can configure different registries for different resources:
C#
var builder = DistributedApplication.CreateBuilder(args);
var publicRegistry = builder.AddContainerRegistry("dockerhub", "docker.io");
var privateRegistry = builder.AddContainerRegistry(
"private",
"registry.company.com");
var publicApi = builder.AddProject<Projects.PublicApi>("public-api")
.WithContainerRegistry(publicRegistry);
var internalApi = builder.AddProject<Projects.InternalApi>("internal-api")
.WithContainerRegistry(privateRegistry);
TypeScript
const builder = await createBuilder();
const publicRegistry = await builder.addContainerRegistry("dockerhub", "docker.io");
const privateRegistry = await builder.addContainerRegistry(
"private",
"registry.company.com");
const publicApi = await builder.addProject("public-api", "../PublicApi/PublicApi.csproj");
await publicApi.withContainerRegistry(publicRegistry);
const internalApi = await builder.addProject("internal-api", "../InternalApi/InternalApi.csproj");
await internalApi.withContainerRegistry(privateRegistry);
Parameterized registry configuration
You can use parameters when you need flexible deployment across environments:
C#
var builder = DistributedApplication.CreateBuilder(args);
var registryEndpoint = builder.AddParameter("registry-endpoint");
var registryRepository = builder.AddParameter("registry-repository");
var registry = builder.AddContainerRegistry(
"container-registry",
registryEndpoint,
registryRepository);
var api = builder.AddProject<Projects.Api>("api")
.WithContainerRegistry(registry);
var worker = builder.AddProject<Projects.Worker>("worker")
.WithContainerRegistry(registry);
builder.Build().Run();
TypeScript
const builder = await createBuilder();
const registryEndpoint = await builder.addParameter("registry-endpoint");
const registryRepository = await builder.addParameter("registry-repository");
const registry = await builder.addContainerRegistry(
"container-registry",
registryEndpoint,
registryRepository);
const api = await builder.addProject("api", "../Api/Api.csproj");
await api.withContainerRegistry(registry);
const worker = await builder.addProject("worker", "../Worker/Worker.csproj");
await worker.withContainerRegistry(registry);
await builder.build().run();
Configure the parameters in your AppHost configuration:
{
"Parameters": {
"registry-endpoint": "ghcr.io",
"registry-repository": "myorg"
}
}
Alternatively, use environment variables to configure them:
export Parameters__registry_endpoint="ghcr.io"
export Parameters__registry_repository="myorg"
Azure deployment with ACR
The following code constitutes a complete AppHost example with Azure Container Apps and ACR:
C#
var builder = DistributedApplication.CreateBuilder(args);
// Create Azure Container Apps environment with ACR
var acaEnv = builder.AddAzureContainerAppEnvironment("production");
// Add Redis cache
var cache = builder.AddRedis("cache");
// Add API with registry configuration
var api = builder.AddProject<Projects.Api>("api")
.WithReference(cache)
.WithContainerRegistry(acaEnv);
// Add frontend with registry configuration
var web = builder.AddProject<Projects.Web>("web")
.WithReference(api)
.WithContainerRegistry(acaEnv);
builder.Build().Run();
TypeScript
const builder = await createBuilder();
// Create Azure Container Apps environment with ACR
const environment = await builder.addAzureContainerAppEnvironment("production");
// Add Redis cache
const cache = await builder.addRedis("cache");
// Add API with registry configuration
const api = await builder.addProject("api", "../Api/Api.csproj");
await api.withReference(cache);
await api.withContainerRegistry(environment);
// Add frontend with registry configuration
const web = await builder.addProject("web", "../Web/Web.csproj");
await web.withReference(api);
await web.withContainerRegistry(environment);
await builder.build().run();
Benefits of explicit configuration
The explicit container registry configuration introduced in Aspire 13.1 provides several benefits:
- Visibility — Clear understanding of where images are pushed
- Control — Explicit configuration of registry endpoints and credentials
- Parallelization — Registry provisioning happens in parallel with other resources
- Early feedback — Faster deployments with images pushing as soon as registries are ready
- Flexibility — Support for any Docker-compatible registry