Skip to content

Latest commit

 

History

History
733 lines (498 loc) · 37.4 KB

File metadata and controls

733 lines (498 loc) · 37.4 KB

Cosmos DB Trigger & Bindings (.NET Core)

Watch the recording of this lesson on YouTube 🎥.

Goal 🎯

Learn about how to trigger a function when adding a document to Cosmos DB, how to use input and output bindings, the usage of Azure Key Vault and the dependency injection pattern in Azure Functions when using Cosmos DB.

This lessons consists of the following exercises:

Nr Exercise
0 Prerequisites
1 Setup the Azure Cosmos DB Emulator
2 Using the Cosmos DB output binding
3 Using the Cosmos DB input binding
4 Creating a Cosmos DB Trigger function
5 Deploying to Azure
6 Using Azure Key Vault for storing the connection string
7 Using Dependency Injection pattern for Cosmos DB connection
8 Homework
9 More Info
10 Feedback

📝 Tip - If you're stuck at any point you can have a look at the source code in this repository

📝 Tip - If you have questions or suggestions about this lesson, feel free to create a Lesson Q&A discussion here on GitHub.


0. Prerequisites

Prerequisite Exercise
An Azure Subscription. 2-3
The Azure Cosmos DB Emulator. 2-7
The Azurite Emulator. 2-7
The Azure Functions extension for VSCode. 2-7
The newplayer-items queue from the Queue lesson. 2-7

1. Setup the Azure Cosmos DB Emulator

Please refer to the official guide corresponding to your platform for the installation steps.

2. Using the Cosmos DB output binding

Let's imagine the following scenario: you need to read new messages from a queue, make a simple transformation to the data and then save it into Cosmos DB. For the implementation, we'll be creating a QueueTrigger function and use the Cosmos DB output binding with a player type in order to read messages from the newplayer-items queue. The same queue that you have used previously in the queue lesson.

2.1 Create the newplayer-items queue

There are two options for this step: use your Azure account or a locally emulated account to work in your local development environment. For demo purposes, we'll opt for the second one.

If you work with Windows you can use the official Azure Storage Emulator and the Azure Storage Explorer app. For OS X and Linux Azurite must be used. Azurite is a Cross-Platform emulator. The last version at the moment of writing this tutorial is 3.10. The step by step instructions to install it can be found here.

2.2 Set up and create the queue locally with Azurite (not Windows)

In this tutorial, we used Azurite. For starting the service with Https use:

azurite --silent --location <installed location> --debug <installed location>/debug.log --oauth basic --cert 127.0.0.1.pem --key 127.0.0.1-key.pem

🔎 Observation - It is recommended to use HTTPS for your connections with Azurite. That way your code won't have to be changed when you move your code to production.

🔎 Observation - The default endpoint used in Azurite for the queue service is https://127.0.0.1:10001. This is the endpoint that will be used to attach the Azure Storage Explorer for the queue storage Account.

At the Azure Storage emulator add the new connection with the ports indicated in your Azurite service. The values shown in the image are the defaults:

Azure Storage

For the queues is used 10001.

Use the ports setup that is more convenient for your environment. There are several settings that can be customized. To get more details, check out the getting started official guide.

After the connection is attached there should be three categories under the connection tree, select Queues and create the newplayer-items queue, or the name of your preference. Once the queue is added, It should be looking as the image below:

new queue image

2.3 Create your Azure Functions Project with VS Code

We need a project with a Queue triggered function. For the step by step guide please refer to the Queue lesson. Name the QueueTrigger function TransformPlayerAndStoreInCosmos.

2.4 Edit the Function code in VS Code

