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..743daf007b 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/01/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 () 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 . 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/includes/cosmos-app-host.md b/docs/database/includes/cosmos-app-host.md index 89d7635890..ec1fc7dd77 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. @@ -105,13 +107,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 +131,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/messaging/azure-service-bus-integration.md b/docs/messaging/azure-service-bus-integration.md index 33727e0a9c..63c9099eac 100644 --- a/docs/messaging/azure-service-bus-integration.md +++ b/docs/messaging/azure-service-bus-integration.md @@ -1,7 +1,7 @@ --- title: .NET Aspire Azure Service Bus integration description: Learn how to install and configure the .NET Aspire Azure Service Bus integration to connect to Azure Service Bus instances from .NET applications. -ms.date: 02/25/2025 +ms.date: 04/01/2025 --- # .NET Aspire Azure Service Bus integration @@ -15,6 +15,9 @@ ms.date: 02/25/2025 The .NET Aspire [Azure Service Bus](https://azure.microsoft.com/services/service-bus/) hosting integration models the various Service Bus resources as the following types: - : Represents an Azure Service Bus resource. +- : Represents an Azure Service Bus queue resource. +- : Represents an Azure Service Bus subscription resource. +- : Represents an Azure Service Bus topic resource. - : Represents an Azure Service Bus emulator resource. To access these types and APIs for expressing them, add the [📦 Aspire.Hosting.Azure.ServiceBus](https://www.nuget.org/packages/Aspire.Hosting.Azure.ServiceBus) NuGet package in the [app host](xref:dotnet/aspire/app-host) project. @@ -120,12 +123,12 @@ To add an Azure Service Bus queue, call the , it configures your Service Bus resources to have a queue named `queue`. The queue is created in the Service Bus namespace that's represented by the `AzureServiceBusResource` that you added earlier. For more information, see [Queues, topics, and subscriptions in Azure Service Bus](/azure/service-bus-messaging/service-bus-queues-topics-subscriptions). +When you call , it configures your Service Bus resources to have a queue named `queue`. The expresses an explicit parent-child relationship, between the `messaging` Service Bus resource and its child `queue`. The queue is created in the Service Bus namespace that's represented by the `AzureServiceBusResource` that you added earlier. For more information, see [Queues, topics, and subscriptions in Azure Service Bus](/azure/service-bus-messaging/service-bus-queues-topics-subscriptions). ### Add Azure Service Bus topic and subscription @@ -135,7 +138,7 @@ To add an Azure Service Bus topic, call the method providing a name: +When you add an Azure Web PubSub resource, you can also add a child hub resource. The hub resource is a logical grouping of connections and event handlers. To add an Azure Web PubSub hub resource to your app host project, chain a call to the method providing a resource and hub name: ```csharp var builder = DistributedApplication.CreateBuilder(args); @@ -68,12 +68,12 @@ var worker = builder.AddProject("worker") .WithExternalHttpEndpoints(); var webPubSub = builder.AddAzureWebPubSub("web-pubsub"); -var messagesHub = webPubSub.AddHub("messages"); +var messagesHub = webPubSub.AddHub(name: "messages", hubName: "messageHub"); // After adding all resources, run the app... ``` -The preceding code adds an Azure Web PubSub hub resource named `messages`, which enables the addition of event handlers. To add an event handler, call the : +The preceding code adds an Azure Web PubSub hub resource named `messages` and a hub name of `messageHub` , which enables the addition of event handlers. To add an event handler, call the : ```csharp var builder = DistributedApplication.CreateBuilder(args); @@ -82,7 +82,7 @@ var worker = builder.AddProject("worker") .WithExternalHttpEndpoints(); var webPubSub = builder.AddAzureWebPubSub("web-pubsub"); -var messagesHub = webPubSub.AddHub("messages"); +var messagesHub = webPubSub.AddHub(name: "messages", hubName: "messageHub"); messagesHub.AddEventHandler( $"{worker.GetEndpoint("https")}/eventhandler/", @@ -258,6 +258,12 @@ public class ExampleService( } ``` +If you want to register a single `WebPubSubServiceClient` instance with a specific connection name, there's an overload that uses the connection name as the service key. Call the `AddKeyedAzureWebPubSubServiceClient` method. This method registers the client as a singleton service in the dependency injection container. + +```csharp +builder.AddKeyedAzureWebPubSubServiceClient(connectionName: "web-pubsub"); +``` + For more information, see [Keyed services in .NET](/dotnet/core/extensions/dependency-injection#keyed-services). ### Configuration