diff --git a/.github/prompts/adhere-to-writing-style.prompt.md b/.github/prompts/adhere-to-writing-style.prompt.md
new file mode 100644
index 0000000000..5140ca11f9
--- /dev/null
+++ b/.github/prompts/adhere-to-writing-style.prompt.md
@@ -0,0 +1,33 @@
+The following is a set of guidelines to follow when writing documentation for .NET Aspire. Please ensure that all documentation adheres to these guidelines:
+
+## Voice and tone
+
+Write for readers who may have a limited vocabulary or are not native English speakers. Use simple, clear language and avoid jargon whenever possible. If you must use technical terms, provide a brief explanation of what they mean.
+
+- Use active voice
+- Use the second person (you, your) to address the reader directly
+
+## Headings
+
+Headings use sentence-style capitalization. Always capitalize the first word of a heading. Do not use gerunds (e.g., "Using" or "Creating") in heading.
+
+## Text styling
+
+Use _italics_ for files, folders, paths (for long items, split onto their own line), new terms.
+
+Use **bold** for UI elements.
+
+Use `code` for inline code, language keywords, NuGet package names, command-line commands, database table and column names, and URLs that you don't want to be clickable.
+
+## Links
+
+Strive to use relative links whenever possible. Use absolute links only when necessary. For example, if you are linking to a page on a different site, use an absolute link. When using absolute links, use the full URL (including the protocol) and remove the locale from the URL. Avoid HTTP links, always prefer HTTPS. Whenever providing additional resources, use the following format:
+
+"For more information, see [link text](../relative/link/to/content.md)."
+
+Never link to `https://learn.microsoft.com` or `https://learn.microsoft.com/en-us`, always remove these and instead use site relative links. For example, if the original link is `https://learn.microsoft.com/en-us/aspnet/core/mvc/overview`, it should be changed to `/aspnet/core/mvc/overview`.
+
+## Things to avoid
+
+- Avoid future tense: In some non-English languages the concept of future tense is not the same as English. Using future tense can make your documents harder to read.
+- Avoid passive voice: Passive voice can make your writing less clear and harder to read. Use active voice whenever possible.
diff --git a/.github/prompts/rephrase-selected-text.prompt.md b/.github/prompts/rephrase-selected-text.prompt.md
new file mode 100644
index 0000000000..bf609e6146
--- /dev/null
+++ b/.github/prompts/rephrase-selected-text.prompt.md
@@ -0,0 +1,3 @@
+Please rephrase the selected text to make it more concise and clear, while maintaining the original meaning. The rephrased text should strive to improve readability. Please ensure that the rephrased text is grammatically correct and flows well. Take obvious opportunities to improve structure, and emphasis of domain specific intent.
+
+Please also always [adhere to our writing style](adhere-to-writing-style.prompt.md).
diff --git a/.gitignore b/.gitignore
index 0b3c331187..acae18877d 100644
--- a/.gitignore
+++ b/.gitignore
@@ -34,6 +34,9 @@ _dependentPackages/
!/xml/System.IO.Log/
!/xml/System.Net.Cache/
+# Playwright dependencies
+**/*/playwright-deps
+
# Visual Studio Code
.vscode/*
!.vscode/extensions.json
diff --git a/docs/app-host/snippets/AspireApp/AspireApp.AppHost/AspireApp.AppHost.csproj b/docs/app-host/snippets/AspireApp/AspireApp.AppHost/AspireApp.AppHost.csproj
index e47acdcb24..db1507a32b 100644
--- a/docs/app-host/snippets/AspireApp/AspireApp.AppHost/AspireApp.AppHost.csproj
+++ b/docs/app-host/snippets/AspireApp/AspireApp.AppHost/AspireApp.AppHost.csproj
@@ -1,13 +1,12 @@
-
+
Exe
net9.0
enable
enable
- true
7b352f08-305b-4032-9a21-90deb02efc04
@@ -17,8 +16,8 @@
-
-
+
+
diff --git a/docs/app-host/snippets/AspireApp/AspireApp.ResourceAppHost/AspireApp.ResourceAppHost.csproj b/docs/app-host/snippets/AspireApp/AspireApp.ResourceAppHost/AspireApp.ResourceAppHost.csproj
index 5d2de61e73..45d5b30662 100644
--- a/docs/app-host/snippets/AspireApp/AspireApp.ResourceAppHost/AspireApp.ResourceAppHost.csproj
+++ b/docs/app-host/snippets/AspireApp/AspireApp.ResourceAppHost/AspireApp.ResourceAppHost.csproj
@@ -1,13 +1,12 @@
-
+
Exe
net9.0
enable
enable
- true
7b352f08-305b-4032-9a21-90deb02efc04
@@ -17,8 +16,8 @@
-
-
+
+
diff --git a/docs/app-host/snippets/AspireApp/AspireApp.ServiceDefaults/AspireApp.ServiceDefaults.csproj b/docs/app-host/snippets/AspireApp/AspireApp.ServiceDefaults/AspireApp.ServiceDefaults.csproj
index 73da5c0d31..763ede8394 100644
--- a/docs/app-host/snippets/AspireApp/AspireApp.ServiceDefaults/AspireApp.ServiceDefaults.csproj
+++ b/docs/app-host/snippets/AspireApp/AspireApp.ServiceDefaults/AspireApp.ServiceDefaults.csproj
@@ -11,7 +11,7 @@
-
+
diff --git a/docs/app-host/snippets/AspireApp/AspireApp.Web/AspireApp.Web.csproj b/docs/app-host/snippets/AspireApp/AspireApp.Web/AspireApp.Web.csproj
index ade0c23b90..041f727dc0 100644
--- a/docs/app-host/snippets/AspireApp/AspireApp.Web/AspireApp.Web.csproj
+++ b/docs/app-host/snippets/AspireApp/AspireApp.Web/AspireApp.Web.csproj
@@ -11,7 +11,7 @@
-
+
diff --git a/docs/authentication/snippets/AspireApp/AspireApp.AppHost/AspireApp.AppHost.csproj b/docs/authentication/snippets/AspireApp/AspireApp.AppHost/AspireApp.AppHost.csproj
index 0e70c4f95d..ecf2b62c33 100644
--- a/docs/authentication/snippets/AspireApp/AspireApp.AppHost/AspireApp.AppHost.csproj
+++ b/docs/authentication/snippets/AspireApp/AspireApp.AppHost/AspireApp.AppHost.csproj
@@ -1,13 +1,12 @@
-
+
Exe
net9.0
enable
enable
- true
10daccf5-26f1-4641-80b2-95f0357607cf
@@ -17,7 +16,7 @@
-
+
-
+
diff --git a/docs/azure/configure-aca-environments.md b/docs/azure/configure-aca-environments.md
new file mode 100644
index 0000000000..0fb9c3d7ee
--- /dev/null
+++ b/docs/azure/configure-aca-environments.md
@@ -0,0 +1,70 @@
+---
+title: Configure Azure Container Apps environments
+description: Learn how to configure Azure Container Apps environments in .NET Aspire.
+ms.topic: how-to
+ms.date: 04/09/2025
+---
+
+# Configure Azure Container Apps environments
+
+It's easy to [publish resources as Azure Container Apps (ACA)](integrations-overview.md#publish-as-azure-container-app) using any of the supported APIs:
+
+-
+-
+-
+
+These APIs automatically create a default ACA environment when you publish your app. While this default setup works well for most scenarios, you might need to customize the ACA environment to meet specific requirements. To achieve this, use the `AddAzureContainerAppEnvironment` method.
+
+The [.NET Aspire app host](../fundamentals/app-host-overview.md) simplifies infrastructure provisioning by generating code to create Azure resources for your applications. This approach enables you to model and configure deployment-related aspects directly in C#, reducing the need to rely on tools like Bicep. These aspects include configuring ACA environments, which provide a serverless platform for running containerized applications.
+
+By using the APIs (explained in [Infrastructure as code](integrations-overview.md#infrastructure-as-code)), you can configure and customize ACA environments along with related resources, such as container registries and file share volumes. Any available deployment setting can be configured. For more information on the available settings, see [Microsoft.App managedEnvironments](/azure/templates/microsoft.app/managedenvironments).
+
+This article guides you through the process of tailoring ACA environments for your .NET Aspire solutions.
+
+## Add an ACA environment
+
+
+
+The `AzureContainerAppEnvironmentResource` type models an ACA environment resource. When you call the `AddAzureContainerAppEnvironment` method, it creates an instance of this type (wrapped in the ).
+
+:::code language="csharp" source="snippets/aca/AspireAca.AppHost/AspireApp.AppHost/Program.cs":::
+
+By default, the calling this API to add an ACA environment generates the following provisioning Bicep module:
+
+:::code language="bicep" source="snippets/aca/AspireAca.AppHost/AspireApp.AppHost/aca-env.module.bicep":::
+
+This module configures:
+
+- A user-assigned managed identity for the ACA environment.
+- An Azure Container Registry (ACR) for the ACA environment.
+- A Log Analytics workspace for the ACA environment.
+- An Azure Container Apps environment.
+- The [.NET Aspire dashboard](../fundamentals/dashboard/overview.md) for the ACA environment.
+- A role assignment for the user principal ID to the ACA environment.
+- Various outputs for the ACA environment.
+
+Using the `acaEnv` variable, you can chain a call to the API to customize the ACA environment to your liking. For more information, see [Configure infrastructure](integrations-overview.md#configure-infrastructure).
+
+## Handle naming conventions
+
+
+
+By default, `AddAzureContainerAppEnvironment` uses a different Azure resource naming scheme than the [Azure Developer CLI (`azd`)](/azure/developer/azure-developer-cli/). If you're upgrading an existing deployment that previously used `azd`, you might see duplicate Azure resources. To avoid this issue, call the `WithAzdResourceNaming` method to revert to the naming convention used by `azd`:
+
+```csharp
+var builder = DistributionApplicationBuilder.Create(args);
+
+var acaEnv = builder.AddAzureContainerAppEnvironment("aca-env")
+ .WithAzdResourceNaming();
+
+// Omitted for brevity...
+
+builder.Build().Run();
+```
+
+Calling this API ensures your existing Azure resources remain consistent and prevents duplication.
+
+## See also
+
+- [.NET Aspire Azure integrations overview](integrations-overview.md)
+- [Azure Container Apps overview](/azure/container-apps/overview)
diff --git a/docs/azure/integrations-overview.md b/docs/azure/integrations-overview.md
index d5499b51e3..85bfae3535 100644
--- a/docs/azure/integrations-overview.md
+++ b/docs/azure/integrations-overview.md
@@ -1,7 +1,7 @@
---
title: Azure integrations overview
description: Overview of the Azure integrations available in the .NET Aspire.
-ms.date: 03/07/2025
+ms.date: 04/10/2025
uid: dotnet/aspire/integrations/azure-overview
---
@@ -107,7 +107,7 @@ You can query whether a resource is marked as an existing resource, by calling t
.NET Aspire provides support for referencing existing Azure resources. You mark an existing resource through the `PublishAsExisting`, `RunAsExisting`, and `AsExisting` APIs. These APIs allow developers to reference already-deployed Azure resources, configure them, and generate appropriate deployment manifests using Bicep templates.
-Existing resources referenced with these APIs can be enhanced with role assignments and other customizations that are available with .NET Aspire's [infrastructure as code capabilities](#infrastructure-as-code). These APIs are limited to Azure resources that can be deployed with Bicep templates.
+Existing resources referenced with these APIs can be enhanced with [role assignments](role-assignments.md) and other customizations that are available with .NET Aspire's [infrastructure as code capabilities](#infrastructure-as-code). These APIs are limited to Azure resources that can be deployed with Bicep templates.
### Configure existing Azure resources for run mode
@@ -321,7 +321,7 @@ The preceding code:
- Adds an environment variable named `Hello` to the container app, using the `env` parameter.
- The `AsProvisioningParameter` method is used to treat `env` as either a new in infrastructure, or reuses an existing bicep parameter if one with the same name already exists.
-For more information, see and .
+To configure the Azure Container App environment, see [Configure Azure Container Apps environments](configure-aca-environments.md). For more information, see and .
## Infrastructure as code
@@ -462,11 +462,11 @@ By default, when you call any of the Bicep-related APIs, a call is also made to
Imagine that you have a Bicep template in a file named `storage.bicep` that provisions an Azure Storage Account:
-:::code language="bicep" source="snippets/AppHost.Bicep/storage.bicep":::
+:::code language="bicep" source="snippets/bicep/AppHost.Bicep/storage.bicep":::
To add a reference to the Bicep file on disk, call the method. Consider the following example:
-:::code language="csharp" source="snippets/AppHost.Bicep/Program.ReferenceBicep.cs" id="addfile":::
+:::code language="csharp" source="snippets/bicep/AppHost.Bicep/Program.ReferenceBicep.cs" id="addfile":::
The preceding code adds a reference to a Bicep file located at `../infra/storage.bicep`. The file paths should be relative to the _app host_ project. This reference results in an being added to the application's resources collection with the `"storage"` name, and the API returns an `IResourceBuilder` instance that can be used to further customize the resource.
@@ -474,7 +474,7 @@ The preceding code adds a reference to a Bicep file located at `../infra/storage
While having a Bicep file on disk is the most common scenario, you can also add Bicep templates inline. Inline templates can be useful when you want to define a template in code or when you want to generate the template dynamically. To add an inline Bicep template, call the method with the Bicep template as a `string`. Consider the following example:
-:::code language="csharp" source="snippets/AppHost.Bicep/Program.InlineBicep.cs" id="addinline":::
+:::code language="csharp" source="snippets/bicep/AppHost.Bicep/Program.InlineBicep.cs" id="addinline":::
In this example, the Bicep template is defined as an inline `string` and added to the application's resources collection with the name `"ai"`. This example provisions an Azure AI resource.
@@ -482,7 +482,7 @@ In this example, the Bicep template is defined as an inline `string` and added t
[Bicep supports accepting parameters](/azure/azure-resource-manager/bicep/parameters), which can be used to customize the behavior of the template. To pass parameters to a Bicep template from .NET Aspire, chain calls to the method as shown in the following example:
-:::code language="csharp" source="snippets/AppHost.Bicep/Program.PassParameter.cs" id="addparameter":::
+:::code language="csharp" source="snippets/bicep/AppHost.Bicep/Program.PassParameter.cs" id="addparameter":::
The preceding code:
@@ -511,11 +511,11 @@ To use a well-known parameter, pass the parameter name to the method on an `IResourceBuilder` instance as demonstrated in following C# code snippet:
-:::code language="csharp" source="snippets/AppHost.Bicep/Program.GetOutputReference.cs" id="getoutput":::
+:::code language="csharp" source="snippets/bicep/AppHost.Bicep/Program.GetOutputReference.cs" id="getoutput":::
In this example, the output from the Bicep template is retrieved and stored in an `endpoint` variable. Typically, you would pass this output as an environment variable to another resource that relies on it. For instance, if you had an ASP.NET Core Minimal API project that depended on this endpoint, you could pass the output as an environment variable to the project using the following code snippet:
@@ -558,11 +558,11 @@ It's important to [avoid outputs for secrets](/azure/azure-resource-manager/bice
Consider the following Bicep template as an example the helps to demonstrate this concept of securing secret outputs:
-:::code language="bicep" source="snippets/AppHost.Bicep/cosmosdb.bicep" highlight="2,41":::
+:::code language="bicep" source="snippets/bicep/AppHost.Bicep/cosmosdb.bicep" highlight="2,41":::
The preceding Bicep template expects a `keyVaultName` parameter, among several other parameters. It then defines an Azure Cosmos DB resource and stashes a secret into Azure Key Vault, named `connectionString` which represents the fully qualified connection string to the Cosmos DB instance. To access this secret connection string value, you can use the following code snippet:
-:::code language="csharp" source="snippets/AppHost.Bicep/Program.cs" id="secrets":::
+:::code language="csharp" source="snippets/bicep/AppHost.Bicep/Program.cs" id="secrets":::
In the preceding code snippet, the `cosmos` Bicep template is added as a reference to the `builder`. The `connectionString` secret output is retrieved from the Bicep template and stored in a variable. The secret output is then passed as an environment variable (`ConnectionStrings__cosmos`) to the `api` project. This environment variable is used to connect to the Cosmos DB instance.
diff --git a/docs/azure/role-assignments.md b/docs/azure/role-assignments.md
new file mode 100644
index 0000000000..e336ab448e
--- /dev/null
+++ b/docs/azure/role-assignments.md
@@ -0,0 +1,261 @@
+---
+title: Manage Azure role assignments
+description: Learn how to override Azure role assignments on .NET Aspire resources.
+ms.date: 03/31/2025
+---
+
+# Manage Azure role assignments
+
+All .NET Aspire Azure hosting integrations define Azure resources. [These resources](integrations-overview.md#add-azure-resources) come with default role assignments. You can replace these default role assignments with built-in role [or custom role assignments](integrations-overview.md#infrastructure-as-code). In this article, you learn how to manage Azure role assignments on .NET Aspire resources.
+
+## Default built-in role assignments
+
+When you add an Azure resource to the [app model](xref:dotnet/aspire/app-host#terminology), it's assigned default roles. If a resource depends on another resource, it inherits the same role assignments as the referenced resource unless explicitly overridden.
+
+Consider a scenario where an API project resource references an [Azure Search](../azureai/azureai-search-document-integration.md) resource. The API project is given the default role assignments, as shown in the following example:
+
+```csharp
+var builder = DistributedApplication.CreateBuilder(args);
+
+var search = builder.AddAzureSearch("search");
+
+var api = builder.AddProject("api")
+ .WithReference(search);
+```
+
+In the example code, the `api` project resource depends on the Azure `search` resource, meaning it references the `search` resource. By default, the `search` resource is assigned the following built-in roles:
+
+-
+-
+
+These role assignments allow the API project to read and write data to the Azure Search resource, and manage it. However, this behavior might not always be desirable. For instance, you might want to restrict the API project to only read data from the Azure Search resource.
+
+## Override default role assignments
+
+
+
+To override the default role assignment, use the `WithRoleAssignments` API and assign built-in roles as shown in the following example:
+
+```csharp
+var builder = DistributedApplication.CreateBuilder(args);
+
+var search = builder.AddAzureSearch("search");
+
+var api = builder.AddProject("api")
+ .WithRoleAssignments(search, SearchBuiltInRole.SearchIndexDataReader)
+ .WithReference(search);
+```
+
+When you use the `WithRoleAssignments` method, it replaces the default role assignments with the specified ones. This method requires two parameters: the resource to which the role assignment applies and the built-in role to assign. In the preceding example, the `search` resource is assigned the role.
+
+When you replace the default role assignments with the `SearchIndexDataReader` role, the API project is restricted to only reading data from the Azure Search resource. This ensures the API project can't write data to the Azure Search resource.
+
+For more information, see [Azure built-in roles](/azure/role-based-access-control/built-in-roles).
+
+## Built-in role assignment reference
+
+All built-in roles are defined within the namespaces and are included in the corresponding [📦 Azure.Provisioning.*](https://www.nuget.org/packages?q=Azure.Provisioning) NuGet packages. Each .NET Aspire Azure hosting integration automatically depends on the appropriate provisioning package. For more information, see [Infrastructure as code](integrations-overview.md#infrastructure-as-code).
+
+The following sections list the built-in roles for each Azure provisioning type that can be used as a parameter to the `WithRoleAssignments` API.
+
+### Azure App Configuration
+
+The provisioning resource type is , and the built-in roles are defined in the struct. The built-in roles are:
+
+-
+-
+
+
+
+### Azure App Container
+
+The provisioning resource type is , and the built-in roles are defined in the struct. The built-in roles are:
+
+-
+-
+-
+
+
+
+### Azure Application Insights
+
+The provisioning resource type is , and the built-in roles are defined in the struct. The built-in roles are:
+
+-
+-
+-
+-
+-
+-
+-
+
+For more information, see [Use Application Insights for .NET Aspire telemetry](../deployment/azure/application-insights.md).
+
+### Azure AI (formerly Cognitive Services)
+
+The provisioning resource type is , and the built-in roles are defined in the struct. The built-in roles are:
+
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+
+For more information, see [.NET Aspire Azure OpenAI integration (Preview)](../azureai/azureai-openai-integration.md).
+
+### Azure Cosmos DB
+
+The provisioning resource type is , and the built-in roles are defined in the struct. The built-in roles are:
+
+-
+-
+-
+
+For more information, see:
+
+- [.NET Aspire Azure Cosmos DB integration](../database/azure-cosmos-db-integration.md).
+- [.NET Aspire Cosmos DB Entity Framework Core integration](../database/azure-cosmos-db-entity-framework-integration.md).
+
+### Azure Event Hubs
+
+The provisioning resource type is , and the built-in roles are defined in the struct. The built-in roles are:
+
+-
+-
+-
+-
+-
+
+For more information, see [.NET Aspire Azure Event Hubs integration](../messaging/azure-event-hubs-integration.md).
+
+### Azure Key Vault
+
+The provisioning resource type is , and the built-in roles are defined in the struct. The built-in roles are:
+
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+
+For more information, see [.NET Aspire Azure Key Vault integration](../security/azure-security-key-vault-integration.md).
+
+### Azure AI Search
+
+The provisioning resource type is , and the built-in roles are defined in the struct. The built-in roles are:
+
+-
+-
+-
+
+For more information, see [.NET Aspire Azure AI Search integration](../azureai/azureai-search-document-integration.md).
+
+### Azure Service Bus
+
+The provisioning resource type is , and the built-in roles are defined in the struct. The built-in roles are:
+
+-
+-
+-
+
+For more information, see [.NET Aspire Azure Service Bus integration](../messaging/azure-service-bus-integration.md).
+
+### Azure SignalR Service
+
+The provisioning resource type is , and the built-in roles are defined in the struct. The built-in roles are:
+
+-
+-
+-
+-
+-
+-
+
+For more information, see [.NET Aspire support for Azure SignalR Service](../real-time/azure-signalr-scenario.md).
+
+### Azure SQL
+
+The provisioning resource type is , and the built-in roles are defined in the struct. The built-in roles are:
+
+-
+-
+-
+-
+-
+
+
+
+### Azure Storage
+
+The provisioning resource type is , and the built-in roles are defined in the struct. The built-in roles are:
+
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+-
+
+For more information, see:
+
+- [.NET Aspire Azure Blob Storage integration](../storage/azure-storage-blobs-integration.md)
+- [.NET Aspire Azure Data Tables integration](../storage/azure-storage-tables-integration.md)
+- [.NET Aspire Azure Queue Storage integration](../storage/azure-storage-queues-integration.md)
+
+### Azure Web PubSub
+
+The provisioning resource type is , and the built-in roles are defined in the struct. The built-in roles are:
+
+-
+-
+-
+
+For more information, see [.NET Aspire Azure Web PubSub integration](../messaging/azure-web-pubsub-integration.md).
+
+## See also
+
+- [.NET Aspire Azure integrations overview](integrations-overview.md)
+- [Azure role-based access control (RBAC)](/azure/role-based-access-control/overview)
diff --git a/docs/azure/snippets/aca/AspireAca.AppHost/AspireAca.AppHost.sln b/docs/azure/snippets/aca/AspireAca.AppHost/AspireAca.AppHost.sln
new file mode 100644
index 0000000000..098595375a
--- /dev/null
+++ b/docs/azure/snippets/aca/AspireAca.AppHost/AspireAca.AppHost.sln
@@ -0,0 +1,24 @@
+Microsoft Visual Studio Solution File, Format Version 12.00
+# Visual Studio Version 17
+VisualStudioVersion = 17.14.35906.104
+MinimumVisualStudioVersion = 10.0.40219.1
+Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "AspireApp.AppHost", "AspireApp.AppHost\AspireApp.AppHost.csproj", "{A27E29D6-FA6E-404D-A41E-1D53EE2DF262}"
+EndProject
+Global
+ GlobalSection(SolutionConfigurationPlatforms) = preSolution
+ Debug|Any CPU = Debug|Any CPU
+ Release|Any CPU = Release|Any CPU
+ EndGlobalSection
+ GlobalSection(ProjectConfigurationPlatforms) = postSolution
+ {A27E29D6-FA6E-404D-A41E-1D53EE2DF262}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
+ {A27E29D6-FA6E-404D-A41E-1D53EE2DF262}.Debug|Any CPU.Build.0 = Debug|Any CPU
+ {A27E29D6-FA6E-404D-A41E-1D53EE2DF262}.Release|Any CPU.ActiveCfg = Release|Any CPU
+ {A27E29D6-FA6E-404D-A41E-1D53EE2DF262}.Release|Any CPU.Build.0 = Release|Any CPU
+ EndGlobalSection
+ GlobalSection(SolutionProperties) = preSolution
+ HideSolutionNode = FALSE
+ EndGlobalSection
+ GlobalSection(ExtensibilityGlobals) = postSolution
+ SolutionGuid = {D45E500B-8092-41D3-BEE7-55C647029894}
+ EndGlobalSection
+EndGlobal
diff --git a/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/AspireApp.AppHost.csproj b/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/AspireApp.AppHost.csproj
new file mode 100644
index 0000000000..1923ae506e
--- /dev/null
+++ b/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/AspireApp.AppHost.csproj
@@ -0,0 +1,19 @@
+
+
+
+
+
+ Exe
+ net9.0
+ enable
+ enable
+ true
+ 4a4b5271-16e5-49fd-a7a1-ff2849aefc25
+
+
+
+
+
+
+
+
diff --git a/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/Program.cs b/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/Program.cs
new file mode 100644
index 0000000000..b1e7d39690
--- /dev/null
+++ b/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/Program.cs
@@ -0,0 +1,7 @@
+var builder = DistributedApplication.CreateBuilder(args);
+
+var acaEnv = builder.AddAzureContainerAppEnvironment("aca-env");
+
+// Omitted for brevity...
+
+builder.Build().Run();
diff --git a/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/Properties/launchSettings.json b/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/Properties/launchSettings.json
new file mode 100644
index 0000000000..d1592b5639
--- /dev/null
+++ b/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/Properties/launchSettings.json
@@ -0,0 +1,35 @@
+{
+ "$schema": "https://json.schemastore.org/launchsettings.json",
+ "profiles": {
+ "https": {
+ "commandName": "Project",
+ "dotnetRunMessages": true,
+ "launchBrowser": true,
+ "applicationUrl": "https://localhost:17189;http://localhost:15026",
+ "environmentVariables": {
+ "ASPNETCORE_ENVIRONMENT": "Development",
+ "DOTNET_ENVIRONMENT": "Development",
+ "DOTNET_DASHBOARD_OTLP_ENDPOINT_URL": "https://localhost:21283",
+ "DOTNET_RESOURCE_SERVICE_ENDPOINT_URL": "https://localhost:22129"
+ }
+ },
+ "http": {
+ "commandName": "Project",
+ "dotnetRunMessages": true,
+ "launchBrowser": true,
+ "applicationUrl": "http://localhost:15026",
+ "environmentVariables": {
+ "ASPNETCORE_ENVIRONMENT": "Development",
+ "DOTNET_ENVIRONMENT": "Development",
+ "DOTNET_DASHBOARD_OTLP_ENDPOINT_URL": "http://localhost:19097",
+ "DOTNET_RESOURCE_SERVICE_ENDPOINT_URL": "http://localhost:20224"
+ }
+ },
+ "generate-manifest": {
+ "commandName": "Project",
+ "launchBrowser": false,
+ "dotnetRunMessages": true,
+ "commandLineArgs": "--publisher manifest --output-path aspire-manifest.json",
+ }
+ }
+}
diff --git a/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/aca-env.module.bicep b/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/aca-env.module.bicep
new file mode 100644
index 0000000000..5e12838131
--- /dev/null
+++ b/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/aca-env.module.bicep
@@ -0,0 +1,100 @@
+@description('The location for the resource(s) to be deployed.')
+param location string = resourceGroup().location
+
+param userPrincipalId string
+
+param tags object = { }
+
+resource aca_env_mi 'Microsoft.ManagedIdentity/userAssignedIdentities@2023-01-31' = {
+ name: take('aca_env_mi-${uniqueString(resourceGroup().id)}', 128)
+ location: location
+ tags: tags
+}
+
+resource aca_env_acr 'Microsoft.ContainerRegistry/registries@2023-07-01' = {
+ name: take('acaenvacr${uniqueString(resourceGroup().id)}', 50)
+ location: location
+ sku: {
+ name: 'Basic'
+ }
+ tags: tags
+}
+
+resource aca_env_acr_aca_env_mi_AcrPull 'Microsoft.Authorization/roleAssignments@2022-04-01' = {
+ name: guid(aca_env_acr.id, aca_env_mi.id, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '7f951dda-4ed3-4680-a7ca-43fe172d538d'))
+ properties: {
+ principalId: aca_env_mi.properties.principalId
+ roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '7f951dda-4ed3-4680-a7ca-43fe172d538d')
+ principalType: 'ServicePrincipal'
+ }
+ scope: aca_env_acr
+}
+
+resource aca_env_law 'Microsoft.OperationalInsights/workspaces@2023-09-01' = {
+ name: take('acaenvlaw-${uniqueString(resourceGroup().id)}', 63)
+ location: location
+ properties: {
+ sku: {
+ name: 'PerGB2018'
+ }
+ }
+ tags: tags
+}
+
+resource aca_env 'Microsoft.App/managedEnvironments@2024-03-01' = {
+ name: take('acaenv${uniqueString(resourceGroup().id)}', 24)
+ location: location
+ properties: {
+ appLogsConfiguration: {
+ destination: 'log-analytics'
+ logAnalyticsConfiguration: {
+ customerId: aca_env_law.properties.customerId
+ sharedKey: aca_env_law.listKeys().primarySharedKey
+ }
+ }
+ workloadProfiles: [
+ {
+ name: 'consumption'
+ workloadProfileType: 'Consumption'
+ }
+ ]
+ }
+ tags: tags
+}
+
+resource aspireDashboard 'Microsoft.App/managedEnvironments/dotNetComponents@2024-10-02-preview' = {
+ name: 'aspire-dashboard'
+ properties: {
+ componentType: 'AspireDashboard'
+ }
+ parent: aca_env
+}
+
+resource aca_env_Contributor 'Microsoft.Authorization/roleAssignments@2022-04-01' = {
+ name: guid(aca_env.id, userPrincipalId, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', 'b24988ac-6180-42a0-ab88-20f7382dd24c'))
+ properties: {
+ principalId: userPrincipalId
+ roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', 'b24988ac-6180-42a0-ab88-20f7382dd24c')
+ }
+ scope: aca_env
+}
+
+output MANAGED_IDENTITY_NAME string = aca_env_mi.name
+
+output MANAGED_IDENTITY_PRINCIPAL_ID string = aca_env_mi.properties.principalId
+
+output AZURE_LOG_ANALYTICS_WORKSPACE_NAME string = aca_env_law.name
+
+output AZURE_LOG_ANALYTICS_WORKSPACE_ID string = aca_env_law.id
+
+output AZURE_CONTAINER_REGISTRY_NAME string = aca_env_acr.name
+
+output AZURE_CONTAINER_REGISTRY_ENDPOINT string = aca_env_acr.properties.loginServer
+
+output AZURE_CONTAINER_REGISTRY_MANAGED_IDENTITY_ID string = aca_env_mi.id
+
+output AZURE_CONTAINER_APPS_ENVIRONMENT_NAME string = aca_env.name
+
+output AZURE_CONTAINER_APPS_ENVIRONMENT_ID string = aca_env.id
+
+output AZURE_CONTAINER_APPS_ENVIRONMENT_DEFAULT_DOMAIN string = aca_env.properties.defaultDomain
\ No newline at end of file
diff --git a/docs/azure/snippets/AppHost.Bicep/appsettings.Development.json b/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/appsettings.Development.json
similarity index 100%
rename from docs/azure/snippets/AppHost.Bicep/appsettings.Development.json
rename to docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/appsettings.Development.json
diff --git a/docs/azure/snippets/AppHost.Bicep/appsettings.json b/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/appsettings.json
similarity index 100%
rename from docs/azure/snippets/AppHost.Bicep/appsettings.json
rename to docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/appsettings.json
diff --git a/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/aspire-manifest.json b/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/aspire-manifest.json
new file mode 100644
index 0000000000..e97b92b5c7
--- /dev/null
+++ b/docs/azure/snippets/aca/AspireAca.AppHost/AspireApp.AppHost/aspire-manifest.json
@@ -0,0 +1,12 @@
+{
+ "$schema": "https://json.schemastore.org/aspire-8.0.json",
+ "resources": {
+ "aca-env": {
+ "type": "azure.bicep.v0",
+ "path": "aca-env.module.bicep",
+ "params": {
+ "userPrincipalId": ""
+ }
+ }
+ }
+}
\ No newline at end of file
diff --git a/docs/azure/snippets/AppHost.Bicep.sln b/docs/azure/snippets/bicep/AppHost.Bicep.sln
similarity index 100%
rename from docs/azure/snippets/AppHost.Bicep.sln
rename to docs/azure/snippets/bicep/AppHost.Bicep.sln
diff --git a/docs/azure/snippets/AppHost.Bicep/AppHost.Bicep.csproj b/docs/azure/snippets/bicep/AppHost.Bicep/AppHost.Bicep.csproj
similarity index 83%
rename from docs/azure/snippets/AppHost.Bicep/AppHost.Bicep.csproj
rename to docs/azure/snippets/bicep/AppHost.Bicep/AppHost.Bicep.csproj
index f6174aadd1..3f7ea1bcd0 100644
--- a/docs/azure/snippets/AppHost.Bicep/AppHost.Bicep.csproj
+++ b/docs/azure/snippets/bicep/AppHost.Bicep/AppHost.Bicep.csproj
@@ -1,16 +1,15 @@
-
+
Exe
net9.0
enable
enable
- true
5e45e8bd-353b-48d8-ac57-a89e5ee3f8ec
-
-
+
+
diff --git a/docs/azure/snippets/AppHost.Bicep/Program.GetOutputReference.cs b/docs/azure/snippets/bicep/AppHost.Bicep/Program.GetOutputReference.cs
similarity index 100%
rename from docs/azure/snippets/AppHost.Bicep/Program.GetOutputReference.cs
rename to docs/azure/snippets/bicep/AppHost.Bicep/Program.GetOutputReference.cs
diff --git a/docs/azure/snippets/AppHost.Bicep/Program.InlineBicep.cs b/docs/azure/snippets/bicep/AppHost.Bicep/Program.InlineBicep.cs
similarity index 100%
rename from docs/azure/snippets/AppHost.Bicep/Program.InlineBicep.cs
rename to docs/azure/snippets/bicep/AppHost.Bicep/Program.InlineBicep.cs
diff --git a/docs/azure/snippets/AppHost.Bicep/Program.PassParameter.cs b/docs/azure/snippets/bicep/AppHost.Bicep/Program.PassParameter.cs
similarity index 100%
rename from docs/azure/snippets/AppHost.Bicep/Program.PassParameter.cs
rename to docs/azure/snippets/bicep/AppHost.Bicep/Program.PassParameter.cs
diff --git a/docs/azure/snippets/AppHost.Bicep/Program.ReferenceBicep.cs b/docs/azure/snippets/bicep/AppHost.Bicep/Program.ReferenceBicep.cs
similarity index 100%
rename from docs/azure/snippets/AppHost.Bicep/Program.ReferenceBicep.cs
rename to docs/azure/snippets/bicep/AppHost.Bicep/Program.ReferenceBicep.cs
diff --git a/docs/azure/snippets/AppHost.Bicep/Program.cs b/docs/azure/snippets/bicep/AppHost.Bicep/Program.cs
similarity index 100%
rename from docs/azure/snippets/AppHost.Bicep/Program.cs
rename to docs/azure/snippets/bicep/AppHost.Bicep/Program.cs
diff --git a/docs/azure/snippets/AppHost.Bicep/Properties/launchSettings.json b/docs/azure/snippets/bicep/AppHost.Bicep/Properties/launchSettings.json
similarity index 100%
rename from docs/azure/snippets/AppHost.Bicep/Properties/launchSettings.json
rename to docs/azure/snippets/bicep/AppHost.Bicep/Properties/launchSettings.json
diff --git a/docs/azure/snippets/WebHook.Api/appsettings.Development.json b/docs/azure/snippets/bicep/AppHost.Bicep/appsettings.Development.json
similarity index 100%
rename from docs/azure/snippets/WebHook.Api/appsettings.Development.json
rename to docs/azure/snippets/bicep/AppHost.Bicep/appsettings.Development.json
diff --git a/docs/azure/snippets/bicep/AppHost.Bicep/appsettings.json b/docs/azure/snippets/bicep/AppHost.Bicep/appsettings.json
new file mode 100644
index 0000000000..31c092aa45
--- /dev/null
+++ b/docs/azure/snippets/bicep/AppHost.Bicep/appsettings.json
@@ -0,0 +1,9 @@
+{
+ "Logging": {
+ "LogLevel": {
+ "Default": "Information",
+ "Microsoft.AspNetCore": "Warning",
+ "Aspire.Hosting.Dcp": "Warning"
+ }
+ }
+}
diff --git a/docs/azure/snippets/AppHost.Bicep/cosmosdb.bicep b/docs/azure/snippets/bicep/AppHost.Bicep/cosmosdb.bicep
similarity index 100%
rename from docs/azure/snippets/AppHost.Bicep/cosmosdb.bicep
rename to docs/azure/snippets/bicep/AppHost.Bicep/cosmosdb.bicep
diff --git a/docs/azure/snippets/AppHost.Bicep/event-grid-webhook.bicep b/docs/azure/snippets/bicep/AppHost.Bicep/event-grid-webhook.bicep
similarity index 100%
rename from docs/azure/snippets/AppHost.Bicep/event-grid-webhook.bicep
rename to docs/azure/snippets/bicep/AppHost.Bicep/event-grid-webhook.bicep
diff --git a/docs/azure/snippets/AppHost.Bicep/storage-out.bicep b/docs/azure/snippets/bicep/AppHost.Bicep/storage-out.bicep
similarity index 100%
rename from docs/azure/snippets/AppHost.Bicep/storage-out.bicep
rename to docs/azure/snippets/bicep/AppHost.Bicep/storage-out.bicep
diff --git a/docs/azure/snippets/AppHost.Bicep/storage.bicep b/docs/azure/snippets/bicep/AppHost.Bicep/storage.bicep
similarity index 100%
rename from docs/azure/snippets/AppHost.Bicep/storage.bicep
rename to docs/azure/snippets/bicep/AppHost.Bicep/storage.bicep
diff --git a/docs/azure/snippets/WebHook.Api/Program.cs b/docs/azure/snippets/bicep/WebHook.Api/Program.cs
similarity index 100%
rename from docs/azure/snippets/WebHook.Api/Program.cs
rename to docs/azure/snippets/bicep/WebHook.Api/Program.cs
diff --git a/docs/azure/snippets/WebHook.Api/Properties/launchSettings.json b/docs/azure/snippets/bicep/WebHook.Api/Properties/launchSettings.json
similarity index 100%
rename from docs/azure/snippets/WebHook.Api/Properties/launchSettings.json
rename to docs/azure/snippets/bicep/WebHook.Api/Properties/launchSettings.json
diff --git a/docs/azure/snippets/WebHook.Api/WebHook.Api.csproj b/docs/azure/snippets/bicep/WebHook.Api/WebHook.Api.csproj
similarity index 100%
rename from docs/azure/snippets/WebHook.Api/WebHook.Api.csproj
rename to docs/azure/snippets/bicep/WebHook.Api/WebHook.Api.csproj
diff --git a/docs/azure/snippets/WebHook.Api/WebHook.Api.http b/docs/azure/snippets/bicep/WebHook.Api/WebHook.Api.http
similarity index 100%
rename from docs/azure/snippets/WebHook.Api/WebHook.Api.http
rename to docs/azure/snippets/bicep/WebHook.Api/WebHook.Api.http
diff --git a/docs/azure/snippets/bicep/WebHook.Api/appsettings.Development.json b/docs/azure/snippets/bicep/WebHook.Api/appsettings.Development.json
new file mode 100644
index 0000000000..0c208ae918
--- /dev/null
+++ b/docs/azure/snippets/bicep/WebHook.Api/appsettings.Development.json
@@ -0,0 +1,8 @@
+{
+ "Logging": {
+ "LogLevel": {
+ "Default": "Information",
+ "Microsoft.AspNetCore": "Warning"
+ }
+ }
+}
diff --git a/docs/azure/snippets/WebHook.Api/appsettings.json b/docs/azure/snippets/bicep/WebHook.Api/appsettings.json
similarity index 100%
rename from docs/azure/snippets/WebHook.Api/appsettings.json
rename to docs/azure/snippets/bicep/WebHook.Api/appsettings.json
diff --git a/docs/azureai/azureai-openai-integration.md b/docs/azureai/azureai-openai-integration.md
index c092a3884b..b25e6fd8b3 100644
--- a/docs/azureai/azureai-openai-integration.md
+++ b/docs/azureai/azureai-openai-integration.md
@@ -1,7 +1,7 @@
---
title: .NET Aspire Azure OpenAI integration (Preview)
description: Learn how to use the .NET Aspire Azure OpenAI integration.
-ms.date: 03/06/2025
+ms.date: 04/03/2025
---
# .NET Aspire Azure OpenAI integration (Preview)
@@ -83,25 +83,9 @@ If you're new to [Bicep](/azure/azure-resource-manager/bicep/overview), it's a d
:::code language="bicep" source="../snippets/azure/AppHost/openai.module.bicep":::
-The preceding Bicep is a module that provisions an Azure Cognitive Services resource with the following defaults:
-
-- `location`: The location of the resource group.
-- `principalType`: The principal type of the Cognitive Services resource.
-- `principalId`: The principal ID of the Cognitive Services resource.
-- `openai`: The Cognitive Services account resource.
- - `kind`: The kind of the resource, set to `OpenAI`.
- - `properties`: The properties of the resource.
- - `customSubDomainName`: The custom subdomain name for the resource, based on the unique string of the resource group ID.
- - `publicNetworkAccess`: Set to `Enabled`.
- - `disableLocalAuth`: Set to `true`.
- - `sku`: The SKU of the resource, set to `S0`.
-- `openai_CognitiveServicesOpenAIContributor`: The Cognitive Services resource owner, based on the build-in `Azure Cognitive Services OpenAI Contributor` role. For more information, see [Azure Cognitive Services OpenAI Contributor](/azure/role-based-access-control/built-in-roles/ai-machine-learning#cognitive-services-openai-contributor).
-- `preview`: The deployment resource, based on the `preview` name.
- - `properties`: The properties of the deployment resource.
- - `format`: The format of the deployment resource, set to `OpenAI`.
- - `modelName`: The model name of the deployment resource, set to `gpt-4.5-preview`.
- - `modelVersion`: The model version of the deployment resource, set to `2025-02-27`.
-- `connectionString`: The connection string, containing the endpoint of the Cognitive Services resource.
+The preceding Bicep is a module that provisions an Azure Cognitive Services resource. Additionally, role assignments are created for the Azure resource in a separate module:
+
+:::code language="bicep" source="../snippets/azure/AppHost/openai-roles.module.bicep":::
The generated Bicep is a starting point and is influenced by changes to the provisioning infrastructure in C#. Customizations to the Bicep file directly will be overwritten, so make changes through the C# provisioning APIs to ensure they are reflected in the generated files.
diff --git a/docs/azureai/azureai-search-document-integration.md b/docs/azureai/azureai-search-document-integration.md
index cc2b3ec146..5d46781821 100644
--- a/docs/azureai/azureai-search-document-integration.md
+++ b/docs/azureai/azureai-search-document-integration.md
@@ -1,7 +1,7 @@
---
title: .NET Aspire Azure AI Search integration
description: Learn how to integrate Azure AI Search with .NET Aspire.
-ms.date: 03/07/2025
+ms.date: 04/03/2025
---
# .NET Aspire Azure AI Search integration
@@ -57,21 +57,9 @@ If you're new to [Bicep](/azure/azure-resource-manager/bicep/overview), it's a d
:::code language="bicep" source="../snippets/azure/AppHost/search.module.bicep":::
-The preceding Bicep is a module that provisions an Azure AI Search service resource with the following defaults:
-
-- `location`: The location parameter of the resource group, defaults to `resourceGroup().location`.
-- `principalType`: The principal type parameter of the Azure AI Search resource.
-- `principalId`: The principal ID parameter of the Azure AI Search resource.
-- `search`: The resource representing the Azure AI Search service.
- - `properties`: The properties of the Azure AI Search service:
- - `hostingMode`: Is set to `default`.
- - `disableLocalAuth`: Is set to `true`.
- - `partitionCount`: Is set to `1`.
- - `replicaCount`: Is set to `1`.
- - `sku`: Defaults to `basic`.
-- `search_SearchIndexDataContributor`: The role assignment for the Azure AI Search index data contributor role. For more information, see [Search Index Data Contributor](/azure/role-based-access-control/built-in-roles/ai-machine-learning#search-index-data-contributor).
-- `search_SearchServiceContributor`: The role assignment for the Azure AI Search service contributor role. For more information, see [Search Service Contributor](/azure/role-based-access-control/built-in-roles/ai-machine-learning#search-service-contributor).
-- `connectionString`: The connection string for the Azure AI Search service, which is used to connect to the service. The connection string is generated using the `Endpoint` property of the Azure AI Search service.
+The preceding Bicep is a module that provisions an Azure AI Search service resource. Additionally, role assignments are created for the Azure resource in a separate module:
+
+:::code language="bicep" source="../snippets/azure/AppHost/search-roles.module.bicep":::
The generated Bicep is a starting point and is influenced by changes to the provisioning infrastructure in C#. Customizations to the Bicep file directly will be overwritten, so make changes through the C# provisioning APIs to ensure they are reflected in the generated files.
diff --git a/docs/caching/includes/azure-redis-app-host.md b/docs/caching/includes/azure-redis-app-host.md
index 597be5dc4e..9837bf8818 100644
--- a/docs/caching/includes/azure-redis-app-host.md
+++ b/docs/caching/includes/azure-redis-app-host.md
@@ -50,19 +50,9 @@ If you're new to [Bicep](/azure/azure-resource-manager/bicep/overview), it's a d
:::code language="bicep" source="../../snippets/azure/AppHost/redis.module.bicep":::
-The preceding Bicep is a module that provisions an Azure Cache for Redis with the following defaults:
-
-- `location`: The location of the Azure Cache for Redis resource. The default is the location of the resource group.
-- `principalId`: The principal ID of the Azure Cache for Redis resource.
-- `principalName`: The principal name of the Azure Cache for Redis resource.
-- `sku`: The SKU of the Azure Cache for Redis resource. The default is `Basic` with a capacity of `1`.
-- `enableNonSslPort`: The non-SSL port of the Azure Cache for Redis resource. The default is `false`.
-- `disableAccessKeyAuthentication`: The access key authentication of the Azure Cache for Redis resource. The default is `true`.
-- `minimumTlsVersion`: The minimum TLS version of the Azure Cache for Redis resource. The default is `1.2`.
-- `redisConfiguration`: The Redis configuration of the Azure Cache for Redis resource. The default is `aad-enabled` set to `true`.
-- `tags`: The tags of the Azure Cache for Redis resource. The default is `aspire-resource-name` set to the name of the Aspire resource, in this case `redis`.
-- `redis_contributor`: The contributor of the Azure Cache for Redis resource, with an access policy name of `Data Contributor`.
-- `connectionString`: The connection string of the Azure Cache for Redis resource.
+The preceding Bicep is a module that provisions an Azure Cache for Redis resource. Additionally, role assignments are created for the Azure resource in a separate module:
+
+:::code language="bicep" source="../../snippets/azure/AppHost/redis-roles.module.bicep":::
In addition to the Azure Cache for Redis, it also provisions an access policy assignment to the application access to the cache. The generated Bicep is a starting point and is influenced by changes to the provisioning infrastructure in C#. Customizations to the Bicep file directly will be overwritten, so make changes through the C# provisioning APIs to ensure they are reflected in the generated files.
diff --git a/docs/compatibility/9.2/generated-bicep-updates.md b/docs/compatibility/9.2/generated-bicep-updates.md
new file mode 100644
index 0000000000..e6c3d771b4
--- /dev/null
+++ b/docs/compatibility/9.2/generated-bicep-updates.md
@@ -0,0 +1,62 @@
+---
+title: "Breaking change - Role Assignments separated from Azure resource bicep"
+description: "Learn about the breaking change in .NET Aspire 9.2 where role assignments are moved to separate bicep modules."
+ms.date: 4/2/2025
+ai-usage: ai-assisted
+ms.custom: https://github.com/dotnet/docs-aspire/issues/2911
+---
+
+# Role Assignments separated from Azure resource bicep
+
+In .NET Aspire 9.2, role assignments for Azure resources are no longer included in the same bicep file as the resource. Instead, they're moved to separate bicep modules. This change affects how role assignments are customized during infrastructure configuration.
+
+## Version introduced
+
+.NET Aspire 9.2
+
+## Previous behavior
+
+Previously, when an Azure resource's bicep file was generated, default role assignments were included in the same bicep module as the resource. This allowed customization of role assignments in the `ConfigureInfrastructure` callback. For example:
+
+```csharp
+var storage = builder.AddAzureStorage("storage")
+ .ConfigureInfrastructure(infra =>
+ {
+ var roles = infra.GetProvisionableResources().OfType().ToList();
+
+ foreach (var role in roles)
+ {
+ infra.Remove(role);
+ }
+
+ var storageAccount = infra.GetProvisionableResources().OfType().Single();
+ infra.Add(storageAccount.CreateRoleAssignment(StorageBuiltInRole.StorageBlobDataContributor, ...));
+ });
+```
+
+## New behavior
+
+Role assignments are now moved to their own bicep modules. The `ConfigureInfrastructure` callback no longer contains any `RoleAssignment` instances. Instead, role assignments are configured using the `WithRoleAssignments` API. For example:
+
+```csharp
+var storage = builder.AddAzureStorage("storage");
+
+builder.AddProject("api")
+ .WithRoleAssignments(storage, StorageBuiltInRole.StorageBlobDataContributor);
+```
+
+## Type of breaking change
+
+This is a [behavioral change](../categories.md#behavioral-change).
+
+## Reason for change
+
+This change was necessary to implement the `WithRoleAssignments` APIs, which provide a more structured and flexible way to configure role assignments per application.
+
+## Recommended action
+
+To customize role assignments in .NET Aspire 9.2, use the `WithRoleAssignments` API instead of relying on the `ConfigureInfrastructure` callback. Update your code as shown in the [preceding example](#new-behavior).
+
+## Affected APIs
+
+-
diff --git a/docs/compatibility/9.2/index.md b/docs/compatibility/9.2/index.md
new file mode 100644
index 0000000000..e44e6b7397
--- /dev/null
+++ b/docs/compatibility/9.2/index.md
@@ -0,0 +1,25 @@
+---
+title: Breaking changes in .NET Aspire 9.2
+titleSuffix: ""
+description: Navigate to the breaking changes in .NET Aspire 9.2.
+ms.date: 04/02/2025
+---
+
+# Breaking changes in .NET Aspire 9.2
+
+If you're migrating an app to .NET Aspire 9.2, the breaking changes listed here might affect you.
+
+[!INCLUDE [binary-source-behavioral](../includes/binary-source-behavioral.md)]
+
+> [!NOTE]
+> This article is a work in progress. It's not a complete list of breaking changes in .NET Aspire 9.2.
+
+## Breaking changes
+
+| Title | Type of change | Introduced version |
+|--|--|--|
+| [AzureContainerApps infrastructure creates managed identity per container app](managed-identity-per-app.md) | Behavioral change | 9.2 |
+| [KeyVault default role assignment changing from KeyVaultAdministrator to KeyVaultSecretsUser](keyvault-role-assignment-changes.md) | Behavioral change | 9.2 |
+| [Role Assignments separated from Azure resource bicep](generated-bicep-updates.md) | Behavioral change | 9.2 |
+| [With authentication API creates keyvault resource in the app model](withauthentication-changes.md) | Behavioral change | 9.2 |
+| [WithCommand obsolete and new overload with CommandOptions](withcommand-obsolete.md) | Source incompatible | 9.2 |
diff --git a/docs/compatibility/9.2/keyvault-role-assignment-changes.md b/docs/compatibility/9.2/keyvault-role-assignment-changes.md
new file mode 100644
index 0000000000..cb6c1c1c24
--- /dev/null
+++ b/docs/compatibility/9.2/keyvault-role-assignment-changes.md
@@ -0,0 +1,59 @@
+---
+title: "Breaking change - KeyVault default role assignment changing from KeyVaultAdministrator to KeyVaultSecretsUser"
+description: "Learn about the breaking change in .NET Aspire 9.2 where the default role for Azure KeyVault applications changes to KeyVaultSecretsUser."
+ms.date: 03/27/2025
+ai-usage: ai-assisted
+ms.custom: https://github.com/dotnet/docs-aspire/issues/2899
+---
+
+# KeyVault default role assignment changing from KeyVaultAdministrator to KeyVaultSecretsUser
+
+In .NET Aspire 9.2, the default role assigned to applications referencing Azure KeyVault has changed from to . This change enhances security by limiting default privileges to only reading secrets. Applications requiring higher privileges must explicitly configure them.
+
+## Version introduced
+
+.NET Aspire 9.2
+
+## Previous behavior
+
+Previously, applications referencing Azure KeyVault were automatically granted the `KeyVaultAdministrator` role, which allowed full management of KeyVault settings.
+
+## New behavior
+
+Applications referencing Azure KeyVault are now granted the `KeyVaultSecretsUser` role by default, which restricts access to reading secrets. If higher privileges are required, they can be configured using the `WithRoleAssignments` API.
+
+Example:
+
+```csharp
+using Azure.Provisioning.KeyVault;
+
+var kv = builder.AddAzureKeyVault("kv");
+
+builder.AddProject("api")
+ .WithRoleAssignments(kv, KeyVaultBuiltInRole.KeyVaultContributor);
+```
+
+## Type of breaking change
+
+This is a [behavioral change](../categories.md#behavioral-change).
+
+## Reason for change
+
+The `KeyVaultAdministrator` role provides excessive privileges for most applications, as they typically only need to read secrets. Assigning the `KeyVaultSecretsUser` role by default improves security by adhering to the principle of least privilege.
+
+## Recommended action
+
+If your application requires higher privileges than the `KeyVaultSecretsUser` role, explicitly configure the necessary roles using the `WithRoleAssignments` API. For example:
+
+```csharp
+using Azure.Provisioning.KeyVault;
+
+var kv = builder.AddAzureKeyVault("kv");
+
+builder.AddProject("api")
+ .WithRoleAssignments(kv, KeyVaultBuiltInRole.KeyVaultContributor);
+```
+
+## Affected APIs
+
+-
diff --git a/docs/compatibility/9.2/managed-identity-per-app.md b/docs/compatibility/9.2/managed-identity-per-app.md
new file mode 100644
index 0000000000..9044b5290e
--- /dev/null
+++ b/docs/compatibility/9.2/managed-identity-per-app.md
@@ -0,0 +1,54 @@
+---
+title: "Breaking change - AzureContainerApps infrastructure creates managed identity per container app"
+description: "Learn about the breaking change in .NET Aspire 9.2 where each ContainerApp now has its own managed identity."
+ms.date: 4/2/2025
+ai-usage: ai-assisted
+ms.custom: https://github.com/dotnet/docs-aspire/issues/2914
+---
+
+# Azure Container Apps managed identity changes
+
+Starting with .NET Aspire 9.2, each Azure Container App created using [📦 Aspire.Hosting.Azure.AppContainers](https://www.nuget.org/packages/Aspire.Hosting.Azure.AppContainers) NuGet package now has its own Azure Managed Identity. This change enables more granular role assignments for Azure resources but might require updates to applications that rely on shared managed identities.
+
+## Version introduced
+
+.NET Aspire 9.2
+
+## Previous behavior
+
+All ContainerApps shared a single Azure Managed Identity. This allowed applications to interact with Azure resources using a common identity.
+
+## New behavior
+
+Each ContainerApp now has its own unique Azure Managed Identity. This enables applications to have distinct role assignments for different Azure resources.
+
+## Type of breaking change
+
+This is a [behavioral change](../categories.md#behavioral-change).
+
+## Reason for change
+
+This change was introduced to support scenarios where applications require different role assignments for different Azure resources. By assigning a unique managed identity to each ContainerApp, applications can now operate with more granular access control.
+
+## Recommended action
+
+The recommended action is to update your Azure resources to use the new managed identities. This may include:
+
+### Azure SQL Server
+
+Grant access to all Azure Managed Identities that need to interact with the database. Follow the guidance in [Configure and manage Azure AD authentication with Azure SQL](/azure/azure-sql/database/authentication-aad-configure).
+
+### Azure PostgreSQL
+
+Grant necessary privileges to all Azure Managed Identities that need to interact with the database. Use the PostgreSQL documentation on [granting privileges](https://www.postgresql.org/docs/current/ddl-priv.html) as a reference. For example:
+
+```sql
+GRANT INSERT, UPDATE, DELETE ON ALL TABLES IN SCHEMA public TO ;
+```
+
+## Affected APIs
+
+- `Aspire.Hosting.AzureContainerAppExtensions.AddAzureContainerAppsInfrastructure`
+- `Aspire.Hosting.AzureContainerAppProjectExtensions.PublishAsAzureContainerApp`
+- `Aspire.Hosting.AzureContainerAppExecutableExtensions.PublishAsAzureContainerApp`
+- `Aspire.Hosting.AzureContainerAppContainerExtensions.PublishAsAzureContainerApp`
diff --git a/docs/compatibility/9.2/withauthentication-changes.md b/docs/compatibility/9.2/withauthentication-changes.md
new file mode 100644
index 0000000000..ffd9e332d4
--- /dev/null
+++ b/docs/compatibility/9.2/withauthentication-changes.md
@@ -0,0 +1,47 @@
+---
+title: "Breaking change - WithAccessKeyAuthentication and WithPasswordAuthentication create a keyvault resource in the app model"
+description: "Learn about the breaking change in .NET Aspire 9.2 where key vault resources are now created or referenced directly in the app model."
+ms.date: 3/25/2025
+ai-usage: ai-assisted
+ms.custom: https://github.com/dotnet/docs-aspire/issues/2889
+---
+
+# With authentication API creates keyvault resource in the app model
+
+Starting in .NET Aspire 9.2, calling any of the following methods:
+
+-
+-
+-
+
+Will now create (or add references to) a key vault resource directly in the app model. This change allows better customization and management of connection strings and secrets.
+
+## Version introduced
+
+.NET Aspire 9.2
+
+## Previous behavior
+
+Previously, calling `WithAccessKeyAuthentication` on CosmosDB or AzureRedis, or `WithPasswordAuthentication` on AzurePostgres, automatically created and managed Bicep resources. These resources were invisible to the app model and could not be managed or customized in C#.
+
+## New behavior
+
+In .NET Aspire 9.2, calling `WithAccessKeyAuthentication` or `WithPasswordAuthentication` adds an empty `keyVaultName` parameter as a known parameter in the Bicep file. The app model now directly creates the key vault resource or allows you to pass a reference to an existing AzureKeyVault resource where secrets will be stored. Key vault secret names for connection strings are now formatted as `connectionstrings--{resourcename}` to avoid conflicts with other connection strings.
+
+## Type of breaking change
+
+This is a [behavioral change](../categories.md#behavioral-change).
+
+## Reason for change
+
+This change moves resource management to the app host, making it easier to customize and manage resources. It allows sharing a key vault across multiple resources or using an existing key vault to manage connection strings and secrets.
+
+## Recommended action
+
+There is currently no workaround for this change. Ensure that your app model is updated to handle the new behavior for key vault resources and connection string management.
+
+## Affected APIs
+
+-
+-
+-
diff --git a/docs/compatibility/9.2/withcommand-obsolete.md b/docs/compatibility/9.2/withcommand-obsolete.md
new file mode 100644
index 0000000000..efbab0e30e
--- /dev/null
+++ b/docs/compatibility/9.2/withcommand-obsolete.md
@@ -0,0 +1,96 @@
+---
+title: "Breaking change - WithCommand obsolete and new overload with CommandOptions"
+description: "Learn about the breaking change in .NET Aspire 9.2 where the WithCommand method overload with optional parameters is marked obsolete."
+ms.date: 3/25/2025
+ai-usage: ai-assisted
+ms.custom: https://github.com/dotnet/docs-aspire/issues/2888
+---
+
+# WithCommand obsolete and new overload with CommandOptions
+
+The method overload that accepted multiple optional parameters is now marked obsolete. A new overload that accepts an instance of `CommandOptions` has been introduced. This change requires updates to existing code to use the new overload.
+
+## Version introduced
+
+.NET Aspire 9.2
+
+## Previous behavior
+
+The `WithCommand` method overload that accepted multiple optional parameters was available and not marked as obsolete.
+
+```csharp
+var builder = DistributedApplication.CreateBuilder(args);
+
+builder.AddProject("apiservice")
+ .WithCommand(
+ name: "command-name",
+ displayName: "Command display name",
+ executeCommand: async (ExecuteCommandContext context) =>
+ {
+ // Command execution logic here
+ await Task.CompletedTask;
+ return CommandResults.Success();
+ },
+ updateState: (UpdateCommandStateContext context) =>
+ {
+ // State update logic here
+ return ResourceCommandState.Enabled;
+ },
+ displayDescription: "Command Description",
+ parameter: new[] { "", "" },
+ confirmationMessage: "Are you sure?",
+ iconName: "Icons",
+ iconVariant: "Red",
+ isHighlighted: false);
+```
+
+## New behavior
+
+The existing method overload is now marked obsolete. A new overload that accepts an instance of `CommandOptions` should be used instead.
+
+```csharp
+var builder = DistributedApplication.CreateBuilder(args);
+
+builder.AddProject("apiservice")
+ .WithCommand(
+ name: "command-name",
+ displayName: "Command display name",
+ executeCommand: async (ExecuteCommandContext context) =>
+ {
+ // Command execution logic here
+ await Task.CompletedTask;
+ return CommandResults.Success();
+ },
+ commandOptions: new CommandOptions
+ {
+ UpdateState = (UpdateCommandStateContext context) =>
+ {
+ // State update logic here
+ return ResourceCommandState.Enabled;
+ },
+ Description = "Command Description",
+ Parameter = new[] { "", "" },
+ ConfirmationMessage = "Are you sure?",
+ IconName = "Icons",
+ IconVariant = "Red",
+ IsHighlighted = false
+ });
+```
+
+The only required parameters are the `name`, `displayName`, and `executeCommand`. The rest of the parameters are now encapsulated within the `CommandOptions` object, which provides a cleaner and more maintainable API.
+
+## Type of breaking change
+
+This is a [source incompatible](../categories.md#source-compatibility) change.
+
+## Reason for change
+
+This change was made following an API review to improve clarity and maintainability by consolidating optional parameters into a single `CommandOptions` object.
+
+## Recommended action
+
+Update your code to use the new `WithCommand` overload that accepts an instance of `CommandOptions`. Replace calls to the obsolete overload with the new overload.
+
+## Affected APIs
+
+-
diff --git a/docs/compatibility/api-removal.md b/docs/compatibility/api-removal.md
index 0a777b74f2..cc99a34d82 100644
--- a/docs/compatibility/api-removal.md
+++ b/docs/compatibility/api-removal.md
@@ -14,3 +14,7 @@ ms.date: 10/24/2024
- By documenting it under [Breaking changes in .NET](breaking-changes.md).
In most cases, an API that shipped in a long-term support (LTS) release is obsoleted in the subsequent LTS release before it's removed. In rare cases, based on business needs, exceptions are made to obsolete an API before the subsequent LTS release. All obsoletions are documented and communicated to customers.
+
+## See also
+
+- [.NET Aspire Support Policy](https://dotnet.microsoft.com/platform/support/policy/aspire)
diff --git a/docs/compatibility/breaking-changes.md b/docs/compatibility/breaking-changes.md
index 6a94a19066..a23b208b74 100644
--- a/docs/compatibility/breaking-changes.md
+++ b/docs/compatibility/breaking-changes.md
@@ -20,4 +20,5 @@ You can also view individual issues that detail the breaking changes introduced
## See also
- [API removal in .NET](api-removal.md)
+- [.NET Aspire Support Policy](https://dotnet.microsoft.com/platform/support/policy/aspire)
- [.NET runtime compatibility](/dotnet/core/versions/#net-runtime-compatibility)
diff --git a/docs/compatibility/categories.md b/docs/compatibility/categories.md
index 6ddbac207d..7d1b1470b2 100644
--- a/docs/compatibility/categories.md
+++ b/docs/compatibility/categories.md
@@ -56,3 +56,7 @@ Forward compatibility refers to the ability of an existing consumer of an API to
Maintaining forward compatibility virtually precludes any changes or additions from version to version, since those changes prevent a consumer that targets a later version from running under an earlier version. Developers expect that a consumer that relies on a newer API may not function correctly against the older API.
Maintaining forward compatibility is not a goal of .NET Aspire.
+
+## See also
+
+- [.NET Aspire Support Policy](https://dotnet.microsoft.com/platform/support/policy/aspire)
diff --git a/docs/compatibility/toc.yml b/docs/compatibility/toc.yml
index 74236c499a..e60fb712ea 100644
--- a/docs/compatibility/toc.yml
+++ b/docs/compatibility/toc.yml
@@ -5,9 +5,28 @@ items:
href: ../get-started/aspire-overview.md
- name: Breaking changes
href: breaking-changes.md
-- name: .NET Aspire 9.1
+- name: .NET Aspire 9.2
expanded: true
items:
+ - name: Overview
+ href: 9.2/index.md
+ - name: Breaking changes in 9.2
+ expanded: true
+ items:
+ - name: Azure Container Apps managed identity changes
+ href: 9.2/managed-identity-per-app.md
+ - name: KeyVault default role assignment changes
+ href: 9.2/keyvault-role-assignment-changes.md
+ - name: Role Assignments is separate bicep
+ href: 9.2/generated-bicep-updates.md
+ - name: With authentication APIs include semantic changes
+ href: 9.2/withauthentication-changes.md
+ - name: WithCommand obsolete, use new overload
+ href: 9.2/withcommand-obsolete.md
+
+- name: .NET Aspire 9.1
+ expanded: false
+ items:
- name: Overview
href: 9.1/index.md
- name: Breaking changes in 9.1
@@ -50,3 +69,5 @@ items:
href: 9.0/azure-resource-name-scheme.md
- name: Ollama integration updates
href: 9.0/ollama-integration-updates.md
+- name: .NET Aspire Support Policy
+ href: https://dotnet.microsoft.com/platform/support/policy/aspire
diff --git a/docs/database/azure-cosmos-db-entity-framework-integration.md b/docs/database/azure-cosmos-db-entity-framework-integration.md
index d6b18de7b7..d05181f287 100644
--- a/docs/database/azure-cosmos-db-entity-framework-integration.md
+++ b/docs/database/azure-cosmos-db-entity-framework-integration.md
@@ -1,7 +1,7 @@
---
title: .NET Aspire Cosmos DB Entity Framework Core integration
description: Learn how to install and configure the .NET Aspire Cosmos DB Entity Framework Core integration to connect to existing Cosmos DB instances or create new instances from .NET with the Azure Cosmos DB emulator.
-ms.date: 02/26/2025
+ms.date: 04/01/2025
uid: dotnet/aspire/azure-cosmos-db-entity-framework-integration
---
@@ -42,12 +42,18 @@ dotnet add package Aspire.Microsoft.EntityFrameworkCore.Cosmos
### Add Cosmos DB context
-In the :::no-loc text="Program.cs"::: file of your client-consuming project, call the extension method to register a for use via the dependency injection container. The method takes a connection name parameter.
+In the :::no-loc text="Program.cs"::: file of your client-consuming project, call the extension method to register a for use via the dependency injection container. The method takes a connection name parameter and a database name parameter.
```csharp
builder.AddCosmosDbContext("cosmosdb", "databaseName");
```
+Alternatively, the database name can be inferred from the connection when there's a single database in the connection string. In this case, you can omit the database name parameter:
+
+```csharp
+builder.AddCosmosDbContext("cosmosdb");
+```
+
> [!TIP]
> The `connectionName` parameter must match the name used when adding the Cosmos DB resource in the app host project. In other words, when you call `AddAzureCosmosDB` and provide a name of `cosmosdb` that same name should be used when calling `AddCosmosDbContext`. For more information, see [Add Azure Cosmos DB resource](#add-azure-cosmos-db-resource).
diff --git a/docs/database/azure-cosmos-db-integration.md b/docs/database/azure-cosmos-db-integration.md
index db311bbdbf..3c42820cec 100644
--- a/docs/database/azure-cosmos-db-integration.md
+++ b/docs/database/azure-cosmos-db-integration.md
@@ -1,7 +1,7 @@
---
title: .NET Aspire Azure Cosmos DB integration
description: Learn how to install and configure the .NET Aspire Azure Cosmos DB integration to connect to existing Cosmos DB instances or create new instances from .NET with the Azure Cosmos DB emulator.
-ms.date: 02/26/2025
+ms.date: 04/10/2025
uid: dotnet/aspire/azure-cosmos-db-integration
---
@@ -11,6 +11,8 @@ uid: dotnet/aspire/azure-cosmos-db-integration
[Azure Cosmos DB](https://azure.microsoft.com/services/cosmos-db/) is a fully managed NoSQL database service for modern app development. The .NET Aspire Azure Cosmos DB integration enables you to connect to existing Cosmos DB instances or create new instances from .NET with the Azure Cosmos DB emulator.
+If you're looking for the Entity Framework Core integration, see [.NET Aspire Cosmos DB Entity Framework Core integration](azure-cosmos-db-entity-framework-integration.md).
+
## Hosting integration
[!INCLUDE [cosmos-app-host](includes/cosmos-app-host.md)]
@@ -51,7 +53,7 @@ builder.AddAzureCosmosClient(connectionName: "cosmos-db");
> [!TIP]
> The `connectionName` parameter must match the name used when adding the Cosmos DB resource in the app host project. In other words, when you call `AddAzureCosmosDB` and provide a name of `cosmos-db` that same name should be used when calling `AddAzureCosmosClient`. For more information, see [Add Azure Cosmos DB resource](#add-azure-cosmos-db-resource).
-You can then retrieve the instance using dependency injection. For example, to retrieve the connection from an example service:
+You can then retrieve the instance using dependency injection. For example, to retrieve the client from an example service:
```csharp
public class ExampleService(CosmosClient client)
@@ -87,6 +89,142 @@ public class ExampleService(
For more information on keyed services, see [.NET dependency injection: Keyed services](/dotnet/core/extensions/dependency-injection#keyed-services).
+### Add Azure Cosmos DB database
+
+
+
+In the app host, the database resource (`AzureCosmosDBDatabaseResource`) can be added as a child resource to the parent . In your client-consuming project, you can deep-link to the database resource by name, registering a instance for use with dependency injection. For example, consider the following code that calls `AddAzureCosmosDatabase` on an instance:
+
+```csharp
+builder.AddAzureCosmosDatabase(connectionName: "customers");
+```
+
+
+
+The `AddAzureCosmosDatabase` API returns a `CosmosDatabaseBuilder` instance that you can use to attach multiple containers under the same database connection. All child containers share the same and database connection and `CosmosClient` instance. This strategy is useful when associating the same with multiple containers.
+
+After calling `AddAzureCosmosDatabase`, you can then retrieve the `Database` instance using dependency injection. For example, to retrieve the database from a delegate in a call consider the following code:
+
+```csharp
+app.MapGet("/api/customers", async (Database database) =>
+{
+ // Query data from database...
+});
+```
+
+### Add keyed Azure Cosmos DB database
+
+
+
+There's also an `AddKeyedAzureCosmosDatabase` API that returns a `CosmosDatabaseBuilder` instance that you can use to attach multiple containers under the same database connection. method that allows you to register multiple databases with different connection names. For example, consider the following code that calls `AddKeyedAzureCosmosDatabase` on an instance:
+
+```csharp
+var builder = WebApplication.CreateBuilder(args);
+
+builder.AddKeyedAzureCosmosDatabase("customers");
+builder.AddKeyedAzureCosmosDatabase("orders");
+
+var app = builder.Build();
+
+app.MapGet("/api/customers", async (
+ [FromKeyedServices("customers")] Database database) =>
+{
+ // Get container from database and query data
+});
+
+app.MapPost("/api/orders", async (
+ [FromKeyedServices("orders")] Database database,
+ [FromBody] OrderRequest order) =>
+{
+ // Get container from database and query data
+});
+
+app.Run();
+```
+
+The preceding example code demonstrates how to register two databases, `details` and `customers`. Each named database can be used to get their corresponding containers to query data.
+
+### Add Azure Cosmos DB container
+
+
+
+When you add a Cosmos DB resource in the app host project, you can also add an Azure Cosmos DB container resource as well. The container resource is considered a child resource to the parent `AzureCosmosDBDatabaseResource`. In your client-consuming project, you can deep-link to the container resource by name, registering a instance for use with dependency injection. For example, consider the following code that calls `AddAzureCosmosContainer` on an instance:
+
+```csharp
+builder.AddAzureCosmosContainer(connectionName: "details");
+```
+
+You can then retrieve the `Container` instance using dependency injection. For example, to retrieve the container from a delegate in a call consider the following code:
+
+```csharp
+app.MapGet("/api/orders/{id:guid}", async (
+ Container container,
+ [FromRoute] Guid id) =>
+{
+ // Query data from container...
+});
+```
+
+### Add keyed Azure Cosmos DB container
+
+
+
+There's also an `AddKeyedAzureCosmosContainer` method that allows you to register multiple containers with different connection names. For example, consider the following code that calls `AddKeyedAzureCosmosContainer` on an instance:
+
+```csharp
+var builder = WebApplication.CreateBuilder(args);
+
+builder.AddKeyedAzureCosmosContainer("customers");
+
+var app = builder.Build();
+
+app.MapGet("/api/customers", async (
+ [FromKeyedServices("customers")] Container container) =>
+{
+ // Query data from container...
+});
+
+app.Run();
+```
+
+If you have multiple containers under the same database connection, you can use the `AddAzureCosmosDatabase` API to attach multiple containers under the same database connection. All child containers share the same and database connection. This strategy is useful when associating the same with multiple containers. Consider the following alternative code, to register multiple containers under the same database connection:
+
+```csharp
+var builder = WebApplication.CreateBuilder(args);
+
+builder.AddAzureCosmosDatabase("customers", configureClientOptions: options =>
+ {
+ options.SerializerOptions = new CosmosSerializationOptions()
+ {
+ PropertyNamingPolicy = CosmosPropertyNamingPolicy.CamelCase
+ };
+ })
+ .AddKeyedContainer(name: "profiles");
+
+builder.AddAzureCosmosDatabase(connectionName: "orders")
+ .AddKeyedContainer(name: "details")
+ .AddKeyedContainer(name: "history");
+
+var app = builder.Build();
+
+app.MapGet("/api/customers", async (
+ [FromKeyedServices("profiles")] Container container) =>
+{
+ // Query data from container
+});
+
+app.MapGet("/api/orders", async (
+ [FromKeyedServices("details")] Container container,
+ [FromKeyedServices("history")] Container container) =>
+{
+ // Query data from container
+});
+
+app.Run();
+```
+
+The preceding example code demonstrates how to register two databases, `customers` and `orders`, each with their own containers. The `customers` database has a single container named `profiles`, while the `orders` database has two containers named `details` and `history`. Each container can be queried individually using its respective key.
+
### Configuration
The .NET Aspire Azure Cosmos DB integration provides multiple options to configure the connection based on the requirements and conventions of your project.
@@ -201,6 +339,7 @@ The .NET Aspire Azure Cosmos DB integration currently doesn't support metrics by
## See also
- [Azure Cosmos DB](https://azure.microsoft.com/services/cosmos-db)
+- [Sample repository showing parent-child relationships](https://github.com/captainsafia/aspire-child-resources)
- [.NET Aspire Cosmos DB Entity Framework Core integration](azure-cosmos-db-entity-framework-integration.md)
- [.NET Aspire integrations overview](../fundamentals/integrations-overview.md)
- [.NET Aspire Azure integrations overview](../azure/integrations-overview.md)
diff --git a/docs/database/azure-postgresql-entity-framework-integration.md b/docs/database/azure-postgresql-entity-framework-integration.md
index d6095fd6ca..eca0f14f38 100644
--- a/docs/database/azure-postgresql-entity-framework-integration.md
+++ b/docs/database/azure-postgresql-entity-framework-integration.md
@@ -1,7 +1,7 @@
---
title: .NET Aspire Azure PostgreSQL Entity Framework Core integration
description: Learn how to integrate Azure PostgreSQL with .NET Aspire applications, using both hosting and Entity Framework Core client integrations.
-ms.date: 01/21/2025
+ms.date: 03/31/2025
uid: dotnet/aspire/azure-postgresql-entity-framework-integration
---
@@ -17,8 +17,6 @@ uid: dotnet/aspire/azure-postgresql-entity-framework-integration
## Client integration
-[!INCLUDE [postgresql-ef-client](includes/postgresql-ef-client.md)]
-
[!INCLUDE [azure-postgresql-ef-client](includes/azure-postgresql-ef-client.md)]
## See also
diff --git a/docs/database/azure-postgresql-integration.md b/docs/database/azure-postgresql-integration.md
index ce3733a0fe..79f4ba482b 100644
--- a/docs/database/azure-postgresql-integration.md
+++ b/docs/database/azure-postgresql-integration.md
@@ -1,7 +1,7 @@
---
title: .NET Aspire Azure PostgreSQL integration
description: Learn how to integrate Azure PostgreSQL with .NET Aspire applications, using both hosting and client integrations.
-ms.date: 01/21/2025
+ms.date: 03/31/2025
uid: dotnet/aspire/azure-postgresql-integration
---
@@ -17,8 +17,6 @@ uid: dotnet/aspire/azure-postgresql-integration
## Client integration
-[!INCLUDE [postgresql-client](includes/postgresql-client.md)]
-
[!INCLUDE [azure-postgresql-client](includes/azure-postgresql-client.md)]
## See also
diff --git a/docs/database/includes/azure-postgresql-client.md b/docs/database/includes/azure-postgresql-client.md
index c0c26f0027..e130f50753 100644
--- a/docs/database/includes/azure-postgresql-client.md
+++ b/docs/database/includes/azure-postgresql-client.md
@@ -2,42 +2,165 @@
ms.topic: include
---
-### Add Azure authenticated Npgsql client
-
-By default, when you call `AddAzurePostgresFlexibleServer` in your PostgreSQL hosting integration, it configures [📦 Azure.Identity](https://www.nuget.org/packages/Azure.Identity) NuGet package to enable authentication:
+To get started with the .NET Aspire Azure PostgreSQL client integration, install the [📦 Aspire.Azure.Npgsql](https://www.nuget.org/packages/Aspire.Azure.Npgsql) NuGet package in the client-consuming project, that is, the project for the application that uses the PostgreSQL client. The PostgreSQL client integration registers an [NpgsqlDataSource](https://www.npgsql.org/doc/api/Npgsql.NpgsqlDataSource.html) instance that you can use to interact with PostgreSQL.
### [.NET CLI](#tab/dotnet-cli)
```dotnetcli
-dotnet add package Azure.Identity
+dotnet add package Aspire.Azure.Npgsql
```
### [PackageReference](#tab/package-reference)
```xml
-
```
---
-The PostgreSQL connection can be consumed using the client integration and :
+
+
+The PostgreSQL connection can be consumed using the client integration by calling the `AddAzureNpgsqlDataSource`:
+
+```csharp
+builder.AddAzureNpgsqlDataSource(connectionName: "postgresdb");
+```
+
+> [!TIP]
+> The `connectionName` parameter must match the name used when adding the PostgreSQL server resource in the app host project.
+
+The preceding code snippet demonstrates how to use the `AddAzureNpgsqlDataSource` method to register an `NpgsqlDataSource` instance that uses Azure authentication ([Microsoft Entra ID](/azure/postgresql/flexible-server/concepts-azure-ad-authentication)). This `"postgresdb"` connection name corresponds to a connection string configuration value.
+
+After adding `NpgsqlDataSource` to the builder, you can get the `NpgsqlDataSource` instance using dependency injection. For example, to retrieve your data source object from an example service define it as a constructor parameter and ensure the `ExampleService` class is registered with the dependency injection container:
+
+```csharp
+public class ExampleService(NpgsqlDataSource dataSource)
+{
+ // Use dataSource...
+}
+```
+
+For more information on dependency injection, see [.NET dependency injection](/dotnet/core/extensions/dependency-injection).
+
+### Add keyed Azure Npgsql client
+
+
+
+There might be situations where you want to register multiple `NpgsqlDataSource` instances with different connection names. To register keyed Npgsql clients, call the `AddKeyedAzureNpgsqlDataSource` method:
+
+```csharp
+builder.AddKeyedAzureNpgsqlDataSource(name: "sales_db");
+builder.AddKeyedAzureNpgsqlDataSource(name: "inventory_db");
+```
+
+Then you can retrieve the `NpgsqlDataSource` instances using dependency injection. For example, to retrieve the connection from an example service:
+
+```csharp
+public class ExampleService(
+ [FromKeyedServices("sales_db")] NpgsqlDataSource salesDataSource,
+ [FromKeyedServices("inventory_db")] NpgsqlDataSource inventoryDataSource)
+{
+ // Use data sources...
+}
+```
+
+For more information on keyed services, see [.NET dependency injection: Keyed services](/dotnet/core/extensions/dependency-injection#keyed-services).
+
+#### Configuration
+
+The .NET Aspire Azure Npgsql integration provides multiple options to configure the database connection based on the requirements and conventions of your project.
+
+##### Use a connection string
+
+When using a connection string defined in the `ConnectionStrings` configuration section, you provide the name of the connection string when calling `AddAzureNpgsqlDataSource`:
```csharp
-builder.AddNpgsqlDataSource(
- "postgresdb",
- configureDataSourceBuilder: (dataSourceBuilder) =>
+builder.AddAzureNpgsqlDataSource("postgresdb");
+```
+
+The connection string is retrieved from the `ConnectionStrings` configuration section, for example, consider the following JSON configuration:
+
+```json
{
- if (string.IsNullOrEmpty(dataSourceBuilder.ConnectionStringBuilder.Password))
- {
- var credentials = new DefaultAzureCredential();
- var tokenRequest = new TokenRequestContext(["https://ossrdbms-aad.database.windows.net/.default"]);
-
- dataSourceBuilder.UsePasswordProvider(
- passwordProvider: _ => credentials.GetToken(tokenRequest).Token,
- passwordProviderAsync: async (_, ct) => (await credentials.GetTokenAsync(tokenRequest, ct)).Token);
+ "ConnectionStrings": {
+ "postgresdb": "Host=myserver;Database=test"
+ }
+}
+```
+
+For more information on how to configure the connection string, see the [Npgsql connection string documentation](https://www.npgsql.org/doc/connection-string-parameters.html).
+
+> [!NOTE]
+> The username and password are automatically inferred from the credential provided in the settings.
+
+##### Use configuration providers
+
+
+
+The .NET Aspire Azure Npgsql integration supports . It loads the `AzureNpgsqlSettings` from configuration using the `Aspire:Azure:Npgsql` key. For example, consider the following _appsettings.json_ file that configures some of the available options:
+
+```json
+{
+ "Aspire": {
+ "Npgsql": {
+ "DisableHealthChecks": true,
+ "DisableTracing": true
}
-});
+ }
+}
```
-The preceding code snippet demonstrates how to use the class from the package to authenticate with [Microsoft Entra ID](/azure/postgresql/flexible-server/concepts-azure-ad-authentication) and retrieve a token to connect to the PostgreSQL database. The [UsePasswordProvider](https://www.npgsql.org/doc/api/Npgsql.NpgsqlDataSourceBuilder.html#Npgsql_NpgsqlDataSourceBuilder_UsePasswordProvider_System_Func_Npgsql_NpgsqlConnectionStringBuilder_System_String__System_Func_Npgsql_NpgsqlConnectionStringBuilder_System_Threading_CancellationToken_System_Threading_Tasks_ValueTask_System_String___) method is used to provide the token to the data source builder.
+##### Use inline delegates
+
+You can configure settings in code, by passing the `Action configureSettings` delegate to set up some or all the options inline, for example to disable health checks from code:
+
+```csharp
+builder.AddAzureNpgsqlDataSource(
+ "postgresdb",
+ settings => settings.DisableHealthChecks = true);
+```
+
+
+
+Use the `AzureNpgsqlSettings.Credential` property to establish a connection. If no credential is configured, the is used. When the connection string contains a username and password, the credential is ignored.
+
+[!INCLUDE [client-integration-health-checks](../../includes/client-integration-health-checks.md)]
+
+- Adds the [`NpgSqlHealthCheck`](https://github.com/Xabaril/AspNetCore.Diagnostics.HealthChecks/blob/master/src/HealthChecks.NpgSql/NpgSqlHealthCheck.cs), which verifies that commands can be successfully executed against the underlying Postgres database.
+- Integrates with the `/health` HTTP endpoint, which specifies all registered health checks must pass for app to be considered ready to accept traffic
+
+[!INCLUDE [integration-observability-and-telemetry](../../includes/integration-observability-and-telemetry.md)]
+
+#### Logging
+
+The .NET Aspire PostgreSQL integration uses the following log categories:
+
+- `Npgsql.Connection`
+- `Npgsql.Command`
+- `Npgsql.Transaction`
+- `Npgsql.Copy`
+- `Npgsql.Replication`
+- `Npgsql.Exception`
+
+#### Tracing
+
+The .NET Aspire PostgreSQL integration will emit the following tracing activities using OpenTelemetry:
+
+- `Npgsql`
+
+#### Metrics
+
+The .NET Aspire PostgreSQL integration will emit the following metrics using OpenTelemetry:
+
+- Npgsql:
+ - `ec_Npgsql_bytes_written_per_second`
+ - `ec_Npgsql_bytes_read_per_second`
+ - `ec_Npgsql_commands_per_second`
+ - `ec_Npgsql_total_commands`
+ - `ec_Npgsql_current_commands`
+ - `ec_Npgsql_failed_commands`
+ - `ec_Npgsql_prepared_commands_ratio`
+ - `ec_Npgsql_connection_pools`
+ - `ec_Npgsql_multiplexing_average_commands_per_batch`
+ - `ec_Npgsql_multiplexing_average_write_time_per_batch`
diff --git a/docs/database/includes/azure-postgresql-ef-client.md b/docs/database/includes/azure-postgresql-ef-client.md
index 6cedb1b170..a97ca201c6 100644
--- a/docs/database/includes/azure-postgresql-ef-client.md
+++ b/docs/database/includes/azure-postgresql-ef-client.md
@@ -2,66 +2,263 @@
ms.topic: include
---
-### Add Azure authenticated Npgsql client
-
-By default, when you call `AddAzurePostgresFlexibleServer` in your PostgreSQL hosting integration, it requires [📦 Azure.Identity](https://www.nuget.org/packages/Azure.Identity) NuGet package to enable authentication:
+To get started with the .NET Aspire PostgreSQL Entity Framework Core client integration, install the [📦 Aspire.Azure.Npgsql.EntityFrameworkCore.PostgreSQL](https://www.nuget.org/packages/) NuGet package in the client-consuming project, that is, the project for the application that uses the PostgreSQL client. The .NET Aspire PostgreSQL Entity Framework Core client integration registers your desired `DbContext` subclass instances that you can use to interact with PostgreSQL.
### [.NET CLI](#tab/dotnet-cli)
```dotnetcli
-dotnet add package Azure.Identity
+dotnet add package Aspire.Azure.Npgsql.EntityFrameworkCore.PostgreSQL
```
### [PackageReference](#tab/package-reference)
```xml
-
```
---
-The PostgreSQL connection can be consumed using the client integration and .
+
+
+The PostgreSQL connection can be consumed using the client integration by calling the `AddAzureNpgsqlDataSource`:
+
+```csharp
+builder.AddAzureNpgsqlDbContext(connectionName: "postgresdb");
+```
+
+> [!TIP]
+> The `connectionName` parameter must match the name used when adding the PostgreSQL server resource in the app host project.
-The following code snippets demonstrate how to use the class from the package to authenticate with [Microsoft Entra ID](/azure/postgresql/flexible-server/concepts-azure-ad-authentication) and retrieve a token to connect to the PostgreSQL database. The [UsePasswordProvider](https://www.npgsql.org/doc/api/Npgsql.NpgsqlDataSourceBuilder.html#Npgsql_NpgsqlDataSourceBuilder_UsePasswordProvider_System_Func_Npgsql_NpgsqlConnectionStringBuilder_System_String__System_Func_Npgsql_NpgsqlConnectionStringBuilder_System_Threading_CancellationToken_System_Threading_Tasks_ValueTask_System_String___) method is used to provide the token to the data source builder.
+The preceding code snippet demonstrates how to use the `AddAzureNpgsqlDbContext` method to register an `YourDbContext` (that's [pooled for performance](/ef/core/performance/advanced-performance-topics)) instance that uses Azure authentication ([Microsoft Entra ID](/azure/postgresql/flexible-server/concepts-azure-ad-authentication)). This `"postgresdb"` connection name corresponds to a connection string configuration value.
-### EF Core version 8
+After adding `YourDbContext` to the builder, you can get the `YourDbContext` instance using dependency injection. For example, to retrieve your data source object from an example service define it as a constructor parameter and ensure the `ExampleService` class is registered with the dependency injection container:
```csharp
-var dsBuilder = new NpgsqlDataSourceBuilder(builder.Configuration.GetConnectionString("postgresdb"));
-if (string.IsNullOrEmpty(dsBuilder.ConnectionStringBuilder.Password))
+public class ExampleService(YourDbContext context)
{
- var credentials = new DefaultAzureCredential();
- var tokenRequest = new TokenRequestContext(["https://ossrdbms-aad.database.windows.net/.default"]);
+ // Use context...
+}
+```
+
+For more information on dependency injection, see [.NET dependency injection](/dotnet/core/extensions/dependency-injection).
+
+### Enrich an Npgsql database context
+
+You may prefer to use the standard Entity Framework method to obtain a database context and add it to the dependency injection container:
+
+```csharp
+builder.Services.AddDbContext(options =>
+ options.UseNpgsql(builder.Configuration.GetConnectionString("postgresdb")
+ ?? throw new InvalidOperationException("Connection string 'postgresdb' not found.")));
+```
+
+> [!NOTE]
+> The connection string name that you pass to the method must match the name used when adding the PostgreSQL server resource in the app host project. For more information, see [Add PostgreSQL server resource](#add-postgresql-server-resource).
+
+You have more flexibility when you create the database context in this way, for example:
+
+- You can reuse existing configuration code for the database context without rewriting it for .NET Aspire.
+- You can use Entity Framework Core interceptors to modify database operations.
+- You can choose not to use Entity Framework Core context pooling, which may perform better in some circumstances.
+
+If you use this method, you can enhance the database context with .NET Aspire-style retries, health checks, logging, and telemetry features by calling the `EnrichAzureNpgsqlDbContext` method:
+
+```csharp
+builder.EnrichAzureNpgsqlDbContext(
+ configureSettings: settings =>
+ {
+ settings.DisableRetry = false;
+ settings.CommandTimeout = 30;
+ });
+```
+
+The `settings` parameter is an instance of the `AzureNpgsqlEntityFrameworkCorePostgreSQLSettings` class.
- dsBuilder.UsePasswordProvider(
- passwordProvider: _ => credentials.GetToken(tokenRequest).Token,
- passwordProviderAsync: async (_, ct) => (await credentials.GetTokenAsync(tokenRequest, ct)).Token);
+
+
+You might also need to configure specific options of Npgsql, or register a in other ways. In this case, you do so by calling the `EnrichAzureNpgsqlDbContext` extension method, as shown in the following example:
+
+```csharp
+var connectionString = builder.Configuration.GetConnectionString("postgresdb");
+
+builder.Services.AddDbContextPool(
+ dbContextOptionsBuilder => dbContextOptionsBuilder.UseNpgsql(connectionString));
+
+builder.EnrichAzureNpgsqlDbContext();
+```
+
+#### Configuration
+
+The .NET Aspire Azure PostgreSQL EntityFrameworkCore Npgsql integration provides multiple options to configure the database connection based on the requirements and conventions of your project.
+
+##### Use a connection string
+
+When using a connection string defined in the `ConnectionStrings` configuration section, you provide the name of the connection string when calling `AddAzureNpgsqlDataSource`:
+
+```csharp
+builder.AddAzureNpgsqlDbContext("postgresdb");
+```
+
+The connection string is retrieved from the `ConnectionStrings` configuration section, for example, consider the following JSON configuration:
+
+```json
+{
+ "ConnectionStrings": {
+ "postgresdb": "Host=myserver;Database=test"
+ }
}
+```
-builder.AddNpgsqlDbContext(
- "postgresdb",
- configureDbContextOptions: (options) => options.UseNpgsql(dsBuilder.Build()));
+For more information on how to configure the connection string, see the [Npgsql connection string documentation](https://www.npgsql.org/doc/connection-string-parameters.html).
+
+> [!NOTE]
+> The username and password are automatically inferred from the credential provided in the settings.
+
+##### Use configuration providers
+
+
+
+The .NET Aspire Azure PostgreSQL EntityFrameworkCore Npgsql integration supports . It loads the `AzureNpgsqlEntityFrameworkCorePostgreSQLSettings` from configuration using the `Aspire:Npgsql:EntityFrameworkCore:PostgreSQL` key. For example, consider the following _appsettings.json_ file that configures some of the available options:
+
+```json
+{
+ "Aspire": {
+ "Npgsql": {
+ "EntityFrameworkCore": {
+ "PostgreSQL": {
+ "DisableHealthChecks": true,
+ "DisableTracing": true
+ }
+ }
+ }
+ }
+}
```
-### EF Core version 9+
+##### Use inline delegates
-With EF Core version 9, you can use the `ConfigureDataSource` method to configure the `NpgsqlDataSourceBuilder` that's used by the integration instead of building one outside of the integration and passing it in.
+You can configure settings in code, by passing the `Action configureSettings` delegate to set up some or all the options inline, for example to disable health checks from code:
```csharp
-builder.AddNpgsqlDbContext(
+builder.AddAzureNpgsqlDbContext(
"postgresdb",
- configureDbContextOptions: (options) => options.UseNpgsql(npgsqlOptions =>
- npgsqlOptions.ConfigureDataSource(dsBuilder =>
- {
- if (string.IsNullOrEmpty(dsBuilder.ConnectionStringBuilder.Password))
- {
- var credentials = new DefaultAzureCredential();
- var tokenRequest = new TokenRequestContext(["https://ossrdbms-aad.database.windows.net/.default"]);
-
- dsBuilder.UsePasswordProvider(
- passwordProvider: _ => credentials.GetToken(tokenRequest).Token,
- passwordProviderAsync: async (_, ct) => (await credentials.GetTokenAsync(tokenRequest, ct)).Token);
- }
- })));
+ settings => settings.DisableHealthChecks = true);
+```
+
+Alternatively, you can use the `EnrichAzureNpgsqlDbContext` extension method to configure the settings:
+
+```csharp
+builder.EnrichAzureNpgsqlDbContext(
+ settings => settings.DisableHealthChecks = true);
+```
+
+
+
+Use the `AzureNpgsqlEntityFrameworkCorePostgreSQLSettings.Credential` property to establish a connection. If no credential is configured, the is used.
+
+When the connection string contains a username and password, the credential is ignored.
+
+##### Troubleshooting
+
+In the rare case that the `Username` property isn't provided and the integration can't detect it using the application's Managed Identity, Npgsql throws an exception with a message similar to the following:
+
+> Npgsql.PostgresException (0x80004005): 28P01: password authentication failed for user ...
+
+In this case you can configure the `Username` property in the connection string and use `EnrichAzureNpgsqlDbContext`, passing the connection string in `UseNpgsql`:
+
+```csharp
+builder.Services.AddDbContextPool(
+ options => options.UseNpgsql(newConnectionString));
+
+builder.EnrichAzureNpgsqlDbContext();
```
+
+#### Configure multiple DbContext classes
+
+If you want to register more than one with different configuration, you can use `$"Aspire:Npgsql:EntityFrameworkCore:PostgreSQL:{typeof(TContext).Name}"` configuration section name. The json configuration would look like:
+
+```json
+{
+ "Aspire": {
+ "Npgsql": {
+ "EntityFrameworkCore": {
+ "PostgreSQL": {
+ "ConnectionString": "",
+ "DisableHealthChecks": true,
+ "DisableTracing": true,
+ "AnotherDbContext": {
+ "ConnectionString": "",
+ "DisableTracing": false
+ }
+ }
+ }
+ }
+ }
+}
+```
+
+Then calling the method with `AnotherDbContext` type parameter would load the settings from `Aspire:Npgsql:EntityFrameworkCore:PostgreSQL:AnotherDbContext` section.
+
+```csharp
+builder.AddAzureNpgsqlDbContext();
+```
+
+[!INCLUDE [client-integration-health-checks](../../includes/client-integration-health-checks.md)]
+
+By default, the .NET Aspire PostgreSQL Entity Framework Core integrations handles the following:
+
+- Adds the [`DbContextHealthCheck`](https://github.com/Xabaril/AspNetCore.Diagnostics.HealthChecks/blob/master/src/HealthChecks.NpgSql/NpgSqlHealthCheck.cs), which calls EF Core's method. The name of the health check is the name of the `TContext` type.
+- Integrates with the `/health` HTTP endpoint, which specifies all registered health checks must pass for app to be considered ready to accept traffic
+
+[!INCLUDE [integration-observability-and-telemetry](../../includes/integration-observability-and-telemetry.md)]
+
+#### Logging
+
+The .NET Aspire PostgreSQL Entity Framework Core integration uses the following Log categories:
+
+- `Microsoft.EntityFrameworkCore.ChangeTracking`
+- `Microsoft.EntityFrameworkCore.Database.Command`
+- `Microsoft.EntityFrameworkCore.Database.Connection`
+- `Microsoft.EntityFrameworkCore.Database.Transaction`
+- `Microsoft.EntityFrameworkCore.Migrations`
+- `Microsoft.EntityFrameworkCore.Infrastructure`
+- `Microsoft.EntityFrameworkCore.Migrations`
+- `Microsoft.EntityFrameworkCore.Model`
+- `Microsoft.EntityFrameworkCore.Model.Validation`
+- `Microsoft.EntityFrameworkCore.Query`
+- `Microsoft.EntityFrameworkCore.Update`
+
+#### Tracing
+
+The .NET Aspire PostgreSQL Entity Framework Core integration will emit the following tracing activities using OpenTelemetry:
+
+- `Npgsql`
+
+#### Metrics
+
+The .NET Aspire PostgreSQL Entity Framework Core integration will emit the following metrics using OpenTelemetry:
+
+- Microsoft.EntityFrameworkCore:
+ - `ec_Microsoft_EntityFrameworkCore_active_db_contexts`
+ - `ec_Microsoft_EntityFrameworkCore_total_queries`
+ - `ec_Microsoft_EntityFrameworkCore_queries_per_second`
+ - `ec_Microsoft_EntityFrameworkCore_total_save_changes`
+ - `ec_Microsoft_EntityFrameworkCore_save_changes_per_second`
+ - `ec_Microsoft_EntityFrameworkCore_compiled_query_cache_hit_rate`
+ - `ec_Microsoft_Entity_total_execution_strategy_operation_failures`
+ - `ec_Microsoft_E_execution_strategy_operation_failures_per_second`
+ - `ec_Microsoft_EntityFramew_total_optimistic_concurrency_failures`
+ - `ec_Microsoft_EntityF_optimistic_concurrency_failures_per_second`
+
+- Npgsql:
+ - `ec_Npgsql_bytes_written_per_second`
+ - `ec_Npgsql_bytes_read_per_second`
+ - `ec_Npgsql_commands_per_second`
+ - `ec_Npgsql_total_commands`
+ - `ec_Npgsql_current_commands`
+ - `ec_Npgsql_failed_commands`
+ - `ec_Npgsql_prepared_commands_ratio`
+ - `ec_Npgsql_connection_pools`
+ - `ec_Npgsql_multiplexing_average_commands_per_batch`
+ - `ec_Npgsql_multiplexing_average_write_time_per_batch`
diff --git a/docs/database/includes/cosmos-app-host.md b/docs/database/includes/cosmos-app-host.md
index 89d7635890..268ffff328 100644
--- a/docs/database/includes/cosmos-app-host.md
+++ b/docs/database/includes/cosmos-app-host.md
@@ -5,6 +5,8 @@ ms.topic: include
The .NET Aspire [Azure Cosmos DB](https://azure.microsoft.com/services/cosmos-db/) hosting integration models the various Cosmos DB resources as the following types:
- : Represents an Azure Cosmos DB resource.
+- : Represents an Azure Cosmos DB container resource.
+- : Represents an Azure Cosmos DB database resource.
- : Represents an Azure Cosmos DB emulator resource.
To access these types and APIs for expressing them, add the [📦 Aspire.Hosting.Azure.CosmosDB](https://www.nuget.org/packages/Aspire.Hosting.Azure.CosmosDB) NuGet package in the [app host](xref:dotnet/aspire/app-host) project.
@@ -49,13 +51,11 @@ If you're new to [Bicep](/azure/azure-resource-manager/bicep/overview), it's a d
:::code language="bicep" source="../../snippets/azure/AppHost/cosmos.module.bicep":::
-The preceding Bicep is a module that provisions an Azure Cosmos DB account with the following defaults:
+The preceding Bicep is a module that provisions an Azure Cosmos DB account resource. Additionally, role assignments are created for the Azure resource in a separate module:
-- `kind`: The kind of Cosmos DB account. The default is `GlobalDocumentDB`.
-- `consistencyPolicy`: The consistency policy of the Cosmos DB account. The default is `Session`.
-- `locations`: The locations for the Cosmos DB account. The default is the resource group's location.
+:::code language="bicep" source="../../snippets/azure/AppHost/cosmos-roles.module.bicep":::
-In addition to the Cosmos DB account, it also adds the current application to the `Data Contributor` role for the Cosmos DB account. The generated Bicep is a starting point and is influenced by changes to the provisioning infrastructure in C#. Customizations to the Bicep file directly will be overwritten, so make changes through the C# provisioning APIs to ensure they are reflected in the generated files.
+The generated Bicep is a starting point and is influenced by changes to the provisioning infrastructure in C#. Customizations to the Bicep file directly will be overwritten, so make changes through the C# provisioning APIs to ensure they are reflected in the generated files.
#### Customize provisioning infrastructure
@@ -105,13 +105,15 @@ The dependent resource can access the injected connection string by calling the
### Add Azure Cosmos DB database and container resources
+.NET Aspire models parent child relationships between Azure Cosmos DB resources. For example, an Azure Cosmos DB account () can have multiple databases (), and each database can have multiple containers (). When you add a database or container resource, you do so on a parent resource.
+
To add an Azure Cosmos DB database resource, call the method on an `IResourceBuilder` instance:
```csharp
var builder = DistributedApplication.CreateBuilder(args);
var cosmos = builder.AddAzureCosmosDB("cosmos-db");
-cosmos.AddCosmosDatabase("db");
+var db = cosmos.AddCosmosDatabase("db");
// After adding all resources, run the app...
```
@@ -127,14 +129,44 @@ var builder = DistributedApplication.CreateBuilder(args);
var cosmos = builder.AddAzureCosmosDB("cosmos-db");
var db = cosmos.AddCosmosDatabase("db");
-db.AddContainer("entries", "/id");
+var container = db.AddContainer("entries", "/id");
// After adding all resources, run the app...
```
-The container is created in the database that's represented by the `AzureCosmosDBDatabaseResource` that you added earlier.
+The container is created in the database that's represented by the `AzureCosmosDBDatabaseResource` that you added earlier. For more information, see [Databases, containers, and items in Azure Cosmos DB](/azure/cosmos-db/resource-model).
+
+#### Parent child resource relationship example
+
+To better understand the parent-child relationship between Azure Cosmos DB resources, consider the following example, which demonstrates adding an Azure Cosmos DB resource along with a database and container:
+
+```csharp
+var builder = DistributedApplication.CreateBuilder(args);
+
+var cosmos = builder.AddAzureCosmosDB("cosmos");
+
+var customers = cosmos.AddCosmosDatabase("customers");
+var profiles = customers.AddContainer("profiles", "/id");
+
+var orders = cosmos.AddCosmosDatabase("orders");
+var details = orders.AddContainer("details", "/id");
+var history = orders.AddContainer("history", "/id");
+
+builder.AddProject("api")
+ .WithReference(profiles)
+ .WithReference(details)
+ .WithReference(history);
+
+builder.Build().Run();
+```
+
+The preceding code adds an Azure Cosmos DB resource named `cosmos` with two databases: `customers` and `orders`. The `customers` database has a single container named `profiles`, while the `orders` database has two containers: `details` and `history`. The partition key for each container is `/id`.
+
+The following diagram illustrates the parent child relationship between the Azure Cosmos DB resources:
+
+:::image type="content" source="media/cosmos-resource-relationships-thumb.png" alt-text="A diagram depicting Azure Cosmos DB resource parent child relationships." lightbox="media/cosmos-resource-relationships.png":::
-For more information, see [Databases, containers, and items in Azure Cosmos DB](/azure/cosmos-db/resource-model).
+When your app host code expresses parent-child relationships, the client can deep-link to these resources by name. For example, the `customers` database can be referenced by name in the client project, registering a instance that connects to the `customers` database. The same applies to named containers, for example, the `details` container can be referenced by name in the client project, registering a instance that connects to the `details` container.
### Add Azure Cosmos DB emulator resource
diff --git a/docs/database/includes/media/cosmos-resource-relationships-thumb.png b/docs/database/includes/media/cosmos-resource-relationships-thumb.png
new file mode 100644
index 0000000000..3a547fa1fa
Binary files /dev/null and b/docs/database/includes/media/cosmos-resource-relationships-thumb.png differ
diff --git a/docs/database/includes/media/cosmos-resource-relationships.excalidraw b/docs/database/includes/media/cosmos-resource-relationships.excalidraw
new file mode 100644
index 0000000000..0076064c28
--- /dev/null
+++ b/docs/database/includes/media/cosmos-resource-relationships.excalidraw
@@ -0,0 +1,1017 @@
+{
+ "type": "excalidraw",
+ "version": 2,
+ "source": "https://excalidraw.com",
+ "elements": [
+ {
+ "id": "BetIdnuJVAsEvfFGxv3z6",
+ "type": "rectangle",
+ "x": 348.0752632417392,
+ "y": 75.95778493021734,
+ "width": 222.2791527694995,
+ "height": 71.7333455034731,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "#a5d8ff",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "a2",
+ "roundness": {
+ "type": 3
+ },
+ "seed": 2079267916,
+ "version": 165,
+ "versionNonce": 1652020980,
+ "isDeleted": false,
+ "boundElements": [
+ {
+ "type": "text",
+ "id": "BXbw_yfxj0B93dXb_UgFq"
+ },
+ {
+ "id": "chAR-1rsc7Lz5cSklA66f",
+ "type": "arrow"
+ },
+ {
+ "id": "KAeYgvQLsk7g-2wwXW4H4",
+ "type": "arrow"
+ }
+ ],
+ "updated": 1743532088033,
+ "link": null,
+ "locked": false
+ },
+ {
+ "id": "BXbw_yfxj0B93dXb_UgFq",
+ "type": "text",
+ "x": 369.73493545168424,
+ "y": 84.8244576819539,
+ "width": 178.95980834960938,
+ "height": 54,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 1,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "a3",
+ "roundness": null,
+ "seed": 2008317556,
+ "version": 193,
+ "versionNonce": 866699340,
+ "isDeleted": false,
+ "boundElements": [],
+ "updated": 1743532086308,
+ "link": null,
+ "locked": false,
+ "text": "Customer Database\n\"customers\"",
+ "fontSize": 20,
+ "fontFamily": 6,
+ "textAlign": "center",
+ "verticalAlign": "middle",
+ "containerId": "BetIdnuJVAsEvfFGxv3z6",
+ "originalText": "Customer Database\n\"customers\"",
+ "autoResize": true,
+ "lineHeight": 1.35
+ },
+ {
+ "id": "ZU3FetO0xJojd3Oha8XrC",
+ "type": "rectangle",
+ "x": 900.1514216527121,
+ "y": 75.31335742581805,
+ "width": 214.6578694645427,
+ "height": 71.7333455034731,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "#ffc9c9",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "a4",
+ "roundness": {
+ "type": 3
+ },
+ "seed": 584954572,
+ "version": 238,
+ "versionNonce": 2111329524,
+ "isDeleted": false,
+ "boundElements": [
+ {
+ "type": "text",
+ "id": "jQ2fd3ucDH63pjUJIvctu"
+ },
+ {
+ "id": "qpjk7KGMDPmEHp3VGv9m7",
+ "type": "arrow"
+ },
+ {
+ "id": "09RFhqle1PAya4XMEkV8R",
+ "type": "arrow"
+ }
+ ],
+ "updated": 1743532141629,
+ "link": null,
+ "locked": false
+ },
+ {
+ "id": "jQ2fd3ucDH63pjUJIvctu",
+ "type": "text",
+ "x": 935.6404287116436,
+ "y": 84.18003017755461,
+ "width": 143.6798553466797,
+ "height": 54,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 1,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "a5",
+ "roundness": null,
+ "seed": 89551220,
+ "version": 269,
+ "versionNonce": 1286166516,
+ "isDeleted": false,
+ "boundElements": [],
+ "updated": 1743531824565,
+ "link": null,
+ "locked": false,
+ "text": "Order Database\n\"orders\"",
+ "fontSize": 20,
+ "fontFamily": 6,
+ "textAlign": "center",
+ "verticalAlign": "middle",
+ "containerId": "ZU3FetO0xJojd3Oha8XrC",
+ "originalText": "Order Database\n\"orders\"",
+ "autoResize": true,
+ "lineHeight": 1.35
+ },
+ {
+ "id": "une2hoBfhcrggQ1PhV1tR",
+ "type": "rectangle",
+ "x": 286.06458413579264,
+ "y": 276.7456465575891,
+ "width": 345.01165597259416,
+ "height": 71.7333455034731,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "#96f2d7",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "a6",
+ "roundness": {
+ "type": 3
+ },
+ "seed": 1286961484,
+ "version": 170,
+ "versionNonce": 82919924,
+ "isDeleted": false,
+ "boundElements": [
+ {
+ "type": "text",
+ "id": "u7LSObHA8IaZai9QWKedt"
+ },
+ {
+ "id": "KAeYgvQLsk7g-2wwXW4H4",
+ "type": "arrow"
+ }
+ ],
+ "updated": 1743532135042,
+ "link": null,
+ "locked": false
+ },
+ {
+ "id": "u7LSObHA8IaZai9QWKedt",
+ "type": "text",
+ "x": 331.62054487355454,
+ "y": 285.61231930932564,
+ "width": 253.8997344970703,
+ "height": 54,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 1,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "a7",
+ "roundness": null,
+ "seed": 1935123572,
+ "version": 192,
+ "versionNonce": 1193860084,
+ "isDeleted": false,
+ "boundElements": [],
+ "updated": 1743532121611,
+ "link": null,
+ "locked": false,
+ "text": "Customer Profiles Container\n\"profiles\"",
+ "fontSize": 20,
+ "fontFamily": 6,
+ "textAlign": "center",
+ "verticalAlign": "middle",
+ "containerId": "une2hoBfhcrggQ1PhV1tR",
+ "originalText": "Customer Profiles Container\n\"profiles\"",
+ "autoResize": true,
+ "lineHeight": 1.35
+ },
+ {
+ "id": "9D8gv-h8iSsVeNTfNzb5y",
+ "type": "rectangle",
+ "x": 686.051613873488,
+ "y": 276.10121905318977,
+ "width": 291.03959772504965,
+ "height": 71.7333455034731,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "#fff9db",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "a8",
+ "roundness": {
+ "type": 3
+ },
+ "seed": 617639884,
+ "version": 166,
+ "versionNonce": 1604913908,
+ "isDeleted": false,
+ "boundElements": [
+ {
+ "type": "text",
+ "id": "14vJvqbbZIuVUVPitvLzU"
+ }
+ ],
+ "updated": 1743532078594,
+ "link": null,
+ "locked": false
+ },
+ {
+ "id": "14vJvqbbZIuVUVPitvLzU",
+ "type": "text",
+ "x": 724.6215302286886,
+ "y": 284.9678918049263,
+ "width": 213.89976501464844,
+ "height": 54,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 1,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "a9",
+ "roundness": null,
+ "seed": 585635700,
+ "version": 198,
+ "versionNonce": 1157176308,
+ "isDeleted": false,
+ "boundElements": [],
+ "updated": 1743531877844,
+ "link": null,
+ "locked": false,
+ "text": "Order Details Container\n\"details\"",
+ "fontSize": 20,
+ "fontFamily": 6,
+ "textAlign": "center",
+ "verticalAlign": "middle",
+ "containerId": "9D8gv-h8iSsVeNTfNzb5y",
+ "originalText": "Order Details Container\n\"details\"",
+ "autoResize": true,
+ "lineHeight": 1.35
+ },
+ {
+ "id": "nEB5lm8By6Meay1Rg8Jlt",
+ "type": "rectangle",
+ "x": 1032.0665601979138,
+ "y": 276.7456465575891,
+ "width": 293.62341305430624,
+ "height": 71.7333455034731,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "#fff5f5",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "aA",
+ "roundness": {
+ "type": 3
+ },
+ "seed": 1531483724,
+ "version": 166,
+ "versionNonce": 1670389324,
+ "isDeleted": false,
+ "boundElements": [
+ {
+ "type": "text",
+ "id": "Lp9UV-3QEa5eptlCOFk5n"
+ },
+ {
+ "id": "09RFhqle1PAya4XMEkV8R",
+ "type": "arrow"
+ }
+ ],
+ "updated": 1743532172121,
+ "link": null,
+ "locked": false
+ },
+ {
+ "id": "Lp9UV-3QEa5eptlCOFk5n",
+ "type": "text",
+ "x": 1070.7583707900083,
+ "y": 285.61231930932564,
+ "width": 216.2397918701172,
+ "height": 54,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 1,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "aB",
+ "roundness": null,
+ "seed": 1765263988,
+ "version": 186,
+ "versionNonce": 2069938380,
+ "isDeleted": false,
+ "boundElements": [],
+ "updated": 1743531882885,
+ "link": null,
+ "locked": false,
+ "text": "Order History Container\n\"history\"",
+ "fontSize": 20,
+ "fontFamily": 6,
+ "textAlign": "center",
+ "verticalAlign": "middle",
+ "containerId": "nEB5lm8By6Meay1Rg8Jlt",
+ "originalText": "Order History Container\n\"history\"",
+ "autoResize": true,
+ "lineHeight": 1.35
+ },
+ {
+ "id": "chAR-1rsc7Lz5cSklA66f",
+ "type": "arrow",
+ "x": 626.9796183029686,
+ "y": -70.47443625087315,
+ "width": 172.96645180474576,
+ "height": 145.1276644030608,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "aC",
+ "roundness": {
+ "type": 2
+ },
+ "seed": 217574604,
+ "version": 1062,
+ "versionNonce": 898774476,
+ "isDeleted": false,
+ "boundElements": [
+ {
+ "type": "text",
+ "id": "IXfcsIfo2VgO3GSaGUbxG"
+ }
+ ],
+ "updated": 1743532277789,
+ "link": null,
+ "locked": false,
+ "points": [
+ [
+ 0,
+ 0
+ ],
+ [
+ -115.56509263206794,
+ 43.88317681716592
+ ],
+ [
+ -172.96645180474576,
+ 145.1276644030608
+ ]
+ ],
+ "lastCommittedPoint": null,
+ "startBinding": {
+ "elementId": "jbGlDS-ndKUGUhKtmmFVr",
+ "focus": -0.28932264734562113,
+ "gap": 1
+ },
+ "endBinding": {
+ "elementId": "BetIdnuJVAsEvfFGxv3z6",
+ "focus": -0.19985777113350967,
+ "gap": 3.9878770661829037
+ },
+ "startArrowhead": null,
+ "endArrowhead": "triangle",
+ "elbowed": false
+ },
+ {
+ "id": "IXfcsIfo2VgO3GSaGUbxG",
+ "type": "text",
+ "x": 419.0207187387015,
+ "y": -2.0694712574785257,
+ "width": 79.09992980957031,
+ "height": 27,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 1,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "aD",
+ "roundness": null,
+ "seed": 259439988,
+ "version": 13,
+ "versionNonce": 1772925900,
+ "isDeleted": false,
+ "boundElements": [],
+ "updated": 1743531693628,
+ "link": null,
+ "locked": false,
+ "text": "Contains",
+ "fontSize": 20,
+ "fontFamily": 6,
+ "textAlign": "center",
+ "verticalAlign": "middle",
+ "containerId": "chAR-1rsc7Lz5cSklA66f",
+ "originalText": "Contains",
+ "autoResize": true,
+ "lineHeight": 1.35
+ },
+ {
+ "id": "qpjk7KGMDPmEHp3VGv9m7",
+ "type": "arrow",
+ "x": 882.6586094806715,
+ "y": -70.47443625087317,
+ "width": 133.8355461871546,
+ "height": 142.08858775480422,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "aE",
+ "roundness": {
+ "type": 2
+ },
+ "seed": 729314124,
+ "version": 813,
+ "versionNonce": 1004641356,
+ "isDeleted": false,
+ "boundElements": [
+ {
+ "type": "text",
+ "id": "6aN3bZRSLclWSyXm4EM8N"
+ }
+ ],
+ "updated": 1743532277789,
+ "link": null,
+ "locked": false,
+ "points": [
+ [
+ 0,
+ 0
+ ],
+ [
+ 90.34434994151366,
+ 54.83859189195486
+ ],
+ [
+ 133.8355461871546,
+ 142.08858775480422
+ ]
+ ],
+ "lastCommittedPoint": null,
+ "startBinding": {
+ "elementId": "jbGlDS-ndKUGUhKtmmFVr",
+ "focus": 0.07223642628949603,
+ "gap": 1
+ },
+ "endBinding": {
+ "elementId": "ZU3FetO0xJojd3Oha8XrC",
+ "focus": 0.22950794369676736,
+ "gap": 10.143362895292128
+ },
+ "startArrowhead": null,
+ "endArrowhead": "triangle",
+ "elbowed": false
+ },
+ {
+ "id": "6aN3bZRSLclWSyXm4EM8N",
+ "type": "text",
+ "x": 965.6748614040347,
+ "y": -2.0694712574785257,
+ "width": 79.09992980957031,
+ "height": 27,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 1,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "aF",
+ "roundness": null,
+ "seed": 1389686900,
+ "version": 13,
+ "versionNonce": 1535262452,
+ "isDeleted": false,
+ "boundElements": [],
+ "updated": 1743531693628,
+ "link": null,
+ "locked": false,
+ "text": "Contains",
+ "fontSize": 20,
+ "fontFamily": 6,
+ "textAlign": "center",
+ "verticalAlign": "middle",
+ "containerId": "qpjk7KGMDPmEHp3VGv9m7",
+ "originalText": "Contains",
+ "autoResize": true,
+ "lineHeight": 1.35
+ },
+ {
+ "id": "KAeYgvQLsk7g-2wwXW4H4",
+ "type": "arrow",
+ "x": 458.9506210487843,
+ "y": 148.00344284536283,
+ "width": 0.37993740529771003,
+ "height": 126.4448360534019,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "aG",
+ "roundness": {
+ "type": 2
+ },
+ "seed": 1696868812,
+ "version": 499,
+ "versionNonce": 1096420684,
+ "isDeleted": false,
+ "boundElements": [
+ {
+ "type": "text",
+ "id": "Kmh88U-Rb2hPDgI8gfkk7"
+ }
+ ],
+ "updated": 1743532121612,
+ "link": null,
+ "locked": false,
+ "points": [
+ [
+ 0,
+ 0
+ ],
+ [
+ -0.3799374052976532,
+ 51.97062764952466
+ ],
+ [
+ -0.37993740529771003,
+ 126.4448360534019
+ ]
+ ],
+ "lastCommittedPoint": null,
+ "startBinding": {
+ "elementId": "BetIdnuJVAsEvfFGxv3z6",
+ "focus": -0.0000024430666897761114,
+ "gap": 1
+ },
+ "endBinding": {
+ "elementId": "une2hoBfhcrggQ1PhV1tR",
+ "focus": 0.000001573983905659971,
+ "gap": 9.498936015644063
+ },
+ "startArrowhead": null,
+ "endArrowhead": "triangle",
+ "elbowed": false
+ },
+ {
+ "id": "Kmh88U-Rb2hPDgI8gfkk7",
+ "type": "text",
+ "x": 419.0207187387015,
+ "y": 198.71838974514202,
+ "width": 79.09992980957031,
+ "height": 27,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 1,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "aH",
+ "roundness": null,
+ "seed": 1686266740,
+ "version": 13,
+ "versionNonce": 1140611660,
+ "isDeleted": false,
+ "boundElements": [],
+ "updated": 1743531693628,
+ "link": null,
+ "locked": false,
+ "text": "Contains",
+ "fontSize": 20,
+ "fontFamily": 6,
+ "textAlign": "center",
+ "verticalAlign": "middle",
+ "containerId": "KAeYgvQLsk7g-2wwXW4H4",
+ "originalText": "Contains",
+ "autoResize": true,
+ "lineHeight": 1.35
+ },
+ {
+ "id": "S2TmZJ4RNJlXyh3YbdmBe",
+ "type": "arrow",
+ "x": 944.4467255196377,
+ "y": 147.3688785251296,
+ "width": 116.57014473472964,
+ "height": 126.06441154118451,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "aI",
+ "roundness": {
+ "type": 2
+ },
+ "seed": 219484236,
+ "version": 611,
+ "versionNonce": 1851380084,
+ "isDeleted": false,
+ "boundElements": [
+ {
+ "type": "text",
+ "id": "45FDc_REpZ_97kG_BzywH"
+ }
+ ],
+ "updated": 1743531954339,
+ "link": null,
+ "locked": false,
+ "points": [
+ [
+ 0,
+ 0
+ ],
+ [
+ -72.919689567506,
+ 48.09415027229582
+ ],
+ [
+ -116.57014473472964,
+ 126.06441154118451
+ ]
+ ],
+ "lastCommittedPoint": null,
+ "startBinding": null,
+ "endBinding": null,
+ "startArrowhead": null,
+ "endArrowhead": "triangle",
+ "elbowed": false
+ },
+ {
+ "id": "45FDc_REpZ_97kG_BzywH",
+ "type": "text",
+ "x": 792.0219757745863,
+ "y": 198.71838974514202,
+ "width": 79.09992980957031,
+ "height": 27,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 1,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "aJ",
+ "roundness": null,
+ "seed": 1773082228,
+ "version": 13,
+ "versionNonce": 1279672436,
+ "isDeleted": false,
+ "boundElements": [],
+ "updated": 1743531693628,
+ "link": null,
+ "locked": false,
+ "text": "Contains",
+ "fontSize": 20,
+ "fontFamily": 6,
+ "textAlign": "center",
+ "verticalAlign": "middle",
+ "containerId": "S2TmZJ4RNJlXyh3YbdmBe",
+ "originalText": "Contains",
+ "autoResize": true,
+ "lineHeight": 1.35
+ },
+ {
+ "id": "09RFhqle1PAya4XMEkV8R",
+ "type": "arrow",
+ "x": 1068.2713705624049,
+ "y": 147.3688785251296,
+ "width": 113.65573117432155,
+ "height": 126.72916984732885,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "aK",
+ "roundness": {
+ "type": 2
+ },
+ "seed": 52766412,
+ "version": 619,
+ "versionNonce": 903483892,
+ "isDeleted": false,
+ "boundElements": [
+ {
+ "type": "text",
+ "id": "DQjdqQy3_nbOQs5wXxRPs"
+ }
+ ],
+ "updated": 1743532067390,
+ "link": null,
+ "locked": false,
+ "points": [
+ [
+ 0,
+ 0
+ ],
+ [
+ 80.96328401413325,
+ 53.249619474157186
+ ],
+ [
+ 113.65573117432155,
+ 126.72916984732885
+ ]
+ ],
+ "lastCommittedPoint": null,
+ "startBinding": {
+ "elementId": "ZU3FetO0xJojd3Oha8XrC",
+ "focus": -0.03563365791928458,
+ "gap": 1
+ },
+ "endBinding": {
+ "elementId": "nEB5lm8By6Meay1Rg8Jlt",
+ "focus": 0.12400707216599659,
+ "gap": 9.498936015644063
+ },
+ "startArrowhead": null,
+ "endArrowhead": "triangle",
+ "elbowed": false
+ },
+ {
+ "id": "DQjdqQy3_nbOQs5wXxRPs",
+ "type": "text",
+ "x": 1139.3288465407904,
+ "y": 198.71838974514202,
+ "width": 79.09992980957031,
+ "height": 27,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 1,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "index": "aL",
+ "roundness": null,
+ "seed": 2072709492,
+ "version": 12,
+ "versionNonce": 848507084,
+ "isDeleted": false,
+ "boundElements": [],
+ "updated": 1743531693628,
+ "link": null,
+ "locked": false,
+ "text": "Contains",
+ "fontSize": 20,
+ "fontFamily": 6,
+ "textAlign": "center",
+ "verticalAlign": "middle",
+ "containerId": "09RFhqle1PAya4XMEkV8R",
+ "originalText": "Contains",
+ "autoResize": true,
+ "lineHeight": 1.35
+ },
+ {
+ "id": "jbGlDS-ndKUGUhKtmmFVr",
+ "type": "rectangle",
+ "x": 586.8288113185816,
+ "y": -254.69543626543532,
+ "width": 332.32608370110074,
+ "height": 183.22100001456215,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "#e7f5ff",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [
+ "XElObUg25iqfyUHyTuTVl"
+ ],
+ "frameId": null,
+ "index": "aM",
+ "roundness": null,
+ "seed": 1229960652,
+ "version": 369,
+ "versionNonce": 591620980,
+ "isDeleted": false,
+ "boundElements": [
+ {
+ "type": "text",
+ "id": "Npgxu_ENRoObT4I3NmU1L"
+ },
+ {
+ "id": "chAR-1rsc7Lz5cSklA66f",
+ "type": "arrow"
+ },
+ {
+ "id": "qpjk7KGMDPmEHp3VGv9m7",
+ "type": "arrow"
+ }
+ ],
+ "updated": 1743532277789,
+ "link": null,
+ "locked": false
+ },
+ {
+ "id": "Npgxu_ENRoObT4I3NmU1L",
+ "type": "text",
+ "x": 626.981973103214,
+ "y": -230.58493625815424,
+ "width": 252.01976013183594,
+ "height": 135,
+ "angle": 0,
+ "strokeColor": "#1e1e1e",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 1,
+ "opacity": 100,
+ "groupIds": [
+ "XElObUg25iqfyUHyTuTVl"
+ ],
+ "frameId": null,
+ "index": "aN",
+ "roundness": null,
+ "seed": 1564593012,
+ "version": 396,
+ "versionNonce": 1082143436,
+ "isDeleted": false,
+ "boundElements": [],
+ "updated": 1743532277789,
+ "link": null,
+ "locked": false,
+ "text": "\n\n\nAzure Cosmos DB Resource\n\"cosmos\"",
+ "fontSize": 20,
+ "fontFamily": 6,
+ "textAlign": "center",
+ "verticalAlign": "middle",
+ "containerId": "jbGlDS-ndKUGUhKtmmFVr",
+ "originalText": "\n\n\nAzure Cosmos DB Resource\n\"cosmos\"",
+ "autoResize": true,
+ "lineHeight": 1.35
+ },
+ {
+ "id": "oUBSF7qo9QB17fZeK8qO8",
+ "type": "image",
+ "x": 706.9228385836044,
+ "y": -239.6199881329922,
+ "width": 83.29079840430518,
+ "height": 83.29079840430518,
+ "angle": 0,
+ "strokeColor": "transparent",
+ "backgroundColor": "#ffffff",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [
+ "XElObUg25iqfyUHyTuTVl"
+ ],
+ "frameId": null,
+ "index": "aO",
+ "roundness": null,
+ "seed": 1869988724,
+ "version": 188,
+ "versionNonce": 753545204,
+ "isDeleted": false,
+ "boundElements": null,
+ "updated": 1743532277789,
+ "link": null,
+ "locked": false,
+ "status": "saved",
+ "fileId": "a4c88afb146a3ec5eabbf5767977de165cc12c32",
+ "scale": [
+ 1,
+ 1
+ ],
+ "crop": null
+ }
+ ],
+ "appState": {
+ "gridSize": 20,
+ "gridStep": 5,
+ "gridModeEnabled": false,
+ "viewBackgroundColor": "#ffffff"
+ },
+ "files": {
+ "a4c88afb146a3ec5eabbf5767977de165cc12c32": {
+ "mimeType": "image/png",
+ "id": "a4c88afb146a3ec5eabbf5767977de165cc12c32",
+ "dataURL": "",
+ "created": 1743532037162,
+ "lastRetrieved": 1743532037162
+ }
+ }
+}
\ No newline at end of file
diff --git a/docs/database/includes/media/cosmos-resource-relationships.png b/docs/database/includes/media/cosmos-resource-relationships.png
new file mode 100644
index 0000000000..f9d29fb4a7
Binary files /dev/null and b/docs/database/includes/media/cosmos-resource-relationships.png differ
diff --git a/docs/database/includes/postgresql-app-host.md b/docs/database/includes/postgresql-app-host.md
index 0e7c08d23f..d00aceb504 100644
--- a/docs/database/includes/postgresql-app-host.md
+++ b/docs/database/includes/postgresql-app-host.md
@@ -44,13 +44,60 @@ var exampleProject = builder.AddProject()
// After adding all resources, run the app...
```
-When .NET Aspire adds a container image to the app host, as shown in the preceding example with the `docker.io/library/postgres` image, it creates a new PostgreSQL server instance on your local machine. A reference to your PostgreSQL server and your PostgreSQL database instance (the `postgresdb` variable) are used to add a dependency to the `ExampleProject`. The PostgreSQL server resource includes default credentials with a `username` of `"postgres"` and randomly generated `password` using the method.
+When .NET Aspire adds a container image to the app host, as shown in the preceding example with the `docker.io/library/postgres` image, it creates a new PostgreSQL server instance on your local machine. A reference to your PostgreSQL server and database instance (the `postgresdb` variable) are used to add a dependency to the `ExampleProject`.
+
+When adding a database resource to the app model, the database is created if it doesn't already exist. The creation of the database relies on the [app host eventing APIs](../../app-host/eventing.md), specifically . In other words, when the `postgres` resource is _ready_, the event is raised and the database resource is created.
+
+The PostgreSQL server resource includes default credentials with a `username` of `"postgres"` and randomly generated `password` using the method.
The method configures a connection in the `ExampleProject` named `"messaging"`. For more information, see [Container resource lifecycle](../../fundamentals/app-host-overview.md#container-resource-lifecycle).
> [!TIP]
> If you'd rather connect to an existing PostgreSQL server, call instead. For more information, see [Reference existing resources](../../fundamentals/app-host-overview.md#reference-existing-resources).
+### Add PostgreSQL resource with database scripts
+
+By default, when you add a , it relies on the following script to create the database:
+
+```sql
+CREATE DATABASE ""
+```
+
+
+
+To alter the default script, chain a call to the `WithCreationScript` method on the database resource builder:
+
+```csharp
+var builder = DistributedApplication.CreateBuilder(args);
+
+var postgres = builder.AddPostgres("postgres");
+
+var databaseName = "app_db";
+var creationScript = $$"""
+ -- Create the database
+ CREATE DATABASE {{databaseName}};
+
+ """;
+
+var db = postgres.AddDatabase(databaseName)
+ .WithCreationScript(creationScript);
+
+builder.AddProject()
+ .WithReference(db)
+ .WaitFor(db);
+
+// After adding all resources, run the app...
+```
+
+The preceding example creates a database named `app_db`. The script is run when the database resource is created. The script is passed as a string to the `WithCreationScript` method, which is then run in the context of the PostgreSQL resource.
+
+> [!NOTE]
+> The connect to a database command (`\c`) isn't supported when using the creation script.
+
### Add PostgreSQL pgAdmin resource
When adding PostgreSQL resources to the `builder` with the `AddPostgres` method, you can chain calls to to add the [**dpage/pgadmin4**](https://www.pgadmin.org/) container. This container is a cross-platform client for PostgreSQL databases, that serves a web-based admin dashboard. Consider the following example:
diff --git a/docs/database/includes/postgresql-flexible-server.md b/docs/database/includes/postgresql-flexible-server.md
index 09e09e08ce..3f570bde06 100644
--- a/docs/database/includes/postgresql-flexible-server.md
+++ b/docs/database/includes/postgresql-flexible-server.md
@@ -62,16 +62,9 @@ If you're new to [Bicep](/azure/azure-resource-manager/bicep/overview), it's a d
:::code language="bicep" source="../../snippets/azure/AppHost/postgres-flexible.module.bicep":::
-The preceding Bicep is a module that provisions an Azure PostgreSQL flexible server with the following defaults:
-
-- `authConfig`: The authentication configuration of the PostgreSQL server. The default is `ActiveDirectoryAuth` enabled and `PasswordAuth` disabled.
-- `availabilityZone`: The availability zone of the PostgreSQL server. The default is `1`.
-- `backup`: The backup configuration of the PostgreSQL server. The default is `BackupRetentionDays` set to `7` and `GeoRedundantBackup` set to `Disabled`.
-- `highAvailability`: The high availability configuration of the PostgreSQL server. The default is `Disabled`.
-- `storage`: The storage configuration of the PostgreSQL server. The default is `StorageSizeGB` set to `32`.
-- `version`: The version of the PostgreSQL server. The default is `16`.
-- `sku`: The SKU of the PostgreSQL server. The default is `Standard_B1ms`.
-- `tags`: The tags of the PostgreSQL server. The default is `aspire-resource-name` set to the name of the Aspire resource, in this case `postgres-flexible`.
+The preceding Bicep is a module that provisions an Azure PostgreSQL flexible server resource. Additionally, role assignments are created for the Azure resource in a separate module:
+
+:::code language="bicep" source="../../snippets/azure/AppHost/postgres-flexible-roles.module.bicep":::
In addition to the PostgreSQL flexible server, it also provisions an Azure Firewall rule to allow all Azure IP addresses. Finally, an administrator is created for the PostgreSQL server, and the connection string is outputted as an output variable. The generated Bicep is a starting point and is influenced by changes to the provisioning infrastructure in C#. Customizations to the Bicep file directly will be overwritten, so make changes through the C# provisioning APIs to ensure they are reflected in the generated files.
diff --git a/docs/database/includes/sql-app-host.md b/docs/database/includes/sql-app-host.md
index 29942842bf..2dbe8dc9dc 100644
--- a/docs/database/includes/sql-app-host.md
+++ b/docs/database/includes/sql-app-host.md
@@ -43,7 +43,11 @@ builder.AddProject()
> [!NOTE]
> The SQL Server container is slow to start, so it's best to use a _persistent_ lifetime to avoid unnecessary restarts. For more information, see [Container resource lifetime](../../fundamentals/app-host-overview.md#container-resource-lifetime).
-When .NET Aspire adds a container image to the app host, as shown in the preceding example with the `mcr.microsoft.com/mssql/server` image, it creates a new SQL Server instance on your local machine. A reference to your SQL Server resource builder (the `sql` variable) is used to add a database. The database is named `database` and then added to the `ExampleProject`. The SQL Server resource includes default credentials with a `username` of `sa` and a random `password` generated using the method.
+When .NET Aspire adds a container image to the app host, as shown in the preceding example with the `mcr.microsoft.com/mssql/server` image, it creates a new SQL Server instance on your local machine. A reference to your SQL Server resource builder (the `sql` variable) is used to add a database. The database is named `database` and then added to the `ExampleProject`.
+
+When adding a database resource to the app model, the database is created if it doesn't already exist. The creation of the database relies on the [app host eventing APIs](../../app-host/eventing.md), specifically . In other words, when the `sql` resource is _ready_, the event is raised and the database resource is created.
+
+The SQL Server resource includes default credentials with a `username` of `sa` and a random `password` generated using the method.
When the app host runs, the password is stored in the app host's secret store. It's added to the `Parameters` section, for example:
@@ -60,6 +64,73 @@ The method conf
> [!TIP]
> If you'd rather connect to an existing SQL Server, call instead. For more information, see [Reference existing resources](../../fundamentals/app-host-overview.md#reference-existing-resources).
+### Add SQL Server resource with database scripts
+
+By default, when you add a , it relies on the following SQL script to create the database:
+
+```sql
+IF
+(
+ NOT EXISTS
+ (
+ SELECT 1
+ FROM sys.databases
+ WHERE name = @DatabaseName
+ )
+)
+CREATE DATABASE [];
+```
+
+
+
+To alter the default script, chain a call to the `WithCreationScript` method on the database resource builder:
+
+```csharp
+var builder = DistributedApplication.CreateBuilder(args);
+
+var sql = builder.AddSqlServer("sql")
+ .WithLifetime(ContainerLifetime.Persistent);
+
+var databaseName = "app_db";
+var creationScript = $$"""
+ IF DB_ID('{{databaseName}}') IS NULL
+ CREATE DATABASE [{{databaseName}}];
+ GO
+
+ -- Use the database
+ USE [{{databaseName}}];
+ GO
+
+ -- Create the todos table
+ CREATE TABLE todos (
+ id INT PRIMARY KEY AUTO_INCREMENT, -- Unique ID for each todo
+ title VARCHAR(255) NOT NULL, -- Short description of the task
+ description TEXT, -- Optional detailed description
+ is_completed BOOLEAN DEFAULT FALSE, -- Completion status
+ due_date DATE, -- Optional due date
+ created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
+ updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP
+ );
+ GO
+
+ """;
+
+var db = sql.AddDatabase(databaseName)
+ .WithCreationScript(creationScript);
+
+builder.AddProject()
+ .WithReference(db)
+ .WaitFor(db);
+
+// After adding all resources, run the app...
+```
+
+The preceding example creates a database named `app_db` with a single `todos` table. The SQL script is executed when the database resource is created. The script is passed as a string to the `WithCreationScript` method, which is then executed in the context of the SQL Server resource.
+
### Add SQL Server resource with data volume
To add a data volume to the SQL Server resource, call the method on the SQL Server resource:
diff --git a/docs/database/snippets/cosmos-db/AspireApp.ApiService/AspireApp.ApiService.csproj b/docs/database/snippets/cosmos-db/AspireApp.ApiService/AspireApp.ApiService.csproj
index 2f5c1eeafd..daa106c456 100644
--- a/docs/database/snippets/cosmos-db/AspireApp.ApiService/AspireApp.ApiService.csproj
+++ b/docs/database/snippets/cosmos-db/AspireApp.ApiService/AspireApp.ApiService.csproj
@@ -7,7 +7,7 @@
-