In the QueueTrigger function, add the name of the queue and the name of the connection setting to the queue trigger, last change the string type of the message to the Player type.

 public static void Run(
     [QueueTrigger(
         "newplayer-items",
         Connection = "QueueConnection")]Player playerMessage

Now, add the new QueueConnection setting to the local.settings.json file as showing below:

{
  "IsEncrypted": false,
  "Values": {
    "AzureWebJobsStorage": "UseDevelopmentStorage=false",
    "FUNCTIONS_WORKER_RUNTIME": "dotnet",
    "QueueConnection": "DefaultEndpointsProtocol=https;AccountName=devstoreaccount1;AccountKey=Eby8vdM02xNOcqFlqUwJPLlmEtlCDXJ1OUzFT50uSRZ6IFsuFq2UVErCz4I6tq/K1SZFPTOtr/KBHBeksoGMGw==;QueueEndpoint=https://127.0.0.1:10001/devstoreaccount1;"
  }
}

2.5 Create the Cosmos DB database locally

Depending on your local environment, please take a look at the official guide with the steps for your setup. The Cosmos DB Emulator is not available for OS X or linux at the moment of writing this lesson, If that is your case, you will have to create a Windows Virtual Machine hosted in Parallels or Virtual Box, since . Then establish the connection between the host and the guest machines and finally set up the certificate to use the HTTPS connection in OS X. If you run into any issues with this setup you can take a look at this github issue for troubleshooting or you can opt for creating an actual Cosmos DB in Azure instead of using the emulator.

Open your Cosmos DB Emulator and select the New Database button at the toolbar. We will name it Players. The next step is adding a new container. Keep in mind that the actual data of a Cosmos DB is stored in containers. Let's add a new one named players. Here is where the data from the queue will be saved.

Take a look at the below image for reference.

Azure Cosmos DB Emulator

When you add the new container you have to define the container key. This is very important since this will be used to scale your Cosmos DB. The container key should be picked based on the distribution of the data. From the Microsoft docs:

"If you choose a partition key that evenly distributes throughput consumption across logical partitions, you will ensure that throughput consumption across physical partitions is balanced."

For this exercise, our Partition Key will be region, and also we will add a unique key using the playerId field.

Container and partition key

🔎 Observation - If you are interested in learning more about how the partition key impacts the performance of your application check out this page from the official docs.

2.6 Add the connection string to your Azure Function

Now that we have the Cosmos DB created locally, let's set up the Azure Function to use the local connection string.

Get the connection string from the local emulator:

Connection string from cosmos db emulator

Since we are running the local emulator in a hosted virtual machine we have to change the IP of localhost to the IP of the virtual machine. After getting the IP from the VM, remove localhost and use the actual IP address. Finally add it to the local.settings.json file.

{
  "IsEncrypted": false,
  "Values": {
    "AzureWebJobsStorage": "UseDevelopmentStorage=false",
    "FUNCTIONS_WORKER_RUNTIME": "dotnet",
    "QueueConnection": "DefaultEndpointsProtocol=https;AccountName=devstoreaccount1;AccountKey=Eby8vdM02xNOcqFlqUwJPLlmEtlCDXJ1OUzFT50uSRZ6IFsuFq2UVErCz4I6tq/K1SZFPTOtr/KBHBeksoGMGw==;QueueEndpoint=https://127.0.0.1:10001/devstoreaccount1;",
    "CosmosDBConnection": "AccountEndpoint=https://192.168.7.108:8081/;AccountKey=C2y6yDjf5/R+ob0N8A7Cgv30VRDJIWEHLM+4QDU5DE2nQ9nDuVTqobD4b8mGGyPMbIZnqyMsEcaGQy67XIw/Jw==;"
  }
}

🔎 Observation - Notice the usage of UseDevelopmentStorage to false. This flag is used in combination with the Azure Local Storage emulator, but it is not needed in this exercise. Make sure to set it to false.

2.7 Add a CosmosDB output binding to your Run method

Add a reference to the Microsoft.Azure.WebJobs.Extensions.CosmosDB package to your app running the dotnet add package Microsoft.Azure.WebJobs.Extensions.CosmosDB command in your terminal. At VS Code, open the TransformPlayerAndStoreInCosmos.cs file, where the Run method definition was created.

Add the following binding definition:

[CosmosDB(
    databaseName: "Players",
    collectionName: "players",
    ConnectionStringSetting = "CosmosDBConnection")]out Player playerDocument

After adding the output binding the signature of the Run method should look like the following:

 public static void Run(
    [QueueTrigger(
        "newplayer-items",
        Connection = "QueueConnection")]Player playerMessage,
    [CosmosDB(
        databaseName: "Players",
        collectionName: "players",
        ConnectionStringSetting = "CosmosDBConnection")]out Player playerDocument,
    ILogger log)

Save your changes.

2.8 Add a Player.cs model to the Azure Function application

For this exercise, we need to map the item from the queue to an Entity that can be deserialized to a C# object, and make the required transformations, and then deserialize this object to save it in CosmosDB.

Add a new folder called Models, and add a new C# class inside of it. Call it Player.

Add the following content to this class.

using Newtonsoft.Json;

namespace AzureFunctionsUniversity.Cosmos.Models
{
    public class Player
    {
        [JsonProperty("id")]
        public string Id { get; set; }
        
        [JsonProperty("nickName")]
        public string NickName { get; set; }
        
        [JsonProperty("playerId")]
        public int PlayerId { get; set; }
        
        [JsonProperty("region")]
        public string Region { get; set; }
    }
}

Save your changes.

2.9 Modify the content of the Run method

At the Run method, get the message from the queue, do a simple transformation and then send the resulting data to Cosmos DB.

The final code is shown below:

[FunctionName(nameof(TransformPlayerAndStoreInCosmos))]
public static void Run(
    [QueueTrigger(
        "newplayer-items",
        Connection = "QueueConnection")]Player playerMessage,
    [CosmosDB(
        databaseName: "Players",
        collectionName: "players",
        ConnectionStringSetting = "CosmosDBConnection")]out Player playerDocument,
    ILogger log)
{
    // The code for the data transformation should be added here.
    playerMessage.NickName = playerMessage.NickName.ToUpperInvariant();

    // Return the player data in the document variable used by the output binding.
    playerDocument = playerMessage;

    log.LogInformation($"C# Queue trigger function inserted one document.");
}

Run your function locally to make sure it is correctly connected to the queue and listens to the new message event trigger. Remember that if you need more detailed instructions about the queue trigger you can always review them in the Queue lesson.

Once your function is running, add a new message to the queue using the Azure Storage Explorer, select the newplayer-items queue then click on the + Add message button for adding the following json data:

{
   "nickName": "Samuel",
   "playerId": 1,
   "region": "United States of America"
}

You should see an image very similar to the below one:

Add message and fill out the message data

After adding the message to the queue, the Azure Function should listen to the event and the Run method executed.

Here's an example of the output messages, once the message was added to the queue, and then saved into Cosmos DB.

output messages of function

Go to your Cosmos DB local emulator and verify that the item was added to the Players container. You should see the list of items very similar to the below image:

new-item-cosmos-db

3. Using the Cosmos DB input binding

For this exercise, it will be used the same Cosmos DB and the items that have already been added in the previous section of this lesson. If you have any issue following along this part of the lesson go to the source code to review the finished code.

3.1 Create a new function using the one created in the previous section

Copy the existing Azure Function TransformPlayerAndStoreInCosmos in your Azure Function project to a new file called QueryPlayerWithHttpTrigger.cs

3.2 Add an input Cosmos DB binding to the Run method

An input binding has several fields:

  • databaseName
  • collectionName
  • partitionKey
  • connectionStringSetting

All these fields are required when using bindings. For this example we will use an Http trigger to request an item from the Cosmos DB based on its Id.

Use the following code in your new QueryPlayerWithHttpTrigger function.

using System;
using AzureFunctionsUniversity.Cosmos.Models;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Mvc;
using Microsoft.Azure.WebJobs;
using Microsoft.Azure.WebJobs.Extensions.Http;
using Microsoft.Extensions.Logging;

namespace AzureFunctionsUniversity.Cosmos.Triggers
{
    public static class QueryPlayerWithHttpTrigger
    {
        [FunctionName(nameof(QueryPlayerWithHttpTrigger))]
        public static IActionResult Run(
            [HttpTrigger(
                AuthorizationLevel.Anonymous,
                nameof(HttpMethods.Get),
                Route = "{collectionName}/{partitionKey}/{id}")]HttpRequest req,
            [CosmosDB(
                databaseName: "Players",
                collectionName: "{collectionName}",
                ConnectionStringSetting = "CosmosDBConnection",
                Id = "{id}",
                PartitionKey = "{partitionKey}")] Player playerItem,            
                ILogger log)
        {            
            log.LogInformation("C# HTTP trigger function processed a request.");

            if (playerItem == null)
            {
                return new NotFoundResult();                
            }

            return new OkObjectResult(playerItem);
        }
    }
}

🔎 Observation - At the moment of writing this lesson there is support only for the SQL API with the Cosmos DB bindings.

🔎 Observation - Notice that there are three route parameters used in the HttpTrigger. These are re-used in the CosmosDB input binding.

3.3 Test the Cosmos DB binding locally

For using your function locally, hit F5 in your VS Code. Make sure to have your Cosmos DB Emulator running and some items added to the players collection that can be queried.

Here's an example of a new item.

{
  "nickName": "Simon",
  "playerId": 2,
  "region": "Spain"
}

Remember to use the QueueTrigger function from the previous section for adding a new item.

Run and test your Azure Function locally using the values as part of the URL path. Here's an example of an URL with the three values in the route:

http://localhost:7071/api/players/United States of America/473501fc-9cbe-47ad-8de0-eb659fcfa514

The following image shows the result:

postman-result-route

4. Creating a Cosmos DB Trigger function

Besides bindings, there is another way to connect an Azure Function to a Cosmos DB collection by using Triggers. In this exercise, we will use the same Azure Functions project that we have so far. If you haven't done the previous sections, you can clone the example from the repository and follow along this lesson. Besides the code, please double check that you have all the requirements at the section 0.

4.1 Create a new Azure Function with a Cosmos DB trigger

The scenario is the following: we need to monitor the changes to our players container so we can add new items in another container, which will be used for another application. For achieving this we will use a Cosmos DB trigger and an output binding for Cosmos DB.

At VS Code, using your Azure Function project, add a new function using the Azure Functions extension. Select the Cosmos DB trigger for the template, keep the same namespace that has been used before, select the same CosmosDBConnection setting for the connection string, enter Players for the database name and finally, players for the collection name.

Your code should look very similar to the following:

public static class StoreTeamPlayerInCosmos
{
    [FunctionName(nameof(StoreTeamPlayerInCosmos))]
    public static void Run([CosmosDBTrigger(
        databaseName: "Players",
        collectionName: "players",
        ConnectionStringSetting = "CosmosDBConnection",
        LeaseCollectionName = "leases",
        CreateLeaseCollectionIfNotExists = true)]IReadOnlyList<Document> input,
        ILogger log)
    {
        if (input != null && input.Count > 0)
        {
            log.LogInformation("Documents modified " + input.Count);
            log.LogInformation("First document Id " + input[0].Id);
        }
    }
}

🔎 Observation - A Cosmos DB Trigger relies on the change feed streams attached to the container. When any changes are made to a container, the change feed stream is sent to the trigger for the Azure Function invocation. Also, there is a new element in this function: a Leases collection. The concept of leases is introduced by the Changes Feed and the Cosmos DB trigger. The leases collection is used to keep track of what changed documents have been passed to external process(es), in this case the Azure Function. Without this collection there would be no way to know the checkpoint of the processed changed documents. When only one function exists for a Cosmos DB collection there is not a problem to use one lease collection as only one checkpoint needs to be stored. When there are more functions watching the changes in a container, it needs to be a way to store different checkpoints for each different function.

Make sure to add the CreateLeaseCollectionIfNotExists = true parameter. This one is not added in the template, you have to add it manually.

Now, add the output binding for sending the new items to the teamplayers collection.

Copy the following code which adds a new parameter in your Run method. This one goes after the Trigger parameter.

[CosmosDB(
    databaseName: "Players",
    collectionName: "teamplayers",
    ConnectionStringSetting = "CosmosDBConnection")]IAsyncCollector<TeamPlayer> teamPlayerItemsOut,

Make sure to add locally the new collection teamplayers to your Cosmos DB players as shown in the following image:

add-new-collection-cosmos

Same, needs to be added to the Azure platform Cosmos DB instance.

We also want to process every new item in the players container and then add or change the corresponding one at the teamplayers container. Let's do the changes to the function for achieving this.

[FunctionName(nameof(StoreTeamPlayerInCosmos))]
public static async void Run(
    [CosmosDBTrigger(
        databaseName: "Players",
        collectionName: "players",
        ConnectionStringSetting = "CosmosDBConnection",
        LeaseCollectionName = "leases",
        CreateLeaseCollectionIfNotExists = true)]IReadOnlyList<Document> input,
            [CosmosDB(
            databaseName: "Players",
            collectionName: "teamplayers",
            ConnectionStringSetting = "CosmosDBConnection")]IAsyncCollector<TeamPlayer>teamPlayerItemsOut,
            ILogger log)
{
    if (input != null && input.Count > 0)
    {
        log.LogInformation("Documents modified " + input.Count);
        log.LogInformation("First document Id " + input[0].Id);
        foreach (var item in input)
        {                    
            Player player = JsonConvert.DeserializeObject<Player>(item.ToString());
            // Any changes to the input documents are here
            TeamPlayer teamPlayer = new TeamPlayer() 
            { 
                Id = player.Id,
                PlayerName = player.NickName,
                Region = player.Region,
                TeamId = 1
            };
            await teamPlayerItemsOut.AddAsync(teamPlayer);          
        }
    }
}

Notice tha we are using a new object type, TeamPlayer. You can copy the contents of this class from the repository files here

In the code we are processing all the items in the input parameter. Each document either inserted or changed in the players container will be sent in this list. We will send these items to the output binding into the type we have for the teamplayers container. In case you need more changes to every document in the input you can add the code here.

Now let's try out the functions in our app locally and see if everything works as expected.

For testing out this function do the following flow:

  • Add a new item to the newplayer-items queue. This will trigger the TransformPlayerAndStoreInCosmos function.
  • Watch how the StoreTeamPlayerInCosmos function gets executed.

At the end of the execution you should see a new item in the players container and a new one at the teamplayers respectively.

5. Deploying to Azure

For deploying the Azure Function there are several options. All the methods are explained in the Deployment lesson.

In this exercise, we'll do the deployment using VS Code.

5.1 Deploy the Azure resources

Click on the Azure Extension. It should already display the subscription that you have for your Azure Account. Select the Azure Function and click on the Deploy Button indicated in the following image:

deploy-function

Indicate the unique global name, the stack, and the region for the new function.

unique-name-function

Check the output of the deployment task to review if there is any error.

output-messages-deployment

🔎 Observation - Notice that there is a Function App, and a Storage Account with the same name as the function.

Besides deploying a new Function App, you have to create in your Azure Subscription, a new Azure Cosmos DB and a Players container.

Follow the first three sections of this tutorial from the Microsoft Docs to create your first Cosmos DB.

Following, you also have to create a new queue named newplayer-items, using the same Azure Storage Account. For this step you can use the Azure Storage Explorer.

All the details on how to add the queue can be reviewed at the Queue lesson.

create-queue-option

After the Queue is added, the Storage Account should look like the following image:

newplayer-items-queue

5.2 Add the connection strings to your Function App

There are two connection strings that should be added to the settings of the Azure Function:

  • QueueConnection
  • CosmosDBConnection

The names of both connections must be the same as the code deployed.

Using the Azure Storage Explorer, copy the Connection String from the queue and add it with the QueueConnection setting:

queue-connection-string

In the Azure portal, go to Home then Function App, select the deployed function, and configuration from the left panel under Settings section. Under Application Settings, select the New application setting button, and add the QueueConnection setting.

Again, in the Azure Portal, go to Home, then Azure Cosmos DB, select your Cosmos DB with the Players container. Then at the left panel, under the Settings section, click on Keys, copy the Primary Connection String. Go back to the Application Settings of the Azure Function and select the New application setting button, and add the CosmosDBConnection setting.

Once both settings have been added, you should see both listed as shown at the below image.

connections-azure-portal

🔎 Observation - Notice that even though both settings are connection strings these are added as Application Settings. The only Connection Strings that have to be in the Connection String section are Entity Framework connection strings.

At this point, the Azure Function with the output binding is fully set up to start reading from the queue and adding items to Cosmos DB. You can test it out using the Azure Storage Emulator just exactly as it was done when working locally, just now you are using the queue from your Azure Subscription. Try it out and make sure everything works as expected.

6. Using Azure Key Vault for storing the connection string

So far we have used two connection strings: one for the queue connection and a second one for the Cosmos DB instance. Both connection string are critical settings that need to be managed and even shared between functions. In order to keep these settings secure we will use the Azure Key Vault service for storing them and share them. A Key Vault allows to manage secrets, certificates and keys from Azure resources using Azure Active Directory for authentication to access any of the resources stored on it. Also it can be used to monitor who and when this resources are being accessed.

For creating a new Key Vault there are 3 options: Azure CLI, Azure Portal and PowerShell. For this exercise we'll use the Azure Portal for simplicity.

6.1 Create the new Azure Key Vault

Follow the first 3 sections at this Quick start guide. Use the name FunctionUniversity-Vault for the name of the Key vault.

6.2 Add the secrets to the Key Vault

Add two secrets: CosmosDBConnection and QueueConnection to the vault. You will use the connection string value from both services: Cosmos DB and the Queue Storage connection and stored them in the Key Vault respectively.

keyvault-secrets

6.3 Add a Managed Identity

We need the Azure Functions Application to be able to read both secrets in the Key Vault, to do so let's add a system-assigned managed identity to our application, which will be used to access the protected secrets. A system-assigned identity is ideal for this case since the identity will be tied to the Functions Application and will have the same life cycle. Follow the instructions at this Microsoft Docs official guide to enable the System Assigned identity. At the Azure Portal, select you Azure Function App, then settings, and Identity at the left panel, as showing in the below image:

managed-identity

6.4 Configure the Access Policy on the Key Vault

Add the new access policy by going to the left panel then Access policies in your Azure Key Vault resource. Find the Service Principal for your Azure Function app using the name of the Azure Function App as shown in the image:

service-principal-adding

On the permissions field select the Get and List permissions for the secrets field. At this moment there is no need to add more than the secret permissions.

Click Save on the Access Policies at the Azure Portal.

6.5 Referencing the Azure key vault secrets at the Azure Functions App settings

The next step is to add the reference to the secrets of the Azure Key Vault service in the Function App. First get the identifiers from your Azure Key Vault and copy both QueueConnection and CosmosDBConnection values for later use. The identifier looks something like this:

https://{name}.vault.azure.net/secrets/{secret}/{id}

Go to your Function App, then configuration and update the value of both connection strings, with their own reference to the Key Vault. Example:

The app setting name is: QueueConnection and the value will be something like @Microsoft.KeyVault(SecretUri={copied identifier for the secret of the QueueConnection})

The format is documented at this page of the official Microsoft Docs.

After both settings are added your App Settings should look like this:

app-settings-keyvault-reference

Now we can execute the Queue Trigger Azure Function by adding a new item to the queue in the Azure Storage Explorer. There is no need to change anything in the deployed code.

More about Managed Identities in the official docs

7. Using Dependency Injection pattern for Cosmos DB connection

One of the good practices when using Connections to resources is to look for sharing connections instead of creating them every time a new Function is executed. When many functions are running concurrently, a new connection is created for every running function. If you want to manage the connections so you can share them across functions, or if you want to use bindings to more than one container in the same function, and many other scenarios, there are a couple of options: Use a static client or use dependency injection in your function method to manage your own Cosmos DB client.

So far we have used Bindings and Triggers for Cosmos DB, for this exercise the demo will be using an HttpTrigger, and we will be adding new items on demand on the players container. We will send a POST with the data for the new item instead of using the Queue Trigger. This is just to show a different approach and how to use the StartUp class as you would use it in ASP.Net Core applications. You want to use this approach instead of using bindings.

Before you can use dependency injection, you must install the following NuGet packages to your Azure Functions App:

  • Microsoft.Azure.Functions.Extensions
  • Microsoft.NET.Sdk.Functions package version 1.0.28 or later
  • Microsoft.Extensions.DependencyInjection (currently, only version 3.x and later supported)

And the following for using the Cosmos Client:

  • Microsoft.Azure.Cosmos

7.1 Add a new class at the root of the App

In VS Code, create a new cs file and name it Startup.cs. Add the following code on it and save it.

using System;
using Microsoft.Azure.Functions.Extensions.DependencyInjection;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Configuration;
using Microsoft.Azure.Cosmos.Fluent;
using AzureFunctionsUniversity.Cosmos;

[assembly: FunctionsStartup(typeof(Startup))]

namespace AzureFunctionsUniversity.Cosmos
{
    public class Startup : FunctionsStartup
    {               
        public override void Configure(IFunctionsHostBuilder builder)
        {

           builder.Services.AddSingleton(s => {
                var connectionString = Environment.GetEnvironmentVariable("CosmosDBConnection");
                if (string.IsNullOrEmpty(connectionString))
                {
                    throw new InvalidOperationException(
                        "Please specify a valid Cosmos DB Connection in the appSettings.json file or your Azure Functions Settings.");
                }

                return new CosmosClientBuilder(connectionString).WithConnectionModeDirect()
                    .Build();
           });
        }
    }
}

The previous code sets the services container including the CosmosClientBuilder object, also reads the configuration from the local.settings.json file. We are not binding the settings configuration to any custom type, but that is also something that can be done in the Configure method. Notice the usage of the WithConnectionModeDirect for the CosmosClientBuilder. This connection mode is the recommended one for better performance. More about connection mode at the official docs. Last, the assembly decoration to the file is required when using the StartUp class.

7.2 Add a new Azure Function manually with Http trigger

In VS Code, add a new file and name it StorePlayerWithDI.cs. Add the following code on it:

using System;
using System.Threading.Tasks;
using Microsoft.AspNetCore.Mvc;
using Microsoft.Azure.WebJobs;
using Microsoft.Azure.WebJobs.Extensions.Http;
using Microsoft.AspNetCore.Http;
using Microsoft.Azure.Cosmos;
using AzureFunctionsUniversity.Cosmos.Models;
using Newtonsoft.Json;

namespace AzureFunctionsUniversity.Cosmos.Output
{
    public class StorePlayerWithDI
    {
        private readonly CosmosClient _cosmosClient;

        public StorePlayerWithDI(CosmosClient cosmosClient)
        {
            _cosmosClient = cosmosClient;
        }

        [FunctionName(nameof(StorePlayerWithDI))]
        public async Task<IActionResult> Run(   
            [HttpTrigger(
                AuthorizationLevel.Function, nameof(HttpMethods.Post))] HttpRequest req)
        {
            var myItem = await req.ReadAsStringAsync();

            Player player = JsonConvert.DeserializeObject<Player>(myItem);            
            player.NickName = player.NickName.ToUpperInvariant();      
            player.Id = Guid.NewGuid().ToString();           
            
             /* Add any validations here */      
            var container =  _cosmosClient.GetContainer("Players", "players");        

            try
            {
                ItemResponse<Player> item  = await container.UpsertItemAsync<Player>(player, new PartitionKey(player.Region));

                return new OkObjectResult(item.Resource);
            }
            catch (CosmosException)
            {
                return new BadRequestResult();                
            }
        }
    }
}

This code has the following changes:

  • The class for the Function is not static. This is required if we are using Dependency Injection, because we need to inject the dependencies at the constructor method.
  • The class does not have any static members, since this is necessary for accessing class member variables.
  • The constructor has the Cosmos Client which was created at the StartUp class.
  • The usage of UpsertItemAsync for adding or updating an item at the players container

With the mentioned changes, the connection to the Cosmos DB instance is shared in all the function executions of this instance.

7.3 Test the function

In VS Code, hit F5 or click on Run at the main menu.

Using Postman or any other API client, trigger the StorePlayerWithDI function with a payload in json of a player item. Use a post action and check that the result is successful.

The following image shows a successful response.

new-item-cosmos-manual

🔎 Observation - If you have done the other functions that are part of this lesson, you should also see that the Cosmos DB trigger function gets executed and a new item is added at the teamplayers container. If you want to deploy this function to your Azure Subscription, just follow the steps at the Deploying section.

If you want to take a look at the code of this lesson, here is the source code of the full lesson.

The official documentation about dependency injection in Azure Functions is at this link, if you want to read more about how to use it in many more scenarios with Azure Functions.

8. Homework

Deploy all the functions to your Azure Subscription and test them in the Azure Cloud.

9. More info

For more info about Cosmos DB and bindings for Azure Functions have a look at the official Azure Cosmos DB Bindings documentation.

10. Feedback

We love to hear from you! Was this lesson useful to you? Is anything missing? Let us know in a Feedback discussion post here on GitHub.


🔼 Lessons Index