Jump to content
Xtreme .Net Talk

Using PostgreSQL with .NET and Entra ID


Recommended Posts

Guest Aaron Powell
Posted

PostgreSQL is a powerful, open-source relational database system that is widely used in the industry. In this blog post, we will explore how to use PostgreSQL with .NET and how to secure your app with Entra ID.

 

[HEADING=1]Getting started with PostgreSQL[/HEADING]

 

Let’s start by creating a new application which is going to use PostgreSQL as the database. There are many ways we could setup PostgreSQL: we could install it, run it in a Docker container, but for this example, I’m going to bootstrap it with .NET Aspire and the PostgreSQL hosting component. While this does run a Docker container behind the scenes, it means we don’t need to worry about setting up Docker or managing the secrets to connect to the database.

 

For this, we’ll use the .NET Aspire starter application:

 

[iCODE]dotnet new aspire-starter --name PostgreSQLDemo[/iCODE]

 

This will create a new .NET Aspire application with the AppHost, Service Defaults, an API backend and a Blazor web frontend.

 

Next, we’ll add the PostgreSQL hosting component to the AppHost project:

 

cd PostgreSQLDemo.AppHost
dotnet add package Aspire.Hosting.PostgreSQL

 

The final piece for our AppHost is to create the PostgreSQL resource and then add it as a reference to the API project. Open the [iCODE]Program.cs[/iCODE] file in the AppHost project and update it to match the following:

 

var builder = DistributedApplication.CreateBuilder(args);

// Create the PostgreSQL resource
var postgres= builder.AddPostgres("postgres");
var db = postgres.AddDatabase("db");

// Update the API Service project to have the PostgreSQL database as a reference
var apiService = builder.AddProject<Projects.PostgreSQLDemo_ApiService>("apiservice")
                   .WithReference(db);

builder.AddProject<Projects.PostgreSQLDemo_Web>("webfrontend")
   .WithExternalHttpEndpoints()
   .WithReference(apiService);

builder.Build().Run();

 

There are two changes that we’ve made here. First, we’ve added the PostgreSQL resource to the AppHost project. This will create a new PostgreSQL database and user for the application, running in a container, when we launch our project. Second, we’ve added the database as a reference to the API project. This will allow the API project to connect to the database.

 

[HEADING=1]Adding PostgreSQL to the API[/HEADING]

 

Now that we are starting a PostgreSQL resource in our AppHost, we can add the necessary code to the API project to interact with the database. We’ll start by adding the Npgsql package to the API project, but we’ll use the .NET Aspire package, which ensures that we get all the logging and telemetry that we need:

 

cd ../PostgreSQLDemo.ApiService
dotnet add package Aspire.Npgsql

 

Note: If you are using EF Core, you can use the [iCODE]Aspire.Npgsql.EntityFrameworkCore.PostgreSQL[/iCODE] package instead.

 

 

Next, we’ll add Npgsql to our service collection, so we can resolve it with Dependency Injection. Open the [iCODE]Program.cs[/iCODE] file in the API project and update it to match the following:

 

[iCODE]builder.AddNpgsqlDataSource("db");[/iCODE]

 

Make sure that the value you provide as the [iCODE]connectionName[/iCODE] is the same as the name you used when you added the database reference in the AppHost project, this way the right connection string will be resolved.

 

With all that in place, you can now use Npgsql in your API project to interact with the PostgreSQL database.

 

[HEADING=1]Securing the database in Azure with Entra ID[/HEADING]

 

So far, we’ve focused on the local development experience. For production, we’ll want to utilize Azure PostgreSQL Flexible Server for a managed instance. To enhance application security, we will employ Entra ID, creating a Managed Identity for database authentication instead of using a SQL username and password.

 

 

Note: Setting up an Azure PostgreSQL Flexible Server and configuring it with Entra ID is beyond the scope of this article, but do check out the docs for information on that.

 

 

