UNPKG

@azure/msal-browser

Version:
1,259 lines (1,163 loc) 93.5 kB
/* * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the MIT License. */ import { CryptoOps } from "../crypto/CryptoOps.js"; import { InteractionRequiredAuthError, AccountInfo, Constants, INetworkModule, Logger, CommonSilentFlowRequest, ICrypto, DEFAULT_CRYPTO_IMPLEMENTATION, AuthError, PerformanceEvents, PerformanceCallbackFunction, IPerformanceClient, BaseAuthRequest, PromptValue, InProgressPerformanceEvent, getRequestThumbprint, AccountEntity, invokeAsync, createClientAuthError, ClientAuthErrorCodes, AccountFilter, buildStaticAuthorityOptions, InteractionRequiredAuthErrorCodes, PkceCodes, } from "@azure/msal-common/browser"; import { BrowserCacheManager, DEFAULT_BROWSER_CACHE_MANAGER, } from "../cache/BrowserCacheManager.js"; import * as AccountManager from "../cache/AccountManager.js"; import { BrowserConfiguration, CacheOptions } from "../config/Configuration.js"; import { InteractionType, ApiId, BrowserCacheLocation, WrapperSKU, CacheLookupPolicy, DEFAULT_REQUEST, BrowserConstants, iFrameRenewalPolicies, INTERACTION_TYPE, } from "../utils/BrowserConstants.js"; import * as BrowserUtils from "../utils/BrowserUtils.js"; import { RedirectRequest } from "../request/RedirectRequest.js"; import { PopupRequest } from "../request/PopupRequest.js"; import { SsoSilentRequest } from "../request/SsoSilentRequest.js"; import { EventCallbackFunction, EventError } from "../event/EventMessage.js"; import { EventType } from "../event/EventType.js"; import { EndSessionRequest } from "../request/EndSessionRequest.js"; import { EndSessionPopupRequest } from "../request/EndSessionPopupRequest.js"; import { INavigationClient } from "../navigation/INavigationClient.js"; import { EventHandler } from "../event/EventHandler.js"; import { PopupClient } from "../interaction_client/PopupClient.js"; import { RedirectClient } from "../interaction_client/RedirectClient.js"; import { SilentIframeClient } from "../interaction_client/SilentIframeClient.js"; import { SilentRefreshClient } from "../interaction_client/SilentRefreshClient.js"; import { TokenCache } from "../cache/TokenCache.js"; import { ITokenCache } from "../cache/ITokenCache.js"; import { NativeInteractionClient } from "../interaction_client/NativeInteractionClient.js"; import { NativeMessageHandler } from "../broker/nativeBroker/NativeMessageHandler.js"; import { SilentRequest } from "../request/SilentRequest.js"; import { NativeAuthError, isFatalNativeAuthError, } from "../error/NativeAuthError.js"; import { SilentCacheClient } from "../interaction_client/SilentCacheClient.js"; import { SilentAuthCodeClient } from "../interaction_client/SilentAuthCodeClient.js"; import { createBrowserAuthError, BrowserAuthErrorCodes, } from "../error/BrowserAuthError.js"; import { AuthorizationCodeRequest } from "../request/AuthorizationCodeRequest.js"; import { NativeTokenRequest } from "../broker/nativeBroker/NativeRequest.js"; import { StandardOperatingContext } from "../operatingcontext/StandardOperatingContext.js"; import { BaseOperatingContext } from "../operatingcontext/BaseOperatingContext.js"; import { IController } from "./IController.js"; import { AuthenticationResult } from "../response/AuthenticationResult.js"; import { ClearCacheRequest } from "../request/ClearCacheRequest.js"; import { createNewGuid } from "../crypto/BrowserCrypto.js"; import { initializeSilentRequest } from "../request/RequestHelpers.js"; import { InitializeApplicationRequest } from "../request/InitializeApplicationRequest.js"; import { generatePkceCodes } from "../crypto/PkceGenerator.js"; import { collectInstanceStats } from "../utils/MsalFrameStatsUtils.js"; function getAccountType( account?: AccountInfo ): "AAD" | "MSA" | "B2C" | undefined { const idTokenClaims = account?.idTokenClaims; if (idTokenClaims?.tfp || idTokenClaims?.acr) { return "B2C"; } if (!idTokenClaims?.tid) { return undefined; } else if (idTokenClaims?.tid === "9188040d-6c67-4c5b-b112-36a304b66dad") { return "MSA"; } return "AAD"; } function preflightCheck( initialized: boolean, performanceEvent: InProgressPerformanceEvent ) { try { BrowserUtils.preflightCheck(initialized); } catch (e) { performanceEvent.end({ success: false }, e); throw e; } } export class StandardController implements IController { // OperatingContext protected readonly operatingContext: StandardOperatingContext; // Crypto interface implementation protected readonly browserCrypto: ICrypto; // Storage interface implementation protected readonly browserStorage: BrowserCacheManager; // Native Cache in memory storage implementation protected readonly nativeInternalStorage: BrowserCacheManager; // Network interface implementation protected readonly networkClient: INetworkModule; // Navigation interface implementation protected navigationClient: INavigationClient; // Input configuration by developer/user protected readonly config: BrowserConfiguration; // Token cache implementation private tokenCache: TokenCache; // Logger protected logger: Logger; // Flag to indicate if in browser environment protected isBrowserEnvironment: boolean; protected readonly eventHandler: EventHandler; // Redirect Response Object protected readonly redirectResponse: Map< string, Promise<AuthenticationResult | null> >; // Native Extension Provider protected nativeExtensionProvider: NativeMessageHandler | undefined; // Hybrid auth code responses private hybridAuthCodeResponses: Map<string, Promise<AuthenticationResult>>; // Performance telemetry client protected readonly performanceClient: IPerformanceClient; // Flag representing whether or not the initialize API has been called and completed protected initialized: boolean; // Active requests private activeSilentTokenRequests: Map< string, Promise<AuthenticationResult> >; // Active Iframe request private activeIframeRequest: [Promise<boolean>, string] | undefined; private ssoSilentMeasurement?: InProgressPerformanceEvent; private acquireTokenByCodeAsyncMeasurement?: InProgressPerformanceEvent; private pkceCode: PkceCodes | undefined; /** * @constructor * Constructor for the PublicClientApplication used to instantiate the PublicClientApplication object * * Important attributes in the Configuration object for auth are: * - clientID: the application ID of your application. You can obtain one by registering your application with our Application registration portal : https://portal.azure.com/#blade/Microsoft_AAD_IAM/ActiveDirectoryMenuBlade/RegisteredAppsPreview * - authority: the authority URL for your application. * - redirect_uri: the uri of your application registered in the portal. * * In Azure AD, authority is a URL indicating the Azure active directory that MSAL uses to obtain tokens. * It is of the form https://login.microsoftonline.com/{Enter_the_Tenant_Info_Here} * If your application supports Accounts in one organizational directory, replace "Enter_the_Tenant_Info_Here" value with the Tenant Id or Tenant name (for example, contoso.microsoft.com). * If your application supports Accounts in any organizational directory, replace "Enter_the_Tenant_Info_Here" value with organizations. * If your application supports Accounts in any organizational directory and personal Microsoft accounts, replace "Enter_the_Tenant_Info_Here" value with common. * To restrict support to Personal Microsoft accounts only, replace "Enter_the_Tenant_Info_Here" value with consumers. * * In Azure B2C, authority is of the form https://{instance}/tfp/{tenant}/{policyName}/ * Full B2C functionality will be available in this library in future versions. * * @param configuration Object for the MSAL PublicClientApplication instance */ constructor(operatingContext: StandardOperatingContext) { this.operatingContext = operatingContext; this.isBrowserEnvironment = this.operatingContext.isBrowserEnvironment(); // Set the configuration. this.config = operatingContext.getConfig(); this.initialized = false; // Initialize logger this.logger = this.operatingContext.getLogger(); // Initialize the network module class. this.networkClient = this.config.system.networkClient; // Initialize the navigation client class. this.navigationClient = this.config.system.navigationClient; // Initialize redirectResponse Map this.redirectResponse = new Map(); // Initial hybrid spa map this.hybridAuthCodeResponses = new Map(); // Initialize performance client this.performanceClient = this.config.telemetry.client; // Initialize the crypto class. this.browserCrypto = this.isBrowserEnvironment ? new CryptoOps(this.logger, this.performanceClient) : DEFAULT_CRYPTO_IMPLEMENTATION; this.eventHandler = new EventHandler(this.logger); // Initialize the browser storage class. this.browserStorage = this.isBrowserEnvironment ? new BrowserCacheManager( this.config.auth.clientId, this.config.cache, this.browserCrypto, this.logger, this.performanceClient, this.eventHandler, buildStaticAuthorityOptions(this.config.auth) ) : DEFAULT_BROWSER_CACHE_MANAGER( this.config.auth.clientId, this.logger, this.performanceClient, this.eventHandler ); // initialize in memory storage for native flows const nativeCacheOptions: Required<CacheOptions> = { cacheLocation: BrowserCacheLocation.MemoryStorage, temporaryCacheLocation: BrowserCacheLocation.MemoryStorage, storeAuthStateInCookie: false, secureCookies: false, cacheMigrationEnabled: false, claimsBasedCachingEnabled: false, }; this.nativeInternalStorage = new BrowserCacheManager( this.config.auth.clientId, nativeCacheOptions, this.browserCrypto, this.logger, this.performanceClient, this.eventHandler ); // Initialize the token cache this.tokenCache = new TokenCache( this.config, this.browserStorage, this.logger, this.browserCrypto ); this.activeSilentTokenRequests = new Map(); // Register listener functions this.trackPageVisibility = this.trackPageVisibility.bind(this); // Register listener functions this.trackPageVisibilityWithMeasurement = this.trackPageVisibilityWithMeasurement.bind(this); } static async createController( operatingContext: BaseOperatingContext, request?: InitializeApplicationRequest ): Promise<IController> { const controller = new StandardController(operatingContext); await controller.initialize(request); return controller; } private trackPageVisibility(correlationId?: string): void { if (!correlationId) { return; } this.logger.info("Perf: Visibility change detected"); this.performanceClient.incrementFields( { visibilityChangeCount: 1 }, correlationId ); } /** * Initializer function to perform async startup tasks such as connecting to WAM extension * @param request {?InitializeApplicationRequest} correlation id */ async initialize( request?: InitializeApplicationRequest, isBroker?: boolean ): Promise<void> { this.logger.trace("initialize called"); if (this.initialized) { this.logger.info( "initialize has already been called, exiting early." ); return; } if (!this.isBrowserEnvironment) { this.logger.info("in non-browser environment, exiting early."); this.initialized = true; this.eventHandler.emitEvent(EventType.INITIALIZE_END); return; } const initCorrelationId = request?.correlationId || this.getRequestCorrelationId(); const allowPlatformBroker = this.config.system.allowPlatformBroker; const initMeasurement = this.performanceClient.startMeasurement( PerformanceEvents.InitializeClientApplication, initCorrelationId ); this.eventHandler.emitEvent(EventType.INITIALIZE_START); // Broker applications are initialized twice, so we avoid double-counting it if (!isBroker) { try { this.logMultipleInstances(initMeasurement); } catch {} } await invokeAsync( this.browserStorage.initialize.bind(this.browserStorage), PerformanceEvents.InitializeCache, this.logger, this.performanceClient, initCorrelationId )(initCorrelationId); if (allowPlatformBroker) { try { this.nativeExtensionProvider = await NativeMessageHandler.createProvider( this.logger, this.config.system.nativeBrokerHandshakeTimeout, this.performanceClient ); } catch (e) { this.logger.verbose(e as string); } } if (!this.config.cache.claimsBasedCachingEnabled) { this.logger.verbose( "Claims-based caching is disabled. Clearing the previous cache with claims" ); await invokeAsync( this.browserStorage.clearTokensAndKeysWithClaims.bind( this.browserStorage ), PerformanceEvents.ClearTokensAndKeysWithClaims, this.logger, this.performanceClient, initCorrelationId )(this.performanceClient, initCorrelationId); } this.config.system.asyncPopups && (await this.preGeneratePkceCodes(initCorrelationId)); this.initialized = true; this.eventHandler.emitEvent(EventType.INITIALIZE_END); initMeasurement.end({ allowPlatformBroker: allowPlatformBroker, success: true, }); } // #region Redirect Flow /** * Event handler function which allows users to fire events after the PublicClientApplication object * has loaded during redirect flows. This should be invoked on all page loads involved in redirect * auth flows. * @param hash Hash to process. Defaults to the current value of window.location.hash. Only needs to be provided explicitly if the response to be handled is not contained in the current value. * @returns Token response or null. If the return value is null, then no auth redirect was detected. */ async handleRedirectPromise( hash?: string ): Promise<AuthenticationResult | null> { this.logger.verbose("handleRedirectPromise called"); // Block token acquisition before initialize has been called BrowserUtils.blockAPICallsBeforeInitialize(this.initialized); if (this.isBrowserEnvironment) { /** * Store the promise on the PublicClientApplication instance if this is the first invocation of handleRedirectPromise, * otherwise return the promise from the first invocation. Prevents race conditions when handleRedirectPromise is called * several times concurrently. */ const redirectResponseKey = hash || ""; let response = this.redirectResponse.get(redirectResponseKey); if (typeof response === "undefined") { response = this.handleRedirectPromiseInternal(hash); this.redirectResponse.set(redirectResponseKey, response); this.logger.verbose( "handleRedirectPromise has been called for the first time, storing the promise" ); } else { this.logger.verbose( "handleRedirectPromise has been called previously, returning the result from the first call" ); } return response; } this.logger.verbose( "handleRedirectPromise returns null, not browser environment" ); return null; } /** * The internal details of handleRedirectPromise. This is separated out to a helper to allow handleRedirectPromise to memoize requests * @param hash * @returns */ private async handleRedirectPromiseInternal( hash?: string ): Promise<AuthenticationResult | null> { if (!this.browserStorage.isInteractionInProgress(true)) { this.logger.info( "handleRedirectPromise called but there is no interaction in progress, returning null." ); return null; } const interactionType = this.browserStorage.getInteractionInProgress()?.type; if (interactionType === INTERACTION_TYPE.SIGNOUT) { this.logger.verbose( "handleRedirectPromise removing interaction_in_progress flag and returning null after sign-out" ); this.browserStorage.setInteractionInProgress(false); return Promise.resolve(null); } const loggedInAccounts = this.getAllAccounts(); const platformBrokerRequest: NativeTokenRequest | null = this.browserStorage.getCachedNativeRequest(); const useNative = platformBrokerRequest && NativeMessageHandler.isPlatformBrokerAvailable( this.config, this.logger, this.nativeExtensionProvider ) && this.nativeExtensionProvider && !hash; let rootMeasurement: InProgressPerformanceEvent; this.eventHandler.emitEvent( EventType.HANDLE_REDIRECT_START, InteractionType.Redirect ); let redirectResponse: Promise<AuthenticationResult | null>; try { if (useNative && this.nativeExtensionProvider) { rootMeasurement = this.performanceClient.startMeasurement( PerformanceEvents.AcquireTokenRedirect, platformBrokerRequest?.correlationId || "" ); this.logger.trace( "handleRedirectPromise - acquiring token from native platform" ); const nativeClient = new NativeInteractionClient( this.config, this.browserStorage, this.browserCrypto, this.logger, this.eventHandler, this.navigationClient, ApiId.handleRedirectPromise, this.performanceClient, this.nativeExtensionProvider, platformBrokerRequest.accountId, this.nativeInternalStorage, platformBrokerRequest.correlationId ); redirectResponse = invokeAsync( nativeClient.handleRedirectPromise.bind(nativeClient), PerformanceEvents.HandleNativeRedirectPromiseMeasurement, this.logger, this.performanceClient, rootMeasurement.event.correlationId )(this.performanceClient, rootMeasurement.event.correlationId); } else { const [standardRequest, codeVerifier] = this.browserStorage.getCachedRequest(); const correlationId = standardRequest.correlationId; // Reset rootMeasurement now that we have correlationId rootMeasurement = this.performanceClient.startMeasurement( PerformanceEvents.AcquireTokenRedirect, correlationId ); this.logger.trace( "handleRedirectPromise - acquiring token from web flow" ); const redirectClient = this.createRedirectClient(correlationId); redirectResponse = invokeAsync( redirectClient.handleRedirectPromise.bind(redirectClient), PerformanceEvents.HandleRedirectPromiseMeasurement, this.logger, this.performanceClient, rootMeasurement.event.correlationId )(hash, standardRequest, codeVerifier, rootMeasurement); } } catch (e) { this.browserStorage.resetRequestCache(); throw e; } return redirectResponse .then((result: AuthenticationResult | null) => { if (result) { this.browserStorage.resetRequestCache(); // Emit login event if number of accounts change const isLoggingIn = loggedInAccounts.length < this.getAllAccounts().length; if (isLoggingIn) { this.eventHandler.emitEvent( EventType.LOGIN_SUCCESS, InteractionType.Redirect, result ); this.logger.verbose( "handleRedirectResponse returned result, login success" ); } else { this.eventHandler.emitEvent( EventType.ACQUIRE_TOKEN_SUCCESS, InteractionType.Redirect, result ); this.logger.verbose( "handleRedirectResponse returned result, acquire token success" ); } rootMeasurement.end({ success: true, accountType: getAccountType(result.account), }); } else { /* * Instrument an event only if an error code is set. Otherwise, discard it when the redirect response * is empty and the error code is missing. */ if (rootMeasurement.event.errorCode) { rootMeasurement.end({ success: false }); } else { rootMeasurement.discard(); } } this.eventHandler.emitEvent( EventType.HANDLE_REDIRECT_END, InteractionType.Redirect ); return result; }) .catch((e) => { this.browserStorage.resetRequestCache(); const eventError = e as EventError; // Emit login event if there is an account if (loggedInAccounts.length > 0) { this.eventHandler.emitEvent( EventType.ACQUIRE_TOKEN_FAILURE, InteractionType.Redirect, null, eventError ); } else { this.eventHandler.emitEvent( EventType.LOGIN_FAILURE, InteractionType.Redirect, null, eventError ); } this.eventHandler.emitEvent( EventType.HANDLE_REDIRECT_END, InteractionType.Redirect ); rootMeasurement.end( { success: false, }, eventError ); throw e; }); } /** * Use when you want to obtain an access_token for your API by redirecting the user's browser window to the authorization endpoint. This function redirects * the page, so any code that follows this function will not execute. * * IMPORTANT: It is NOT recommended to have code that is dependent on the resolution of the Promise. This function will navigate away from the current * browser window. It currently returns a Promise in order to reflect the asynchronous nature of the code running in this function. * * @param request */ async acquireTokenRedirect(request: RedirectRequest): Promise<void> { // Preflight request const correlationId = this.getRequestCorrelationId(request); this.logger.verbose("acquireTokenRedirect called", correlationId); const atrMeasurement = this.performanceClient.startMeasurement( PerformanceEvents.AcquireTokenPreRedirect, correlationId ); atrMeasurement.add({ accountType: getAccountType(request.account), scenarioId: request.scenarioId, }); // Override on request only if set, as onRedirectNavigate field is deprecated const onRedirectNavigateCb = request.onRedirectNavigate; if (onRedirectNavigateCb) { request.onRedirectNavigate = (url: string) => { const navigate = typeof onRedirectNavigateCb === "function" ? onRedirectNavigateCb(url) : undefined; if (navigate !== false) { atrMeasurement.end({ success: true }); } else { atrMeasurement.discard(); } return navigate; }; } else { const configOnRedirectNavigateCb = this.config.auth.onRedirectNavigate; this.config.auth.onRedirectNavigate = (url: string) => { const navigate = typeof configOnRedirectNavigateCb === "function" ? configOnRedirectNavigateCb(url) : undefined; if (navigate !== false) { atrMeasurement.end({ success: true }); } else { atrMeasurement.discard(); } return navigate; }; } // If logged in, emit acquire token events const isLoggedIn = this.getAllAccounts().length > 0; try { BrowserUtils.redirectPreflightCheck(this.initialized, this.config); this.browserStorage.setInteractionInProgress( true, INTERACTION_TYPE.SIGNIN ); if (isLoggedIn) { this.eventHandler.emitEvent( EventType.ACQUIRE_TOKEN_START, InteractionType.Redirect, request ); } else { this.eventHandler.emitEvent( EventType.LOGIN_START, InteractionType.Redirect, request ); } let result: Promise<void>; if ( this.nativeExtensionProvider && this.canUsePlatformBroker(request) ) { const nativeClient = new NativeInteractionClient( this.config, this.browserStorage, this.browserCrypto, this.logger, this.eventHandler, this.navigationClient, ApiId.acquireTokenRedirect, this.performanceClient, this.nativeExtensionProvider, this.getNativeAccountId(request), this.nativeInternalStorage, correlationId ); result = nativeClient .acquireTokenRedirect(request, atrMeasurement) .catch((e: AuthError) => { if ( e instanceof NativeAuthError && isFatalNativeAuthError(e) ) { this.nativeExtensionProvider = undefined; // If extension gets uninstalled during session prevent future requests from continuing to attempt const redirectClient = this.createRedirectClient(correlationId); return redirectClient.acquireToken(request); } else if (e instanceof InteractionRequiredAuthError) { this.logger.verbose( "acquireTokenRedirect - Resolving interaction required error thrown by native broker by falling back to web flow" ); const redirectClient = this.createRedirectClient(correlationId); return redirectClient.acquireToken(request); } throw e; }); } else { const redirectClient = this.createRedirectClient(correlationId); result = redirectClient.acquireToken(request); } return await result; } catch (e) { this.browserStorage.resetRequestCache(); atrMeasurement.end({ success: false }, e); if (isLoggedIn) { this.eventHandler.emitEvent( EventType.ACQUIRE_TOKEN_FAILURE, InteractionType.Redirect, null, e as EventError ); } else { this.eventHandler.emitEvent( EventType.LOGIN_FAILURE, InteractionType.Redirect, null, e as EventError ); } throw e; } } // #endregion // #region Popup Flow /** * Use when you want to obtain an access_token for your API via opening a popup window in the user's browser * * @param request * * @returns A promise that is fulfilled when this function has completed, or rejected if an error was raised. */ acquireTokenPopup(request: PopupRequest): Promise<AuthenticationResult> { const correlationId = this.getRequestCorrelationId(request); const atPopupMeasurement = this.performanceClient.startMeasurement( PerformanceEvents.AcquireTokenPopup, correlationId ); atPopupMeasurement.add({ scenarioId: request.scenarioId, accountType: getAccountType(request.account), }); try { this.logger.verbose("acquireTokenPopup called", correlationId); preflightCheck(this.initialized, atPopupMeasurement); this.browserStorage.setInteractionInProgress( true, INTERACTION_TYPE.SIGNIN ); } catch (e) { // Since this function is syncronous we need to reject return Promise.reject(e); } // If logged in, emit acquire token events const loggedInAccounts = this.getAllAccounts(); if (loggedInAccounts.length > 0) { this.eventHandler.emitEvent( EventType.ACQUIRE_TOKEN_START, InteractionType.Popup, request ); } else { this.eventHandler.emitEvent( EventType.LOGIN_START, InteractionType.Popup, request ); } let result: Promise<AuthenticationResult>; const pkce = this.getPreGeneratedPkceCodes(correlationId); if (this.canUsePlatformBroker(request)) { result = this.acquireTokenNative( { ...request, correlationId, }, ApiId.acquireTokenPopup ) .then((response) => { atPopupMeasurement.end({ success: true, isNativeBroker: true, accountType: getAccountType(response.account), }); return response; }) .catch((e: AuthError) => { if ( e instanceof NativeAuthError && isFatalNativeAuthError(e) ) { this.nativeExtensionProvider = undefined; // If extension gets uninstalled during session prevent future requests from continuing to attempt const popupClient = this.createPopupClient(correlationId); return popupClient.acquireToken(request, pkce); } else if (e instanceof InteractionRequiredAuthError) { this.logger.verbose( "acquireTokenPopup - Resolving interaction required error thrown by native broker by falling back to web flow" ); const popupClient = this.createPopupClient(correlationId); return popupClient.acquireToken(request, pkce); } throw e; }); } else { const popupClient = this.createPopupClient(correlationId); result = popupClient.acquireToken(request, pkce); } return result .then((result) => { /* * If logged in, emit acquire token events */ const isLoggingIn = loggedInAccounts.length < this.getAllAccounts().length; if (isLoggingIn) { this.eventHandler.emitEvent( EventType.LOGIN_SUCCESS, InteractionType.Popup, result ); } else { this.eventHandler.emitEvent( EventType.ACQUIRE_TOKEN_SUCCESS, InteractionType.Popup, result ); } atPopupMeasurement.end({ success: true, accessTokenSize: result.accessToken.length, idTokenSize: result.idToken.length, accountType: getAccountType(result.account), }); return result; }) .catch((e: Error) => { if (loggedInAccounts.length > 0) { this.eventHandler.emitEvent( EventType.ACQUIRE_TOKEN_FAILURE, InteractionType.Popup, null, e ); } else { this.eventHandler.emitEvent( EventType.LOGIN_FAILURE, InteractionType.Popup, null, e ); } atPopupMeasurement.end( { success: false, }, e ); // Since this function is syncronous we need to reject return Promise.reject(e); }) .finally(async () => { this.browserStorage.setInteractionInProgress(false); if (this.config.system.asyncPopups) { await this.preGeneratePkceCodes(correlationId); } }); } private trackPageVisibilityWithMeasurement(): void { const measurement = this.ssoSilentMeasurement || this.acquireTokenByCodeAsyncMeasurement; if (!measurement) { return; } this.logger.info( "Perf: Visibility change detected in ", measurement.event.name ); measurement.increment({ visibilityChangeCount: 1, }); } // #endregion // #region Silent Flow /** * This function uses a hidden iframe to fetch an authorization code from the eSTS. There are cases where this may not work: * - Any browser using a form of Intelligent Tracking Prevention * - If there is not an established session with the service * * In these cases, the request must be done inside a popup or full frame redirect. * * For the cases where interaction is required, you cannot send a request with prompt=none. * * If your refresh token has expired, you can use this function to fetch a new set of tokens silently as long as * you session on the server still exists. * @param request {@link SsoSilentRequest} * * @returns A promise that is fulfilled when this function has completed, or rejected if an error was raised. */ async ssoSilent(request: SsoSilentRequest): Promise<AuthenticationResult> { const correlationId = this.getRequestCorrelationId(request); const validRequest = { ...request, // will be PromptValue.NONE or PromptValue.NO_SESSION prompt: request.prompt, correlationId: correlationId, }; this.ssoSilentMeasurement = this.performanceClient.startMeasurement( PerformanceEvents.SsoSilent, correlationId ); this.ssoSilentMeasurement?.add({ scenarioId: request.scenarioId, accountType: getAccountType(request.account), }); preflightCheck(this.initialized, this.ssoSilentMeasurement); this.ssoSilentMeasurement?.increment({ visibilityChangeCount: 0, }); document.addEventListener( "visibilitychange", this.trackPageVisibilityWithMeasurement ); this.logger.verbose("ssoSilent called", correlationId); this.eventHandler.emitEvent( EventType.SSO_SILENT_START, InteractionType.Silent, validRequest ); let result: Promise<AuthenticationResult>; if (this.canUsePlatformBroker(validRequest)) { result = this.acquireTokenNative( validRequest, ApiId.ssoSilent ).catch((e: AuthError) => { // If native token acquisition fails for availability reasons fallback to standard flow if (e instanceof NativeAuthError && isFatalNativeAuthError(e)) { this.nativeExtensionProvider = undefined; // If extension gets uninstalled during session prevent future requests from continuing to attempt const silentIframeClient = this.createSilentIframeClient( validRequest.correlationId ); return silentIframeClient.acquireToken(validRequest); } throw e; }); } else { const silentIframeClient = this.createSilentIframeClient( validRequest.correlationId ); result = silentIframeClient.acquireToken(validRequest); } return result .then((response) => { this.eventHandler.emitEvent( EventType.SSO_SILENT_SUCCESS, InteractionType.Silent, response ); this.ssoSilentMeasurement?.end({ success: true, isNativeBroker: response.fromNativeBroker, accessTokenSize: response.accessToken.length, idTokenSize: response.idToken.length, accountType: getAccountType(response.account), }); return response; }) .catch((e: Error) => { this.eventHandler.emitEvent( EventType.SSO_SILENT_FAILURE, InteractionType.Silent, null, e ); this.ssoSilentMeasurement?.end( { success: false, }, e ); throw e; }) .finally(() => { document.removeEventListener( "visibilitychange", this.trackPageVisibilityWithMeasurement ); }); } /** * This function redeems an authorization code (passed as code) from the eSTS token endpoint. * This authorization code should be acquired server-side using a confidential client to acquire a spa_code. * This API is not indended for normal authorization code acquisition and redemption. * * Redemption of this authorization code will not require PKCE, as it was acquired by a confidential client. * * @param request {@link AuthorizationCodeRequest} * @returns A promise that is fulfilled when this function has completed, or rejected if an error was raised. */ async acquireTokenByCode( request: AuthorizationCodeRequest ): Promise<AuthenticationResult> { const correlationId = this.getRequestCorrelationId(request); this.logger.trace("acquireTokenByCode called", correlationId); const atbcMeasurement = this.performanceClient.startMeasurement( PerformanceEvents.AcquireTokenByCode, correlationId ); preflightCheck(this.initialized, atbcMeasurement); this.eventHandler.emitEvent( EventType.ACQUIRE_TOKEN_BY_CODE_START, InteractionType.Silent, request ); atbcMeasurement.add({ scenarioId: request.scenarioId }); try { if (request.code && request.nativeAccountId) { // Throw error in case server returns both spa_code and spa_accountid in exchange for auth code. throw createBrowserAuthError( BrowserAuthErrorCodes.spaCodeAndNativeAccountIdPresent ); } else if (request.code) { const hybridAuthCode = request.code; let response = this.hybridAuthCodeResponses.get(hybridAuthCode); if (!response) { this.logger.verbose( "Initiating new acquireTokenByCode request", correlationId ); response = this.acquireTokenByCodeAsync({ ...request, correlationId, }) .then((result: AuthenticationResult) => { this.eventHandler.emitEvent( EventType.ACQUIRE_TOKEN_BY_CODE_SUCCESS, InteractionType.Silent, result ); this.hybridAuthCodeResponses.delete(hybridAuthCode); atbcMeasurement.end({ success: true, isNativeBroker: result.fromNativeBroker, accessTokenSize: result.accessToken.length, idTokenSize: result.idToken.length, accountType: getAccountType(result.account), }); return result; }) .catch((error: Error) => { this.hybridAuthCodeResponses.delete(hybridAuthCode); this.eventHandler.emitEvent( EventType.ACQUIRE_TOKEN_BY_CODE_FAILURE, InteractionType.Silent, null, error ); atbcMeasurement.end( { success: false, }, error ); throw error; }); this.hybridAuthCodeResponses.set(hybridAuthCode, response); } else { this.logger.verbose( "Existing acquireTokenByCode request found", correlationId ); atbcMeasurement.discard(); } return await response; } else if (request.nativeAccountId) { if ( this.canUsePlatformBroker(request, request.nativeAccountId) ) { const result = await this.acquireTokenNative( { ...request, correlationId, }, ApiId.acquireTokenByCode, request.nativeAccountId ).catch((e: AuthError) => { // If native token acquisition fails for availability reasons fallback to standard flow if ( e instanceof NativeAuthError && isFatalNativeAuthError(e) ) { this.nativeExtensionProvider = undefined; // If extension gets uninstalled during session prevent future requests from continuing to attempt } throw e; }); atbcMeasurement.end({ accountType: getAccountType(result.account), success: true, }); return result; } else { throw createBrowserAuthError( BrowserAuthErrorCodes.unableToAcquireTokenFromNativePlatform ); } } else { throw createBrowserAuthError( BrowserAuthErrorCodes.authCodeOrNativeAccountIdRequired ); } } catch (e) { this.eventHandler.emitEvent( EventType.ACQUIRE_TOKEN_BY_CODE_FAILURE, InteractionType.Silent, null, e as EventError ); atbcMeasurement.end( { success: false, }, e ); throw e; } } /** * Creates a SilentAuthCodeClient to redeem an authorization code. * @param request * @returns Result of the operation to redeem the authorization code */ private async acquireTokenByCodeAsync( request: AuthorizationCodeRequest ): Promise<AuthenticationResult> { this.logger.trace( "acquireTokenByCodeAsync called", request.correlationId ); this.acquireTokenByCodeAsyncMeasurement = this.performanceClient.startMeasurement( PerformanceEvents.AcquireTokenByCodeAsync, request.correlationId ); this.acquireTokenByCodeAsyncMeasurement?.increment({ visibilityChangeCount: 0, }); document.addEventListener( "visibilitychange", this.trackPageVisibilityWithMeasurement ); const silentAuthCodeClient = this.createSilentAuthCodeClient( request.correlationId ); const silentTokenResult = await silentAuthCodeClient .acquireToken(request) .then((response) => { this.acquireTokenByCodeAsyncMeasurement?.end({ success: true, fromCache: response.fromCache, isNativeBroker: response.fromNativeBroker, }); return response; }) .catch((tokenRenewalError: Error) => { this.acquireTokenByCodeAsyncMeasurement?.end( { success: false, }, tokenRenewalError ); throw tokenRenewalError; }) .finally(() => { document.removeEventListener( "visibilitychange", this.trackPageVisibilityWithMeasurement ); }); return silentTokenResult; } /** * Attempt to acquire an access token from the cache * @param silen