Note
Access to this page requires authorization. You can try signing in or changing directories.
Access to this page requires authorization. You can try changing directories.
Important
Effective May 1, 2025, Azure AD B2C will no longer be available to purchase for new customers. Learn more in our FAQ.
You can use the OAuth 2.0 authorization code grant in apps installed on a device to gain access to protected resources, such as web APIs. By using the Azure Active Directory B2C (Azure AD B2C) implementation of OAuth 2.0, you can add sign-up, sign-in, and other identity management tasks to your single-page, mobile, and desktop apps. In this article, we describe how to send and receive HTTP messages without using any open-source libraries. This article is language-independent. When possible, we recommend you use the supported Microsoft Authentication Libraries (MSAL). Take a look at the sample apps that use MSAL.
The OAuth 2.0 authorization code flow is described in section 4.1 of the OAuth 2.0 specification. You can use it for authentication and authorization in most application types, including web applications, single-page applications, and natively installed applications. You can use the OAuth 2.0 authorization code flow to securely acquire access tokens and refresh tokens for your applications, which can be used to access resources that are secured by an authorization server. The refresh token allows the client to acquire new access (and refresh) tokens once the access token expires, typically after one hour.
This article focuses on the public clients OAuth 2.0 authorization code flow. A public client is any client application that can't be trusted to securely maintain the integrity of a secret password. This includes single-page applications, mobile apps, desktop applications, and essentially any application that doesn't run on a server.
Note
To add identity management to a web app by using Azure AD B2C, use OpenID Connect instead of OAuth 2.0.
Azure AD B2C extends the standard OAuth 2.0 flows to do more than simple authentication and authorization. It introduces the user flow. With user flows, you can use OAuth 2.0 to add user experiences to your application, such as sign-up, sign-in, and profile management. Identity providers that use the OAuth 2.0 protocol include Amazon, Microsoft Entra ID, Facebook, GitHub, Google, and LinkedIn.
To try the HTTP requests in this article:
- Replace {tenant}with the name of your Azure AD B2C tenant.
- Replace 00001111-aaaa-2222-bbbb-3333cccc4444with the app ID of an application you've previously registered in your Azure AD B2C tenant.
- Replace {policy}with the name of a policy you've created in your tenant, for exampleb2c_1_sign_in.
Redirect URI setup required for single-page apps
The authorization code flow for single page applications requires some additional setup.  Follow the instructions for creating your single-page application to correctly mark your redirect URI as enabled for CORS. To update an existing redirect URI to enable CORS,  you can click on the migrate prompt in the "Web" section of the App registration's Authentication tab. Alternatively, you can open the App registrations manifest editor and set the type field for your redirect URI to spa in the replyUrlsWithType section.
The spa redirect type is backwards compatible with the implicit flow. Apps currently using the implicit flow to get tokens can move to the spa redirect URI type without issues and continue using the implicit flow.
1. Get an authorization code
The authorization code flow begins with the client directing the user to the /authorize endpoint. This is the interactive part of the flow, where the user takes action. In this request, the client indicates in the scope parameter the permissions that it needs to acquire from the user. The following examples (with line breaks for readability) show how to acquire an authorization code. If you're testing this GET HTTP request, use your browser.
GET https://{tenant}.b2clogin.com/{tenant}.onmicrosoft.com/{policy}/oauth2/v2.0/authorize?
client_id=00001111-aaaa-2222-bbbb-3333cccc4444
&response_type=code
&redirect_uri=urn%3Aietf%3Awg%3Aoauth%3A2.0%3Aoob
&response_mode=query
&scope=00001111-aaaa-2222-bbbb-3333cccc4444%20offline_access%20https://{tenant-name}/{app-id-uri}/{scope}
&state=arbitrary_data_you_can_receive_in_the_response
&code_challenge=YTFjNjI1OWYzMzA3MTI4ZDY2Njg5M2RkNmVjNDE5YmEyZGRhOGYyM2IzNjdmZWFhMTQ1ODg3NDcxY2Nl
&code_challenge_method=S256
| Parameter | Required? | Description | 
|---|---|---|
| {tenant} | Required | Name of your Azure AD B2C tenant | 
| {policy} | Required | The user flow to be run. Specify the name of a user flow you've created in your Azure AD B2C tenant. For example: b2c_1_sign_in,b2c_1_sign_up, orb2c_1_edit_profile. | 
| client_id | Required | The application ID assigned to your app in the Azure portal. | 
| response_type | Required | The response type, which must include codefor the authorization code flow. You can receive an ID token if you include it in the response type, such ascode+id_token, and in this case, the scope needs to includeopenid. | 
| redirect_uri | Required | The redirect URI of your app, where authentication responses are sent and received by your app. It must exactly match one of the redirect URIs that you registered in the portal, except that it must be URL-encoded. | 
| scope | Required | A space-separated list of scopes. The openidscope indicates a permission to sign in the user and get data about the user in the form of ID tokens. Theoffline_accessscope is optional for web applications. It indicates that your application needs a refresh token for extended access to resources. The client-id indicates the token issued are intended for use by Azure AD B2C registered client. Thehttps://{tenant-name}/{app-id-uri}/{scope}indicates a permission to protected resources, such as a web API. For more information, see Request an access token. | 
| response_mode | Recommended | The method that you use to send the resulting authorization code back to your app. It can be query,form_post, orfragment. | 
| state | Recommended | A value included in the request that can be a string of any content that you want to use. Usually, a randomly generated unique value is used, to prevent cross-site request forgery attacks. The state also is used to encode information about the user's state in the app before the authentication request occurred. For example, the page the user was on, or the user flow that was being executed. | 
| prompt | Optional | The type of user interaction that is required. Currently, the only valid value is login, which forces the user to enter their credentials on that request. Single sign-on won't take effect. | 
| code_challenge | recommended / required | Used to secure authorization code grants via Proof Key for Code Exchange (PKCE). Required if code_challenge_methodis included. You need to add logic in your application to generate thecode_verifierandcode_challenge. Thecode_challengeis a Base64 URL-encoded SHA256 hash of thecode_verifier. You store thecode_verifierin your application for later use, and send thecode_challengealong with the authorization request. For more information, see the PKCE RFC. This is now recommended for all application types - native apps, SPAs, and confidential clients like web apps. | 
| code_challenge_method | recommended / required | The method used to encode the code_verifierfor thecode_challengeparameter. This SHOULD beS256, but the spec allows the use ofplainif for some reason the client can't support SHA256.If you exclude the code_challenge_method, but still include thecode_challenge, then thecode_challengeis assumed to be plaintext. Microsoft identity platform supports bothplainandS256. For more information, see the PKCE RFC. This is required for single page apps using the authorization code flow. | 
| login_hint | No | Can be used to prefill the sign-in name field of the sign-in page. For more information, see Prepopulate the sign-in name. | 
| domain_hint | No | Provides a hint to Azure AD B2C about the social identity provider that should be used for sign-in. If a valid value is included, the user goes directly to the identity provider sign-in page. For more information, see Redirect sign-in to a social provider. | 
| Custom parameters | No | Custom parameters that can be used with custom policies. For example, dynamic custom page content URI, or key-value claim resolvers. | 
At this point, the user is asked to complete the user flow's workflow. This might involve the user entering their username and password, signing in with a social identity, signing up for the directory, or any other number of steps. User actions depend on how the user flow is defined.
After the user completes the user flow, Microsoft Entra ID returns a response to your app at the value you used for redirect_uri. It uses the method specified in the response_mode parameter. The response is exactly the same for each of the user action scenarios, independent of the user flow that was executed.
A successful response that uses response_mode=query looks like this:
GET urn:ietf:wg:oauth:2.0:oob?
code=AwABAAAAvPM1KaPlrEqdFSBzjqfTGBCmLdgfSTLEMPGYuNHSUYBrq...        // the authorization_code, truncated
&state=arbitrary_data_you_can_receive_in_the_response                // the value provided in the request
| Parameter | Description | 
|---|---|
| code | The authorization code that the app requested. The app can use the authorization code to request an access token for a target resource. Authorization codes are short-lived. Typically, they expire after about 10 minutes. | 
| state | See the full description in the table in the preceding section. If a stateparameter is included in the request, the same value should appear in the response. The app should verify that thestatevalues in the request and response are identical. | 
Error responses also can be sent to the redirect URI so that the app can handle them appropriately:
GET urn:ietf:wg:oauth:2.0:oob?
error=access_denied
&error_description=The+user+has+cancelled+entering+self-asserted+information
&state=arbitrary_data_you_can_receive_in_the_response
| Parameter | Description | 
|---|---|
| error | An error code string that you can use to classify the types of errors that occur. You also can use the string to react to errors. | 
| error_description | A specific error message that can help you identify the root cause of an authentication error. | 
| state | See the full description in the preceding table. If a stateparameter is included in the request, the same value should appear in the response. The app should verify that thestatevalues in the request and response are identical. | 
2. Get an access token
Now that you've acquired an authorization code, you can redeem the code for a token to the intended resource by sending a POST request to the /token endpoint. In Azure AD B2C, you can request access tokens for other APIs as usual by specifying their scope(s) in the request.
You can also request an access token for your app's own back-end Web API by convention of using the app's client ID as the requested scope (which will result in an access token with that client ID as the "audience"):
POST https://{tenant}.b2clogin.com/{tenant}.onmicrosoft.com/{policy}/oauth2/v2.0/token HTTP/1.1
Content-Type: application/x-www-form-urlencoded
grant_type=authorization_code
&client_id=00001111-aaaa-2222-bbbb-3333cccc4444
&scope=00001111-aaaa-2222-bbbb-3333cccc4444 offline_access
&code=AwABAAAAvPM1KaPlrEqdFSBzjqfTGBCmLdgfSTLEMPGYuNHSUYBrq...
&redirect_uri=urn:ietf:wg:oauth:2.0:oob
&code_verifier=ThisIsntRandomButItNeedsToBe43CharactersLong 
| Parameter | Required? | Description | 
|---|---|---|
| {tenant} | Required | Name of your Azure AD B2C tenant | 
| {policy} | Required | The user flow that was used to acquire the authorization code. You can't use a different user flow in this request. | 
| client_id | Required | The application ID assigned to your app in the Azure portal. | 
| client_secret | Yes, in Web Apps | The application secret that was generated in the Azure portal. Client secrets are used in this flow for Web App scenarios, where the client can securely store a client secret. For Native App (public client) scenarios, client secrets can't be securely stored, and therefore aren't used in this call. If you use a client secret, change it on a periodic basis. | 
| grant_type | Required | The type of grant. For the authorization code flow, the grant type must be authorization_code. | 
| scope | Recommended | A space-separated list of scopes. A single scope value indicates to Azure AD B2C both of the permissions that are being requested. Using the client ID as the scope indicates that your app needs an access token that can be used against your own service or web API, represented by the same client ID.  The offline_accessscope indicates that your app needs a refresh token for long-lived access to resources.  You also can use theopenidscope to request an ID token from Azure AD B2C. | 
| code | Required | The authorization code that you acquired in from the /authorizeendpoint. | 
| redirect_uri | Required | The redirect URI of the application where you received the authorization code. | 
| code_verifier | recommended | The same code_verifierused to obtain the authorization code. Required if PKCE was used in the authorization code grant request. For more information, see the PKCE RFC. | 
If you're testing this POST HTTP request, you can use any HTTP client such as Microsoft PowerShell.
A successful token response looks like this:
{
    "not_before": "1442340812",
    "token_type": "Bearer",
    "access_token": "eyJ0eXAiOiJKV1QiLCJhbGciOiJSUzI1NiIsIng1dCI6Ik5HVEZ2ZEstZnl0aEV1Q...",
    "scope": "00001111-aaaa-2222-bbbb-3333cccc4444 offline_access",
    "expires_in": "3600",
    "refresh_token": "AAQfQmvuDy8WtUv-sd0TBwWVQs1rC-Lfxa_NDkLqpg50Cxp5Dxj0VPF1mx2Z...",
}
| Parameter | Description | 
|---|---|
| not_before | The time at which the token is considered valid, in epoch time. | 
| token_type | The token type value. The only type that Microsoft Entra ID supports is Bearer. | 
| access_token | The signed JSON Web Token (JWT) that you requested. | 
| scope | The scopes that the token is valid for. You also can use scopes to cache tokens for later use. | 
| expires_in | The length of time that the token is valid (in seconds). | 
| refresh_token | An OAuth 2.0 refresh token. The app can use this token to acquire additional tokens after the current token expires. Refresh tokens are long-lived. You can use them to retain access to resources for extended periods of time. For more information, see the Azure AD B2C token reference. | 
Error responses look like this:
{
    "error": "access_denied",
    "error_description": "The user revoked access to the app.",
}
| Parameter | Description | 
|---|---|
| error | An error code string that you can use to classify the types of errors that occur. You also can use the string to react to errors. | 
| error_description | A specific error message that can help you identify the root cause of an authentication error. | 
3. Use the token
Now that you've successfully acquired an access token, you can use the token in requests to your back-end web APIs by including it in the Authorization header:
GET /tasks
Host: mytaskwebapi.com
Authorization: Bearer eyJ0eXAiOiJKV1QiLCJhbGciOiJSUzI1NiIsIng1dCI6Ik5HVEZ2ZEstZnl0aEV1Q...
4. Refresh the token
Access tokens and ID tokens are short-lived. After they expire, you must refresh them to continue to access resources. When you refresh the access token, Azure AD B2C returns a new token. The refreshed access token will have updated nbf (not before), iat (issued at), and exp (expiration) claim values. All other claim values are the same as the originally issued access token.
To refresh the token, submit another POST request to the /token endpoint. This time, provide the refresh_token instead of the code:
POST https://{tenant}.b2clogin.com/{tenant}.onmicrosoft.com/{policy}/oauth2/v2.0/token HTTP/1.1
Content-Type: application/x-www-form-urlencoded
grant_type=refresh_token
&client_id=00001111-aaaa-2222-bbbb-3333cccc4444
&scope=00001111-aaaa-2222-bbbb-3333cccc4444 offline_access
&refresh_token=AwABAAAAvPM1KaPlrEqdFSBzjqfTGBCmLdgfSTLEMPGYuNHSUYBrq...
&redirect_uri=urn:ietf:wg:oauth:2.0:oob
| Parameter | Required? | Description | 
|---|---|---|
| {tenant} | Required | Name of your Azure AD B2C tenant | 
| {policy} | Required | The user flow that was used to acquire the original refresh token. You can't use a different user flow in this request. | 
| client_id | Required | The application ID assigned to your app in the Azure portal. | 
| client_secret | Yes, in Web Apps | The application secret that was generated in the Azure portal. Client secrets are used in this flow for Web App scenarios, where the client can securely store a client secret. For Native App (public client) scenarios, client secrets can't be securely stored, and therefore aren't used in this call. If you use a client secret, please change it on a periodic basis. | 
| grant_type | Required | The type of grant. For this leg of the authorization code flow, the grant type must be refresh_token. | 
| scope | Recommended | A space-separated list of scopes. A single scope value indicates to Microsoft Entra ID both of the permissions that are being requested. Using the client ID as the scope indicates that your app needs an access token that can be used against your own service or web API, represented by the same client ID.  The offline_accessscope indicates that your app needs a refresh token for long-lived access to resources.  You also can use theopenidscope to request an ID token from Azure AD B2C. | 
| redirect_uri | Optional | The redirect URI of the application where you received the authorization code. | 
| refresh_token | Required | The original refresh token that you acquired in the second leg of the flow. | 
A successful token response looks like this:
{
    "not_before": "1442340812",
    "token_type": "Bearer",
    "access_token": "eyJ0eXAiOiJKV1QiLCJhbGciOiJSUzI1NiIsIng1dCI6Ik5HVEZ2ZEstZnl0aEV1Q...",
    "scope": "00001111-aaaa-2222-bbbb-3333cccc4444 offline_access",
    "expires_in": "3600",
    "refresh_token": "AAQfQmvuDy8WtUv-sd0TBwWVQs1rC-Lfxa_NDkLqpg50Cxp5Dxj0VPF1mx2Z...",
}
| Parameter | Description | 
|---|---|
| not_before | The time at which the token is considered valid, in epoch time. | 
| token_type | The token type value. The only type that Microsoft Entra ID supports is Bearer. | 
| access_token | The signed JWT that you requested. | 
| scope | The scopes that the token is valid for. You also can use the scopes to cache tokens for later use. | 
| expires_in | The length of time that the token is valid (in seconds). | 
| refresh_token | An OAuth 2.0 refresh token. The app can use this token to acquire additional tokens after the current token expires. Refresh tokens are long-lived and can be used to retain access to resources for extended periods of time. For more information, see the Azure AD B2C token reference. | 
Error responses look like this:
{
    "error": "access_denied",
    "error_description": "The user revoked access to the app.",
}
| Parameter | Description | 
|---|---|
| error | An error code string that you can use to classify types of errors that occur. You also can use the string to react to errors. | 
| error_description | A specific error message that can help you identify the root cause of an authentication error. | 
Use your own Azure AD B2C directory
To try these requests yourself, complete the following steps. Replace the example values we used in this article with your own values.
- Create an Azure AD B2C directory. Use the name of your directory in the requests.
- Create an application to obtain an application ID and a redirect URI. Include a native client in your app.
- Create your user flows to obtain your user flow names.