-
Notifications
You must be signed in to change notification settings - Fork 214
AuthenticationContext: the connection to Azure AD
ADAL.NET, has one class representing a connection to Azure AD: AuthenticationContext.
An AuthenticationContext
represents the authority you want to use for gaining access to resources (ie the authority you refer to when you need tokens). Contrary to MSAL, AuthenticationContext
does not even represent an Azure AD v1 application: as you can see in the class diagram representation below ClientID
(ApplicationID) of the application is not passed at the construction of this class, but needs to be passed in all (AcquireTokenXXX) requests, and sometimes even in the constructor of data structures. The AuthenticationContext is really:
- a connection to the Security Token Service (STS) or authorization server , through the Authority.
- and a token cache.
It might be useful to think of the Authority as the source of identities/tokens, in the business sense: I am getting tokens from Contoso. Now Contoso can choose to surface its issuing capacity as an ADFS instance, or as a cloud tenant. Examples of clouds are the Microsoft Cloud, national clouds like the German cloud, or the Chinese Cloud, or even sovereign clouds, like the US government cloud
- All the constructors of AuthenticationContext take the authority URL as first parameter.
- Overrides of the first constructor also enable application developers to:
- By pass the authority validation (by setting the validateAuthority parameter to false)
- Set their own token cache in cases where the application needs or wants to manage the token serialization itself. A token cache is provided by default, with serialization in some platforms. Serialization can be customized for .NET Framework and .NET Core and even needs to be customized in these platforms to ensure persistence of the token cache. [See more in paragraph about Broker on Android and iOS ]
The authority needs to be set to the URL to the STS. Examples of valid authority are:
-
https://login.microsoftonline.com/f31e6716-26e8-4651-b323-2563936b4163
for a single tenant application defined in the tenant whichTenantId
isf31e6716-26e8-4651-b323-2563936b4163
-
https://login.microsoftonline.com/contoso.onmicrosoft.com
. This representation is like the previous one, but uses the tenant domain name instead of the tenant Id. -
https://login.microsoftonline.de/contoso.de
also uses a domain name, but in this case the Azure AD tenant admins have set a custom domain for their tenant. And this one is for the German national Cloud -
https://login.microsoftonline.com/common
in the case of a multi-tenant application, that is an application available in several Azure AD tenants - It can finally be an Active Directory Federation Services (ADFS) URL, which is recognized with the convention that the URL should contain
adfs
likehttps://contoso.com/adfs
.
Note that the authority might also be an Azure AD B2C tenant, but ADAL does not support B2C.
In the case of a multi-tenant application, the instanciation of the AuthenticationContext is done in the following way:
var authenticationContext = new AuthenticationContext("https://login.microsoftonline.com/common");
var authenticationResult = authenticationContext.AcquireTokenAsync( .... );
however, if you do only that, next time you call AcquireTokenSilentAsync
, the cache won't be hit as you got an access token and a refresh token for the real tenant Id, and it was cached, whereas here you are requesting it for common.
The right pattern to use in multi-tenant applications is to create a new AuthenticationContext with the authority containing the real tenant ID. In practice this is different depending on the platforms, as the mobile platforms offer a default cache serialization, whereas .NET Framework and .NET core don't. See Token cache serialization
var authenticationContext = new AuthenticationContext("https://login.microsoftonline.com/common");
var authenticationResult = authenticationContext.AcquireTokenAsync( .... );
// set the same authenticationContext variable. This time the app knows the tenant
authenticationContext = new AuthenticationContext(authenticationResult.Authority);
In the case of .NET Framework and .NET Core applications, assuming you wrote a method named DeserializeCache
to deserialize the token cache (as explained in Token cache serialization), the pattern is the following:
// Case of multi-tenant applications. We don't know yet what will be the tenant.
string commonAuthority = "https://login.microsoftonline.com/common";
// In the case you deserialize the token cache, you can already instanciate the authority based on the
// authority of the previously cached token
TokenCache tokenCache = DeserializeTokenCache();
// Setting the authority
string authority;
var cacheElements = tokenCache.ReadItems();
var authorities = cacheElements.Select(ce => ce.Authority).Distinct();
if (cacheElements.Any())
{
authority = authorities.First(); // or choose one ?
}
else
{
authority = commonAuthority;
}
// Instanciate the cache with the right authority
AuthenticationContext authenticationContext = new AuthenticationContext(authority, tokenCache);
// Acquire a token
AuthenticationResult authenticationResult = await authenticationContext.AcquireTokenAsync(...);
// Case where there was no cached token. the authority is still the common authority. We need to
// re-instanciate an AuthenticationContext with the real (tenanted) authority
if (authority == commonAuthority)
{
// We now know the tenant (it's in authenticationResult.Authority
authenticationContext = new AuthenticationContext(authenticationResult.Authority, tokenCache);
}
In the past, the Azure AD authority URL used to be https://login.windows.net. It has then changed to https://login.microsoftonline.com. Of course, there are still applications (for instance Office) which use the old base URL, whereas others use the new base URL. Azure AD maintains a dictionary of authority aliases and will both automatically redirect your application to https://login.microsoftonline.com and provide the same tokens for the aliases. This redirection had, in past versions of ADAL, the effect of triggering un-necessary sign-ins for users - multiple authentication prompts, possibly within the same application, and "islands of SSO" effects when multiple applications were involved. The root cause of this behavior is that the ADAL libraries were using the authority URL passed in the AuthenticationContext constructor as a key for the token cache and were not aware of this aliasing. To avoid un-necessary authentication user prompts, recent versions of ADAL (for .NET from ADAL.NET 3.18) start by downloading the authority aliases dictionary from the STS and then take them into account when building the token cache keys.
Authentication context exposes other properties in addition to an accessor to the already mentioned constructor's parameters (Authority
, ValidateAuthority
and TokenCache
). Those are:
-
CorrelationId
is a GUID that the application developers can set, and which will be passed through all the interactions with the STS and the Web APIs, as well as in logs and telemetry. This enables them to diagnose issues. This can also enable Microsoft to provide advanced support. -
ExtendedLifeTimeEnabled
is a Boolean that first party applications (read Office) can set to true in case when the STS has an outage, to be more resilient. Indeed, Azure AD returns the token with an expiration time, and also with an extended expiration time. The tokens are automatically refreshed by ADAL and MSAL when the time is more than the expiration time, except when ExtendedLifeTimeEnabled is true and the time is less than the extended expiration time. This goes in pair with Web APIs middleware which, when this extended life time is enabled, can accept slightly expired tokens.
- Home
- Why use ADAL.NET?
- Register your app with AAD
- AuthenticationContext
- Acquiring Tokens
- Calling a protected API
- Acquiring a token interactively
- Acquiring tokens silently
- Using Device Code Flow
- Using Embedded Webview and System Browser in ADAL.NET and MSAL.NET
- With no user
- In the name of a user
- on behalf of (Service to service calls)
- by authorization code (Web Apps)
- Use async controller actions
- Exception types
- using Broker on iOS and Android
- Logging
- Token Cache serialization
- User management
- Using ADAL with a proxy
- Authentication context in multi-tenant scenarios
- Troubleshooting MFA in a WebApp or Web API
- Provide your own HttpClient
- iOS Keychain Access