How to authenticate to Entra ID protected APIs with a certificate—step-by-step
Last updated on November 10, 2024
This blog post is meant to serve as one place to get all the info you need to set up certificate authentication. This scenario is vital when you want to authenticate to an API with application permissions but can’t use an Azure Managed Identity, and the API does not support using a client secret.
You may encounter such a situation if you, for example, want to call the SharePoint Online REST API with application permissions while debugging because it does not support authentication with a client secret, and you can’t use an Azure Managed Identity when debugging locally. And even if an API supported authentication with a client secret (e.g., the Microsoft Graph API), using a certificate is still recommended for production scenarios—if Azure Managed Identity can’t be used—as it is more secure than using a secret. You can also use these steps to authenticate to a custom API protected by Entra ID.
I wanted to write this article because whenever I had to set up certificate authentication, there wasn’t just one article I could look at to find all the steps that I needed to take. I want this article to be such a place, so do let me know in the comments if you notice a step missing or if you think something should be elaborated on more to make following the steps easier. Hopefully, this is also a guide you can give to your tenant admin if you don’t have the required permissions to perform all the steps yourself.
Table of contents
- Create the certificate
- Create an Entra ID application registration
- Store the certificate on Azure Key Vault
- Certificate authentication in Azure App Service
- Certificate authentication in Azure Logic Apps and Power Automate
- Afterword
Create the certificate
The first thing we need to do is to generate the required certificate files (.cer and .pfx). You can do that by running the script below. Before running the script, adjust the $commonName
and $password
variable values to match your scenario. The certificate files will appear in the same directory where you execute the script.
Create an Entra ID application registration
If you don’t yet have an existing app registration, here are the steps. If you do, you can skip this section and go to the Add the certificate to the app registration section.
- Go to https://portal.azure.com
- Click on Microsoft Entra ID on the left navigation
- Under Manage, click on App registrations
- New registration:
- Give it a descriptive name
- Select Accounts in this organizational directory only (Single tenant)
- No need to specify a redirect URI
After your app has been created, note down the following information from the Overview tab; you’ll need these in your solution.
- Application (client) ID
- Directory (tenant) ID
Add the certificate to the app registration
In your application registration:
- Go to Certificates & secrets
- Open the Certificates tab
- Click Upload certificate
- Select the .cer file we generated with the PowerShell script, and click Add.
Configure the required API permissions
Don’t forget to configure the permissions that are required to use the desired API.
- On the application registration, go to API permissions.
- Click Add a permission.
- Select the API you want to use, e.g., SharePoint or Microsoft Graph. Custom APIs can be found on the My APIs tab.
- Select Application permissions.
- Tick the permissions you need. Typically, the API documentation tells you what permissions you need to perform different API operations.
- Click on Add permissions.
- And finally, you also need to click the Grant admin consent for organization button and then Yes. If you don’t have the permission to do this, you need to ask your tenant admin to hit the button for you.
When permissions have been granted successfully, the warning triangles are replaced with green checkmarks in the view.
Store the certificate on Azure Key Vault
Azure Key Vault offers a secure place to store the certificate so that our app can access it during runtime.
Enable the Azure app’s Managed Identity
Your app needs permission to get the certificate from the key vault. The best way for Azure apps is to enable the app’s managed identity and grant it the required permissions.
- Open the Identity blade under Settings.
- Change the system assigned managed identity Status to On.
- Click on Save and Yes when prompted
- Copy the Object ID that appears upon saving to, e.g., Notepad. You’ll need this later.
Create an Azure Key Vault
On Azure Portal, go to the resource group that contains your app. Then, click the button to create a new resource and search for Key Vault on the Marketplace. Then, create one with the following settings.
- Ensure the correct subscription and resource group are selected.
- Give the key vault a name according to your organization’s naming conventions.
- Select the region you wish to use, probably the same as what your app is using.
- Leave the pricing tier to Standard
- Consider if you wish to enable Purge protection.
- Proceed to the Access configuration tab and ensure Azure role-based access control (recommended) is selected.
- If you want, you can tag the resource.
- Click on Review + create
When the Key Vault resource has been provisioned, go to it.
Configure the vault permissions
You need to grant yourself permission to add the certificate to the vault before you can do so.
- Go to the Access control (IAM) blade
- Click on Add role assignment
- Select the Key Vault Certificates Officer permission, and click Next
- Click on Select members and add yourself.
- Click on Review + assign
You also need to grant your Azure app’s Managed Identity permissions to read the certificate from the vault.
- Go to the Access control (IAM) blade
- Click on Add role assignment
- Select the Key Vault Certificates User permission, and click Next
- Select Managed identity, click on Select members, and add your app resource.
- Click on Review + assign
Add the certificate to the vault
- Navigate to the Certificates blade under Objects.
- Click on the Generate/Import button at the top.
- Select Import from the Method of Certificate Creation dropdown.
- Give a descriptive name for the vault object.
- Upload the .pfx certificate file you generated earlier.
- Provide the same password as what you used when generating the certificate.
To reference the certificate in your app later, you’ll need its URL.
- Click on the certificate you just added to the key vault.
- Similarly, also click on the current certificate version.
- Copy the Certificate Identifier (URL) to, e.g., Notepad. You can remove the version (GUID) from the end of the URL. This way, the latest version of the certificate will always be used. We’ll need this value in just a second!
Now, you can remove the permissions you added for yourself earlier, as they are no longer needed.
Certificate authentication in Azure App Service (C# .NET 8)
Configure the app’s environment variables
The app needs the following three pieces of information for authentication: the tenant ID, the client ID, and the certificate. We will provide the app with this information by adding them as App settings on the Environment variables blade.
Name | Value | Description |
---|---|---|
TenantId | The Entra ID application registration tenant ID | You copied this earlier from the app registration Overview blade. This setting is needed for authentication. |
ClientId | The Entra ID application registration client ID | You copied this earlier from the app registration Overview blade. This setting is needed for authentication. |
Certificate | @Microsoft.KeyVault(SecretUri= https://yourvaultname.vault.azure.net /certificates/yourcertname) |
This is called a key vault reference. It allows our Azure app to retrieve the certificate from the vault using its Azure Managed Identity. You must paste the certificate identifier URL we copied as the SecretUri value. This setting is needed for authentication. |
WEBSITE_LOAD_USER_PROFILE | 1 | This setting is required to ensure the app service can fetch the certificate from the key vault (needed for authentication). |
C# .NET 8 code for certificate authentication
You can use the code below for certificate authentication both when running the app on Azure and when debugging locally because the code checks if a debugger is attached or not and, based on that information, directs the execution to use the correct method. If the debugger is NOT attached, it means the code is running on Azure, and the code will use the certificate that is now readily available in the app settings, thanks to the key vault reference. However, if the debugger is attached, the code will use the local certificate file path and its password that are specified on the local.user.settings
file (example after the code snippet).
Here’s an example of the local variables you need for debugging.
local.user.settings
file should never be checked into version control (luckily, it is excluded by default, so do not include it!), and you should use a different certificate and application registration for debugging so you do not store the production certificate password in a text file even on your machine. Ideally, you’d have an entirely separate developer tenant for development and initial testing purposes, especially if your application uses the tenant data, e.g., via Microsoft Graph or SharePoint Online REST API.For the code to compile, you’ll also need to add the related interface to your project.
For the code to actually work, you also need to add the service to be available via dependency injection on the Program.cs
file. The file syntax can vary a bit depending on the project type, but regardless, you need to include the services.AddSingleton<IAuthService, AuthService>();
line there in one form or another. The following code snippet is for an Azure Function App.
Certificate authentication in Azure Logic Apps and Power Automate
Fetch the certificate with the Get secret action
As the first thing, we want to get the certificate in its base64 format from the Azure Key Vault we created and configured earlier. We can do this with the Get secret action found under Azure Key Vault.
Before we configure the action itself, we need to form the connection to the key vault. For Azure Logic Apps, we should use its Managed identity for authenticating to the vault, which you should have already enabled as instructed in the Enable the Azure app’s Managed Identity section. For Power Automate, you should choose Default Microsoft Entra ID application for OAuth as the Authentication Type; otherwise, the parameters are the same.
If you have successfully granted the required permissions to the key vault as instructed earlier in the Configure the vault permissions section, you should now be able to select the certificate from the dropdown.
Now, because the certificate value should be kept a secret, we should go to the Settings tab and flip the Secure outputs switch to ensure the certificate value does not show up on the run logs.
Authenticate to the API in an HTTP action
After the Get secret action, you should add an HTTP action for calling the API. To be able to specify the authentication parameters, you first need to open the Advanced parameters dropdown and select Authentication. Then, as the Authentication Type, you need to select Active Directory OAuth.
In the view that opens, you need to specify the Tenant ID and Client ID, which you should have already noted down somewhere if you followed the instructions in the Create an Entra ID application registration section. I recommend that you add these values as Parameters (Type: String) so they can be easily adjusted between environments if needed. You can open the Parameters configuration panel via the button at the top of the logic app editor view. Then, reference the created parameter values in the HTTP action settings.
The Audience field requires you to insert the base URL of the API you wish to call. For example, to call the SharePoint REST API, you’d insert the tenant base URL.
As the Credential Type, you should select Certificate and after that reference the value output variable from the earlier Get secret action. That’s it! There’s no need to add the certificate password here because it was already previously supplied when the certificate was added to the vault.
Afterword
I hope you enjoyed reading this article and found it helpful. As I mentioned in the beginning, I want this post to serve as a place to get all the info you need to set up certificate authentication. Let me know if you notice a step missing or if you feel like something should be elaborated on more!
If you’d like to subscribe to my blog to get notified when I publish something new or follow me on social media, you can find buttons for those in the sidebar or at the bottom for mobile.
Thank you for reading; it is much appreciated, as always. Until next time!
Laura
Hi Laura,
just want to let you know that the ways shown in this article does not work anymore. I struggled with a AADSTS90002: Tenant ‘token’ not found. error. I am writing, because the blog post seemed to became the most valueable resource around this topic.
I have used the following way to get the token:
var app = ConfidentialClientApplicationBuilder.Create(_clientId)
.WithCertificate(GetCertificateFromStore(_certificateThumbprint))
.WithAuthority($”https://login.microsoftonline.com/{_tenantId}”)
.Build();
return (await app.AcquireTokenForClient(_scopes).ExecuteAsync()).AccessToken;
_scopes must only contain what you have put into the url. e.g. https://graph.microsoft.com/.default
It is important to have a .default suffix.
Thanks, Sebastian, for letting me know about the issue and that this blog post is still considered valuable even though it has been written so long ago. I’ll try to update it to reflect modern times in the near future!
Laura
Hi Laura,
Awesome Blog,
Have you had experience using the certificate for SharePoint Online, we have project where we are trying to set up Mutual TLS
CDS
Hi CDS,
Yes, I specifically wrote this blog post because I needed to authenticate with a certificate to use the SharePoint REST API.
Laura
Thank you very much for this! 🙂
I tried to implement your solution with PHP, but failed since it seems there is some issue I can not make out myself.
When sending the request to get the token I receive a 400 error ‘logon_cert’ is not a supported value for requested_token_use parameter.
I’ve posted my question on stackoverflow:
https://stackoverflow.com/questions/68199359/microsoft-graph-getting-access-token-with-certificate-results-in-400-logon-cer
One of my biggest problems is, that I do not exactly know how the request should look like. If there was an exact request definition or anything like postman request, that would also help out a lot!
I asked the question more generic here: https://stackoverflow.com/questions/68275394/microsoft-graph-how-to-get-access-token-with-certificate
Hi, I noticed your and all guides (yours is the best I´ve found btw!) similiar to it is using self signed certificate. Is it also possible to use a internal or external CA certificate?
Absolutely! And thanks. 🙂
Laura
Thanks for the quick reply! Could you please point me in the right direction on how to do so? I’ve spent hours trying to get an answer via Google but no luck.
Hi Laura
Do you have similar implementation for Java?
No, unfortunately.
Laura