UNPKG

angular-auth-oidc-client

Version:
116 lines 20.7 kB
import { DOCUMENT } from '@angular/common'; import { Injectable, inject } from '@angular/core'; import { of, throwError } from 'rxjs'; import { catchError, switchMap, tap } from 'rxjs/operators'; import { AuthStateService } from '../../auth-state/auth-state.service'; import { LoggerService } from '../../logging/logger.service'; import { StoragePersistenceService } from '../../storage/storage-persistence.service'; import { ValidationResult } from '../../validation/validation-result'; import { FlowsDataService } from '../flows-data.service'; import { ResetAuthDataService } from '../reset-auth-data.service'; import { SigninKeyDataService } from '../signin-key-data.service'; import * as i0 from "@angular/core"; const JWT_KEYS = 'jwtKeys'; export class HistoryJwtKeysCallbackHandlerService { constructor() { this.loggerService = inject(LoggerService); this.authStateService = inject(AuthStateService); this.flowsDataService = inject(FlowsDataService); this.signInKeyDataService = inject(SigninKeyDataService); this.storagePersistenceService = inject(StoragePersistenceService); this.resetAuthDataService = inject(ResetAuthDataService); this.document = inject(DOCUMENT); } // STEP 3 Code Flow, STEP 2 Implicit Flow, STEP 3 Refresh Token callbackHistoryAndResetJwtKeys(callbackContext, config, allConfigs) { let toWrite = { ...callbackContext.authResult }; if (!this.responseHasIdToken(callbackContext)) { const existingIdToken = this.storagePersistenceService.getIdToken(config); toWrite = { ...toWrite, id_token: existingIdToken, }; } this.storagePersistenceService.write('authnResult', toWrite, config); if (config.allowUnsafeReuseRefreshToken && callbackContext.authResult?.refresh_token) { this.storagePersistenceService.write('reusable_refresh_token', callbackContext.authResult.refresh_token, config); } if (this.historyCleanUpTurnedOn(config) && !callbackContext.isRenewProcess) { this.resetBrowserHistory(); } else { this.loggerService.logDebug(config, 'history clean up inactive'); } if (callbackContext.authResult?.error) { const errorMessage = `AuthCallback AuthResult came with error: ${callbackContext.authResult.error}`; this.loggerService.logDebug(config, errorMessage); this.resetAuthDataService.resetAuthorizationData(config, allConfigs); this.flowsDataService.setNonce('', config); this.handleResultErrorFromCallback(callbackContext.authResult, callbackContext.isRenewProcess); return throwError(() => new Error(errorMessage)); } this.loggerService.logDebug(config, `AuthResult '${JSON.stringify(callbackContext.authResult, null, 2)}'. AuthCallback created, begin token validation`); return this.signInKeyDataService.getSigningKeys(config).pipe(tap((jwtKeys) => this.storeSigningKeys(jwtKeys, config)), catchError((err) => { // fallback: try to load jwtKeys from storage const storedJwtKeys = this.readSigningKeys(config); if (!!storedJwtKeys) { this.loggerService.logWarning(config, `Failed to retrieve signing keys, fallback to stored keys`); return of(storedJwtKeys); } return throwError(() => new Error(err)); }), switchMap((jwtKeys) => { if (jwtKeys) { callbackContext.jwtKeys = jwtKeys; return of(callbackContext); } const errorMessage = `Failed to retrieve signing key`; this.loggerService.logWarning(config, errorMessage); return throwError(() => new Error(errorMessage)); }), catchError((err) => { const errorMessage = `Failed to retrieve signing key with error: ${err}`; this.loggerService.logWarning(config, errorMessage); return throwError(() => new Error(errorMessage)); })); } responseHasIdToken(callbackContext) { return !!callbackContext?.authResult?.id_token; } handleResultErrorFromCallback(result, isRenewProcess) { let validationResult = ValidationResult.SecureTokenServerError; if (result && typeof result === 'object' && 'error' in result && result.error === 'login_required') { validationResult = ValidationResult.LoginRequired; } this.authStateService.updateAndPublishAuthState({ isAuthenticated: false, validationResult, isRenewProcess, }); } historyCleanUpTurnedOn(config) { const { historyCleanupOff } = config; return !historyCleanupOff; } resetBrowserHistory() { this.document.defaultView?.history.replaceState({}, this.document.title, this.document.defaultView.location.origin + this.document.defaultView.location.pathname); } storeSigningKeys(jwtKeys, config) { this.storagePersistenceService.write(JWT_KEYS, jwtKeys, config); } readSigningKeys(config) { return this.storagePersistenceService.read(JWT_KEYS, config); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: HistoryJwtKeysCallbackHandlerService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); } static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: HistoryJwtKeysCallbackHandlerService, providedIn: 'root' }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: HistoryJwtKeysCallbackHandlerService, decorators: [{ type: Injectable, args: [{ providedIn: 'root' }] }] }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"history-jwt-keys-callback-handler.service.js","sourceRoot":"","sources":["../../../../../../projects/angular-auth-oidc-client/src/lib/flows/callback-handling/history-jwt-keys-callback-handler.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { DOCUMENT } from '@angular/common';\r\nimport { Injectable, inject } from '@angular/core';\r\nimport { Observable, of, throwError } from 'rxjs';\r\nimport { catchError, switchMap, tap } from 'rxjs/operators';\r\nimport { AuthStateService } from '../../auth-state/auth-state.service';\r\nimport { OpenIdConfiguration } from '../../config/openid-configuration';\r\nimport { LoggerService } from '../../logging/logger.service';\r\nimport { StoragePersistenceService } from '../../storage/storage-persistence.service';\r\nimport { JwtKeys } from '../../validation/jwtkeys';\r\nimport { ValidationResult } from '../../validation/validation-result';\r\nimport { CallbackContext } from '../callback-context';\r\nimport { FlowsDataService } from '../flows-data.service';\r\nimport { ResetAuthDataService } from '../reset-auth-data.service';\r\nimport { SigninKeyDataService } from '../signin-key-data.service';\r\n\r\nconst JWT_KEYS = 'jwtKeys';\r\n\r\n@Injectable({ providedIn: 'root' })\r\nexport class HistoryJwtKeysCallbackHandlerService {\r\n  private readonly loggerService = inject(LoggerService);\r\n\r\n  private readonly authStateService = inject(AuthStateService);\r\n\r\n  private readonly flowsDataService = inject(FlowsDataService);\r\n\r\n  private readonly signInKeyDataService = inject(SigninKeyDataService);\r\n\r\n  private readonly storagePersistenceService = inject(\r\n    StoragePersistenceService\r\n  );\r\n\r\n  private readonly resetAuthDataService = inject(ResetAuthDataService);\r\n\r\n  private readonly document = inject(DOCUMENT);\r\n\r\n  // STEP 3 Code Flow, STEP 2 Implicit Flow, STEP 3 Refresh Token\r\n  callbackHistoryAndResetJwtKeys(\r\n    callbackContext: CallbackContext,\r\n    config: OpenIdConfiguration,\r\n    allConfigs: OpenIdConfiguration[]\r\n  ): Observable<CallbackContext> {\r\n    let toWrite = { ...callbackContext.authResult };\r\n\r\n    if (!this.responseHasIdToken(callbackContext)) {\r\n      const existingIdToken = this.storagePersistenceService.getIdToken(config);\r\n\r\n      toWrite = {\r\n        ...toWrite,\r\n        id_token: existingIdToken,\r\n      };\r\n    }\r\n\r\n    this.storagePersistenceService.write('authnResult', toWrite, config);\r\n\r\n    if (\r\n      config.allowUnsafeReuseRefreshToken &&\r\n      callbackContext.authResult?.refresh_token\r\n    ) {\r\n      this.storagePersistenceService.write(\r\n        'reusable_refresh_token',\r\n        callbackContext.authResult.refresh_token,\r\n        config\r\n      );\r\n    }\r\n\r\n    if (\r\n      this.historyCleanUpTurnedOn(config) &&\r\n      !callbackContext.isRenewProcess\r\n    ) {\r\n      this.resetBrowserHistory();\r\n    } else {\r\n      this.loggerService.logDebug(config, 'history clean up inactive');\r\n    }\r\n\r\n    if (callbackContext.authResult?.error) {\r\n      const errorMessage = `AuthCallback AuthResult came with error: ${callbackContext.authResult.error}`;\r\n\r\n      this.loggerService.logDebug(config, errorMessage);\r\n      this.resetAuthDataService.resetAuthorizationData(config, allConfigs);\r\n      this.flowsDataService.setNonce('', config);\r\n      this.handleResultErrorFromCallback(\r\n        callbackContext.authResult,\r\n        callbackContext.isRenewProcess\r\n      );\r\n\r\n      return throwError(() => new Error(errorMessage));\r\n    }\r\n\r\n    this.loggerService.logDebug(\r\n      config,\r\n      `AuthResult '${JSON.stringify(callbackContext.authResult, null, 2)}'.\r\n      AuthCallback created, begin token validation`\r\n    );\r\n\r\n    return this.signInKeyDataService.getSigningKeys(config).pipe(\r\n      tap((jwtKeys: JwtKeys) => this.storeSigningKeys(jwtKeys, config)),\r\n      catchError((err) => {\r\n        // fallback: try to load jwtKeys from storage\r\n        const storedJwtKeys = this.readSigningKeys(config);\r\n\r\n        if (!!storedJwtKeys) {\r\n          this.loggerService.logWarning(\r\n            config,\r\n            `Failed to retrieve signing keys, fallback to stored keys`\r\n          );\r\n\r\n          return of(storedJwtKeys);\r\n        }\r\n\r\n        return throwError(() => new Error(err));\r\n      }),\r\n      switchMap((jwtKeys) => {\r\n        if (jwtKeys) {\r\n          callbackContext.jwtKeys = jwtKeys;\r\n\r\n          return of(callbackContext);\r\n        }\r\n\r\n        const errorMessage = `Failed to retrieve signing key`;\r\n\r\n        this.loggerService.logWarning(config, errorMessage);\r\n\r\n        return throwError(() => new Error(errorMessage));\r\n      }),\r\n      catchError((err) => {\r\n        const errorMessage = `Failed to retrieve signing key with error: ${err}`;\r\n\r\n        this.loggerService.logWarning(config, errorMessage);\r\n\r\n        return throwError(() => new Error(errorMessage));\r\n      })\r\n    );\r\n  }\r\n\r\n  private responseHasIdToken(callbackContext: CallbackContext): boolean {\r\n    return !!callbackContext?.authResult?.id_token;\r\n  }\r\n\r\n  private handleResultErrorFromCallback(\r\n    result: unknown,\r\n    isRenewProcess: boolean\r\n  ): void {\r\n    let validationResult = ValidationResult.SecureTokenServerError;\r\n\r\n    if (\r\n      result &&\r\n      typeof result === 'object' &&\r\n      'error' in result &&\r\n      (result.error as string) === 'login_required'\r\n    ) {\r\n      validationResult = ValidationResult.LoginRequired;\r\n    }\r\n\r\n    this.authStateService.updateAndPublishAuthState({\r\n      isAuthenticated: false,\r\n      validationResult,\r\n      isRenewProcess,\r\n    });\r\n  }\r\n\r\n  private historyCleanUpTurnedOn(config: OpenIdConfiguration): boolean {\r\n    const { historyCleanupOff } = config;\r\n\r\n    return !historyCleanupOff;\r\n  }\r\n\r\n  private resetBrowserHistory(): void {\r\n    this.document.defaultView?.history.replaceState(\r\n      {},\r\n      this.document.title,\r\n      this.document.defaultView.location.origin +\r\n        this.document.defaultView.location.pathname\r\n    );\r\n  }\r\n\r\n  private storeSigningKeys(\r\n    jwtKeys: JwtKeys,\r\n    config: OpenIdConfiguration\r\n  ): void {\r\n    this.storagePersistenceService.write(JWT_KEYS, jwtKeys, config);\r\n  }\r\n\r\n  private readSigningKeys(config: OpenIdConfiguration): any {\r\n    return this.storagePersistenceService.read(JWT_KEYS, config);\r\n  }\r\n}\r\n"]}