angular-auth-oidc-client
Version:
Angular Lib for OpenID Connect & OAuth2
119 lines • 18.7 kB
JavaScript
import { HttpParams } from '@angular/common/http';
import { Injectable, inject } from '@angular/core';
import { Subject, throwError } from 'rxjs';
import { catchError } from 'rxjs/operators';
import { AuthStateService } from '../auth-state/auth-state.service';
import { ImplicitFlowCallbackService } from '../callback/implicit-flow-callback.service';
import { IntervalService } from '../callback/interval.service';
import { FlowsDataService } from '../flows/flows-data.service';
import { FlowsService } from '../flows/flows.service';
import { ResetAuthDataService } from '../flows/reset-auth-data.service';
import { LoggerService } from '../logging/logger.service';
import { FlowHelper } from '../utils/flowHelper/flow-helper.service';
import { ValidationResult } from '../validation/validation-result';
import { IFrameService } from './existing-iframe.service';
import * as i0 from "@angular/core";
const IFRAME_FOR_SILENT_RENEW_IDENTIFIER = 'myiFrameForSilentRenew';
export class SilentRenewService {
constructor() {
this.refreshSessionWithIFrameCompletedInternal$ = new Subject();
this.loggerService = inject(LoggerService);
this.iFrameService = inject(IFrameService);
this.flowsService = inject(FlowsService);
this.resetAuthDataService = inject(ResetAuthDataService);
this.flowsDataService = inject(FlowsDataService);
this.authStateService = inject(AuthStateService);
this.flowHelper = inject(FlowHelper);
this.implicitFlowCallbackService = inject(ImplicitFlowCallbackService);
this.intervalService = inject(IntervalService);
}
get refreshSessionWithIFrameCompleted$() {
return this.refreshSessionWithIFrameCompletedInternal$.asObservable();
}
getOrCreateIframe(config) {
const existingIframe = this.getExistingIframe();
if (!existingIframe) {
return this.iFrameService.addIFrameToWindowBody(IFRAME_FOR_SILENT_RENEW_IDENTIFIER, config);
}
return existingIframe;
}
isSilentRenewConfigured(configuration) {
const { useRefreshToken, silentRenew } = configuration;
return !useRefreshToken && Boolean(silentRenew);
}
codeFlowCallbackSilentRenewIframe(urlParts, config, allConfigs) {
const params = new HttpParams({
fromString: urlParts[1],
});
const errorParam = params.get('error');
if (errorParam) {
this.authStateService.updateAndPublishAuthState({
isAuthenticated: false,
validationResult: ValidationResult.LoginRequired,
isRenewProcess: true,
});
this.resetAuthDataService.resetAuthorizationData(config, allConfigs);
this.flowsDataService.setNonce('', config);
this.intervalService.stopPeriodicTokenCheck();
return throwError(() => new Error(errorParam));
}
const code = params.get('code') ?? '';
const state = params.get('state') ?? '';
const sessionState = params.get('session_state');
const callbackContext = {
code,
refreshToken: '',
state,
sessionState,
authResult: null,
isRenewProcess: true,
jwtKeys: null,
validationResult: null,
existingIdToken: null,
};
return this.flowsService
.processSilentRenewCodeFlowCallback(callbackContext, config, allConfigs)
.pipe(catchError((error) => {
this.intervalService.stopPeriodicTokenCheck();
this.resetAuthDataService.resetAuthorizationData(config, allConfigs);
return throwError(() => new Error(error));
}));
}
silentRenewEventHandler(e, config, allConfigs) {
this.loggerService.logDebug(config, 'silentRenewEventHandler');
if (!e.detail) {
return;
}
let callback$;
const isCodeFlow = this.flowHelper.isCurrentFlowCodeFlow(config);
if (isCodeFlow) {
const urlParts = e.detail.toString().split('?');
callback$ = this.codeFlowCallbackSilentRenewIframe(urlParts, config, allConfigs);
}
else {
callback$ =
this.implicitFlowCallbackService.authenticatedImplicitFlowCallback(config, allConfigs, e.detail);
}
callback$.subscribe({
next: (callbackContext) => {
this.refreshSessionWithIFrameCompletedInternal$.next(callbackContext);
this.flowsDataService.resetSilentRenewRunning(config);
},
error: (err) => {
this.loggerService.logError(config, 'Error: ' + err);
this.refreshSessionWithIFrameCompletedInternal$.next(null);
this.flowsDataService.resetSilentRenewRunning(config);
},
});
}
getExistingIframe() {
return this.iFrameService.getExistingIFrame(IFRAME_FOR_SILENT_RENEW_IDENTIFIER);
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: SilentRenewService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: SilentRenewService, providedIn: 'root' }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: SilentRenewService, decorators: [{
type: Injectable,
args: [{ providedIn: 'root' }]
}] });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"silent-renew.service.js","sourceRoot":"","sources":["../../../../../projects/angular-auth-oidc-client/src/lib/iframe/silent-renew.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,sBAAsB,CAAC;AAClD,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACnD,OAAO,EAAc,OAAO,EAAE,UAAU,EAAE,MAAM,MAAM,CAAC;AACvD,OAAO,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAC5C,OAAO,EAAE,gBAAgB,EAAE,MAAM,kCAAkC,CAAC;AACpE,OAAO,EAAE,2BAA2B,EAAE,MAAM,4CAA4C,CAAC;AACzF,OAAO,EAAE,eAAe,EAAE,MAAM,8BAA8B,CAAC;AAG/D,OAAO,EAAE,gBAAgB,EAAE,MAAM,6BAA6B,CAAC;AAC/D,OAAO,EAAE,YAAY,EAAE,MAAM,wBAAwB,CAAC;AACtD,OAAO,EAAE,oBAAoB,EAAE,MAAM,kCAAkC,CAAC;AACxE,OAAO,EAAE,aAAa,EAAE,MAAM,2BAA2B,CAAC;AAC1D,OAAO,EAAE,UAAU,EAAE,MAAM,yCAAyC,CAAC;AACrE,OAAO,EAAE,gBAAgB,EAAE,MAAM,iCAAiC,CAAC;AACnE,OAAO,EAAE,aAAa,EAAE,MAAM,2BAA2B,CAAC;;AAE1D,MAAM,kCAAkC,GAAG,wBAAwB,CAAC;AAGpE,MAAM,OAAO,kBAAkB;IAD/B;QAEmB,+CAA0C,GACzD,IAAI,OAAO,EAA0B,CAAC;QAMvB,kBAAa,GAAG,MAAM,CAAC,aAAa,CAAC,CAAC;QAEtC,kBAAa,GAAG,MAAM,CAAC,aAAa,CAAC,CAAC;QAEtC,iBAAY,GAAG,MAAM,CAAC,YAAY,CAAC,CAAC;QAEpC,yBAAoB,GAAG,MAAM,CAAC,oBAAoB,CAAC,CAAC;QAEpD,qBAAgB,GAAG,MAAM,CAAC,gBAAgB,CAAC,CAAC;QAE5C,qBAAgB,GAAG,MAAM,CAAC,gBAAgB,CAAC,CAAC;QAE5C,eAAU,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;QAEhC,gCAA2B,GAAG,MAAM,CACnD,2BAA2B,CAC5B,CAAC;QAEe,oBAAe,GAAG,MAAM,CAAC,eAAe,CAAC,CAAC;KAyH5D;IA/IC,IAAI,kCAAkC;QACpC,OAAO,IAAI,CAAC,0CAA0C,CAAC,YAAY,EAAE,CAAC;IACxE,CAAC;IAsBD,iBAAiB,CAAC,MAA2B;QAC3C,MAAM,cAAc,GAAG,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAEhD,IAAI,CAAC,cAAc,EAAE,CAAC;YACpB,OAAO,IAAI,CAAC,aAAa,CAAC,qBAAqB,CAC7C,kCAAkC,EAClC,MAAM,CACP,CAAC;QACJ,CAAC;QAED,OAAO,cAAc,CAAC;IACxB,CAAC;IAED,uBAAuB,CAAC,aAAkC;QACxD,MAAM,EAAE,eAAe,EAAE,WAAW,EAAE,GAAG,aAAa,CAAC;QAEvD,OAAO,CAAC,eAAe,IAAI,OAAO,CAAC,WAAW,CAAC,CAAC;IAClD,CAAC;IAED,iCAAiC,CAC/B,QAAkB,EAClB,MAA2B,EAC3B,UAAiC;QAEjC,MAAM,MAAM,GAAG,IAAI,UAAU,CAAC;YAC5B,UAAU,EAAE,QAAQ,CAAC,CAAC,CAAC;SACxB,CAAC,CAAC;QAEH,MAAM,UAAU,GAAG,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;QAEvC,IAAI,UAAU,EAAE,CAAC;YACf,IAAI,CAAC,gBAAgB,CAAC,yBAAyB,CAAC;gBAC9C,eAAe,EAAE,KAAK;gBACtB,gBAAgB,EAAE,gBAAgB,CAAC,aAAa;gBAChD,cAAc,EAAE,IAAI;aACrB,CAAC,CAAC;YACH,IAAI,CAAC,oBAAoB,CAAC,sBAAsB,CAAC,MAAM,EAAE,UAAU,CAAC,CAAC;YACrE,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC;YAC3C,IAAI,CAAC,eAAe,CAAC,sBAAsB,EAAE,CAAC;YAE9C,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,KAAK,CAAC,UAAU,CAAC,CAAC,CAAC;QACjD,CAAC;QAED,MAAM,IAAI,GAAG,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;QACtC,MAAM,KAAK,GAAG,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;QACxC,MAAM,YAAY,GAAG,MAAM,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;QAEjD,MAAM,eAAe,GAAoB;YACvC,IAAI;YACJ,YAAY,EAAE,EAAE;YAChB,KAAK;YACL,YAAY;YACZ,UAAU,EAAE,IAAI;YAChB,cAAc,EAAE,IAAI;YACpB,OAAO,EAAE,IAAI;YACb,gBAAgB,EAAE,IAAI;YACtB,eAAe,EAAE,IAAI;SACtB,CAAC;QAEF,OAAO,IAAI,CAAC,YAAY;aACrB,kCAAkC,CAAC,eAAe,EAAE,MAAM,EAAE,UAAU,CAAC;aACvE,IAAI,CACH,UAAU,CAAC,CAAC,KAAK,EAAE,EAAE;YACnB,IAAI,CAAC,eAAe,CAAC,sBAAsB,EAAE,CAAC;YAC9C,IAAI,CAAC,oBAAoB,CAAC,sBAAsB,CAAC,MAAM,EAAE,UAAU,CAAC,CAAC;YAErE,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC;QAC5C,CAAC,CAAC,CACH,CAAC;IACN,CAAC;IAED,uBAAuB,CACrB,CAAc,EACd,MAA2B,EAC3B,UAAiC;QAEjC,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,MAAM,EAAE,yBAAyB,CAAC,CAAC;QAC/D,IAAI,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC;YACd,OAAO;QACT,CAAC;QAED,IAAI,SAAsC,CAAC;QAC3C,MAAM,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC,qBAAqB,CAAC,MAAM,CAAC,CAAC;QAEjE,IAAI,UAAU,EAAE,CAAC;YACf,MAAM,QAAQ,GAAG,CAAC,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YAEhD,SAAS,GAAG,IAAI,CAAC,iCAAiC,CAChD,QAAQ,EACR,MAAM,EACN,UAAU,CACX,CAAC;QACJ,CAAC;aAAM,CAAC;YACN,SAAS;gBACP,IAAI,CAAC,2BAA2B,CAAC,iCAAiC,CAChE,MAAM,EACN,UAAU,EACV,CAAC,CAAC,MAAM,CACT,CAAC;QACN,CAAC;QAED,SAAS,CAAC,SAAS,CAAC;YAClB,IAAI,EAAE,CAAC,eAAe,EAAE,EAAE;gBACxB,IAAI,CAAC,0CAA0C,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;gBACtE,IAAI,CAAC,gBAAgB,CAAC,uBAAuB,CAAC,MAAM,CAAC,CAAC;YACxD,CAAC;YACD,KAAK,EAAE,CAAC,GAAY,EAAE,EAAE;gBACtB,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,MAAM,EAAE,SAAS,GAAG,GAAG,CAAC,CAAC;gBACrD,IAAI,CAAC,0CAA0C,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBAC3D,IAAI,CAAC,gBAAgB,CAAC,uBAAuB,CAAC,MAAM,CAAC,CAAC;YACxD,CAAC;SACF,CAAC,CAAC;IACL,CAAC;IAEO,iBAAiB;QACvB,OAAO,IAAI,CAAC,aAAa,CAAC,iBAAiB,CACzC,kCAAkC,CACnC,CAAC;IACJ,CAAC;8GAlJU,kBAAkB;kHAAlB,kBAAkB,cADL,MAAM;;2FACnB,kBAAkB;kBAD9B,UAAU;mBAAC,EAAE,UAAU,EAAE,MAAM,EAAE","sourcesContent":["import { HttpParams } from '@angular/common/http';\nimport { Injectable, inject } from '@angular/core';\nimport { Observable, Subject, throwError } from 'rxjs';\nimport { catchError } from 'rxjs/operators';\nimport { AuthStateService } from '../auth-state/auth-state.service';\nimport { ImplicitFlowCallbackService } from '../callback/implicit-flow-callback.service';\nimport { IntervalService } from '../callback/interval.service';\nimport { OpenIdConfiguration } from '../config/openid-configuration';\nimport { CallbackContext } from '../flows/callback-context';\nimport { FlowsDataService } from '../flows/flows-data.service';\nimport { FlowsService } from '../flows/flows.service';\nimport { ResetAuthDataService } from '../flows/reset-auth-data.service';\nimport { LoggerService } from '../logging/logger.service';\nimport { FlowHelper } from '../utils/flowHelper/flow-helper.service';\nimport { ValidationResult } from '../validation/validation-result';\nimport { IFrameService } from './existing-iframe.service';\n\nconst IFRAME_FOR_SILENT_RENEW_IDENTIFIER = 'myiFrameForSilentRenew';\n\n@Injectable({ providedIn: 'root' })\nexport class SilentRenewService {\n  private readonly refreshSessionWithIFrameCompletedInternal$ =\n    new Subject<CallbackContext | null>();\n\n  get refreshSessionWithIFrameCompleted$(): Observable<CallbackContext | null> {\n    return this.refreshSessionWithIFrameCompletedInternal$.asObservable();\n  }\n\n  private readonly loggerService = inject(LoggerService);\n\n  private readonly iFrameService = inject(IFrameService);\n\n  private readonly flowsService = inject(FlowsService);\n\n  private readonly resetAuthDataService = inject(ResetAuthDataService);\n\n  private readonly flowsDataService = inject(FlowsDataService);\n\n  private readonly authStateService = inject(AuthStateService);\n\n  private readonly flowHelper = inject(FlowHelper);\n\n  private readonly implicitFlowCallbackService = inject(\n    ImplicitFlowCallbackService\n  );\n\n  private readonly intervalService = inject(IntervalService);\n\n  getOrCreateIframe(config: OpenIdConfiguration): HTMLIFrameElement {\n    const existingIframe = this.getExistingIframe();\n\n    if (!existingIframe) {\n      return this.iFrameService.addIFrameToWindowBody(\n        IFRAME_FOR_SILENT_RENEW_IDENTIFIER,\n        config\n      );\n    }\n\n    return existingIframe;\n  }\n\n  isSilentRenewConfigured(configuration: OpenIdConfiguration): boolean {\n    const { useRefreshToken, silentRenew } = configuration;\n\n    return !useRefreshToken && Boolean(silentRenew);\n  }\n\n  codeFlowCallbackSilentRenewIframe(\n    urlParts: string[],\n    config: OpenIdConfiguration,\n    allConfigs: OpenIdConfiguration[]\n  ): Observable<CallbackContext> {\n    const params = new HttpParams({\n      fromString: urlParts[1],\n    });\n\n    const errorParam = params.get('error');\n\n    if (errorParam) {\n      this.authStateService.updateAndPublishAuthState({\n        isAuthenticated: false,\n        validationResult: ValidationResult.LoginRequired,\n        isRenewProcess: true,\n      });\n      this.resetAuthDataService.resetAuthorizationData(config, allConfigs);\n      this.flowsDataService.setNonce('', config);\n      this.intervalService.stopPeriodicTokenCheck();\n\n      return throwError(() => new Error(errorParam));\n    }\n\n    const code = params.get('code') ?? '';\n    const state = params.get('state') ?? '';\n    const sessionState = params.get('session_state');\n\n    const callbackContext: CallbackContext = {\n      code,\n      refreshToken: '',\n      state,\n      sessionState,\n      authResult: null,\n      isRenewProcess: true,\n      jwtKeys: null,\n      validationResult: null,\n      existingIdToken: null,\n    };\n\n    return this.flowsService\n      .processSilentRenewCodeFlowCallback(callbackContext, config, allConfigs)\n      .pipe(\n        catchError((error) => {\n          this.intervalService.stopPeriodicTokenCheck();\n          this.resetAuthDataService.resetAuthorizationData(config, allConfigs);\n\n          return throwError(() => new Error(error));\n        })\n      );\n  }\n\n  silentRenewEventHandler(\n    e: CustomEvent,\n    config: OpenIdConfiguration,\n    allConfigs: OpenIdConfiguration[]\n  ): void {\n    this.loggerService.logDebug(config, 'silentRenewEventHandler');\n    if (!e.detail) {\n      return;\n    }\n\n    let callback$: Observable<CallbackContext>;\n    const isCodeFlow = this.flowHelper.isCurrentFlowCodeFlow(config);\n\n    if (isCodeFlow) {\n      const urlParts = e.detail.toString().split('?');\n\n      callback$ = this.codeFlowCallbackSilentRenewIframe(\n        urlParts,\n        config,\n        allConfigs\n      );\n    } else {\n      callback$ =\n        this.implicitFlowCallbackService.authenticatedImplicitFlowCallback(\n          config,\n          allConfigs,\n          e.detail\n        );\n    }\n\n    callback$.subscribe({\n      next: (callbackContext) => {\n        this.refreshSessionWithIFrameCompletedInternal$.next(callbackContext);\n        this.flowsDataService.resetSilentRenewRunning(config);\n      },\n      error: (err: unknown) => {\n        this.loggerService.logError(config, 'Error: ' + err);\n        this.refreshSessionWithIFrameCompletedInternal$.next(null);\n        this.flowsDataService.resetSilentRenewRunning(config);\n      },\n    });\n  }\n\n  private getExistingIframe(): HTMLIFrameElement | null {\n    return this.iFrameService.getExistingIFrame(\n      IFRAME_FOR_SILENT_RENEW_IDENTIFIER\n    );\n  }\n}\n"]}