Using Managed Identity differs from traditional SQL username and password authentication because it does not require a persistent password. Instead, you need to request an access token that acts in place of a persistent password, and this access token is only short lived, which means we’re going to need to request a new one from time to time.

 

[HEADING=2]Getting an access token[/HEADING]

 

To get an access token, we’ll use the [iCODE]Azure.Identity[/iCODE] package, which is part of the Azure SDK. We’ll add this package to the API project:

 

cd ../PostgreSQLDemo.ApiService
dotnet add package Azure.Identity

 

With the [iCODE]Azure.Identity[/iCODE] package, we can use the [iCODE]DefaultAzureCredential[/iCODE] class to load the Managed Identity from the environment, and then use it to request an access token. Here’s an example of how you would do that:

 

var credentials = new DefaultAzureCredential();
var token = await credentials.GetTokenAsync(new TokenRequestContext(["https://ossrdbms-aad.database.windows.net/.default"]), CancellationToken.None);
Console.WriteLine(token.Token);

 

That’s great to get a token one-time, but we’ll need to refresh it, and to do that we can leverage a feature of Npgsql, the [iCODE]PeriodicPasswordProvider[/iCODE].

 

[HEADING=2]Using PeriodicPasswordProvider[/HEADING]

 

The [iCODE]PeriodicPasswordProvider[/iCODE] is a feature of Npgsql that allows you to provide a callback that will be called whenever Npgsql needs a password. This is perfect for our use case, as we can use it to request a new access token whenever Npgsql needs it. It will also cache that token for a period of time, so we don’t need to request a new one every time.

 

To use the [iCODE]PeriodicPasswordProvider[/iCODE], we’ll adjust the registration of Npgsql in the service collection to use it. Open the [iCODE]Program.cs[/iCODE] file in the API project and update it to match the following:

 

builder.AddNpgsqlDataSource("db", configureDataSourceBuilder: (dataSourceBuilder) =>
{
   if (string.IsNullOrEmpty(dataSourceBuilder.ConnectionStringBuilder.Password))
   {
       dataSourceBuilder.UsePeriodicPasswordProvider(async (_, ct) =>
       {
           var credentials = new DefaultAzureCredential();
           var token = await credentials.GetTokenAsync(new TokenRequestContext(["https://ossrdbms-aad.database.windows.net/.default"]), ct);
           return token.Token;
       }, TimeSpan.FromHours(24), TimeSpan.FromSeconds(10));
   }
});

 

Here, we’re providing a [iCODE]configureDataSourceBuilder[/iCODE] callback to the [iCODE]AddNpgsqlDataSource[/iCODE] method, which allows us to configure the [iCODE]DataSourceBuilder[/iCODE] that Npgsql will use. We’re checking if the password is empty, which means that Npgsql is expecting a password. If so, we’re setting up the [iCODE]PeriodicPasswordProvider[/iCODE] to cache the password for 24 hours and to retry every 10 seconds in case of failure. When running locally with .NET Aspire, the password will be set, so we do not need to use the [iCODE]PeriodicPasswordProvider[/iCODE] because we have a known and consistent password.

 

[HEADING=1]Next Steps[/HEADING]

 

In this blog post, we’ve looked at how to get started with PostgreSQL in .NET, how we can easily setup a local development experience with .NET Aspire, and how, when deployed to Azure, we can secure our application with Entra ID using the [iCODE]PeriodicPasswordProvider[/iCODE] in Npgsql to request access tokens for Managed Identity.

 

 

The post Using PostgreSQL with .NET and Entra ID appeared first on .NET Blog.

 

Continue reading...

Join the conversation

You can post now and register later. If you have an account, sign in now to post with your account.

Guest
Reply to this topic...

×   Pasted as rich text.   Paste as plain text instead

  Only 75 emoji are allowed.

×   Your link has been automatically embedded.   Display as a link instead

×   Your previous content has been restored.   Clear editor

×   You cannot paste images directly. Upload or insert images from URL.

×
×
  • Create New...