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,
thank you for the great and detailed post. It was very helpful. Should the URL passed to AuthenticationContext constructor on line 31 of Authorization class have the “/oauth2/token” stripped out? In my code I had to change it to: $”https://login.microsoftonline.com/{_tenantId}”
Thanks,
Ivan.
Hi Ivan,
Not based on my experience. Could it be that something else in your solution is adding those final fragments to the token endpoint URL and that’s why you don’t need to specify them yourself?
Laura
Hi I had followed the instructions and ran into problem.
I was trying with Active Directory OAuth but was getting error – Access token validation failure. Invalid audience.
Error information :-
{
“error”: {
“code”: “InvalidAuthenticationToken”,
“message”: “Access token validation failure. Invalid audience.”,
“innerError”: {
“request-id”: “6653167e-f63a-4252-b2b9-b5a95f96b945”,
“date”: “2019-12-03T21:54:23”
}
}
}
Hi Vasanth,
It is difficult to say what is going wrong without seeing your code. Ensure that you are passing in the correct token and scopes.
Laura
Can you figure it out how to use this in Postman?
Hi Laura,
There is a small tweak required (I think) to your instructions for getting the base64 string of the certificate for the Logic App.
You are asking to get the base64 string of the .cer file by navigating to the App Reg manifest, this won’t work as this is not the pfx file (and therefore won’t have the keys).
What worked for me was to get the base64 string of the pfx file by running the following powershell:
$certpfxfile = Get-Content .\sharepointapponly.pfx -Encoding Byte
[System.Convert]::ToBase64String($certpfxfile) | Out-File .\sharepointapponlypfxcertasbase64.txt
Then take the contents of this .txt file and use it in the logic app.
Hi Colin,
Thanks a lot!!! I’ve updated the post accordingly. 🙂
Laura
Hi Laura,
Thanks for this fantastic and comprehensive blog and all the very interesting topics you describe.
I tried and wanted to implement exactly what you described here – using a certificate in a Logic App. I implemented all steps (verified it works with the client secret for the app) but when I use the certificate I only get this error message:
“BadRequest. Could not load the certificate private key. Please check the authentication certificate password is correct and try again.”
I tried it with both certificate settings – “Active Directory OAuth” and “Client Certificate”. Both only resulting in BadRequest.
Do you have any input what could be the cause for the BadRequest and how to get it to work with certificates?
Thanks,
Stefan
Hi Stefan,
Sounds like there’s something wrong with the configurations; either in Azure AD or possibly a typo in the pfx certificate base64 encoded string.
Anyone else here faced this issue when following the instructions? I could make the instructions clearer if there is a step that can be misunderstood, or if there’s something missing.
Laura