UNPKG

angular-auth-oidc-client

Version:
185 lines 33.2 kB
import { DOCUMENT } from '@angular/common'; import { Injectable, NgZone, inject } from '@angular/core'; import { BehaviorSubject, Observable, of } from 'rxjs'; import { take } from 'rxjs/operators'; import { LoggerService } from '../logging/logger.service'; import { EventTypes } from '../public-events/event-types'; import { PublicEventsService } from '../public-events/public-events.service'; import { StoragePersistenceService } from '../storage/storage-persistence.service'; import { IFrameService } from './existing-iframe.service'; import * as i0 from "@angular/core"; const IFRAME_FOR_CHECK_SESSION_IDENTIFIER = 'myiFrameForCheckSession'; // http://openid.net/specs/openid-connect-session-1_0-ID4.html export class CheckSessionService { constructor() { this.loggerService = inject(LoggerService); this.storagePersistenceService = inject(StoragePersistenceService); this.iFrameService = inject(IFrameService); this.eventService = inject(PublicEventsService); this.zone = inject(NgZone); this.document = inject(DOCUMENT); this.checkSessionReceived = false; this.scheduledHeartBeatRunning = null; this.lastIFrameRefresh = 0; this.outstandingMessages = 0; this.heartBeatInterval = 3000; this.iframeRefreshInterval = 60000; this.checkSessionChangedInternal$ = new BehaviorSubject(false); } get checkSessionChanged$() { return this.checkSessionChangedInternal$.asObservable(); } ngOnDestroy() { this.stop(); const windowAsDefaultView = this.document.defaultView; if (windowAsDefaultView && this.iframeMessageEventListener) { windowAsDefaultView.removeEventListener('message', this.iframeMessageEventListener, false); } } isCheckSessionConfigured(configuration) { const { startCheckSession } = configuration; return Boolean(startCheckSession); } start(configuration) { if (!!this.scheduledHeartBeatRunning) { return; } const { clientId } = configuration; this.pollServerSession(clientId, configuration); } stop() { if (!this.scheduledHeartBeatRunning) { return; } this.clearScheduledHeartBeat(); this.checkSessionReceived = false; } serverStateChanged(configuration) { const { startCheckSession } = configuration; return Boolean(startCheckSession) && this.checkSessionReceived; } getExistingIframe() { return this.iFrameService.getExistingIFrame(IFRAME_FOR_CHECK_SESSION_IDENTIFIER); } init(configuration) { if (this.lastIFrameRefresh + this.iframeRefreshInterval > Date.now()) { return of(undefined); } const authWellKnownEndPoints = this.storagePersistenceService.read('authWellKnownEndPoints', configuration); if (!authWellKnownEndPoints) { this.loggerService.logWarning(configuration, 'CheckSession - init check session: authWellKnownEndpoints is undefined. Returning.'); return of(); } const existingIframe = this.getOrCreateIframe(configuration); // https://www.w3.org/TR/2000/REC-DOM-Level-2-Events-20001113/events.html#Events-EventTarget-addEventListener // If multiple identical EventListeners are registered on the same EventTarget with the same parameters the duplicate instances are discarded. They do not cause the EventListener to be called twice and since they are discarded they do not need to be removed with the removeEventListener method. // this is done even if iframe exists for HMR to work, since iframe exists on service init this.bindMessageEventToIframe(configuration); const checkSessionIframe = authWellKnownEndPoints.checkSessionIframe; const contentWindow = existingIframe.contentWindow; if (!checkSessionIframe) { this.loggerService.logWarning(configuration, 'CheckSession - init check session: checkSessionIframe is not configured to run'); } if (!contentWindow) { this.loggerService.logWarning(configuration, 'CheckSession - init check session: IFrame contentWindow does not exist'); } else { contentWindow.location.replace(checkSessionIframe); } return new Observable((observer) => { existingIframe.onload = () => { this.lastIFrameRefresh = Date.now(); observer.next(); observer.complete(); }; }); } pollServerSession(clientId, configuration) { this.outstandingMessages = 0; const pollServerSessionRecur = () => { this.init(configuration) .pipe(take(1)) .subscribe(() => { const existingIframe = this.getExistingIframe(); if (existingIframe && clientId) { this.loggerService.logDebug(configuration, `CheckSession - clientId : '${clientId}' - existingIframe: '${existingIframe}'`); const sessionState = this.storagePersistenceService.read('session_state', configuration); const authWellKnownEndPoints = this.storagePersistenceService.read('authWellKnownEndPoints', configuration); const contentWindow = existingIframe.contentWindow; if (sessionState && authWellKnownEndPoints?.checkSessionIframe && contentWindow) { const iframeOrigin = new URL(authWellKnownEndPoints.checkSessionIframe)?.origin; this.outstandingMessages++; contentWindow.postMessage(clientId + ' ' + sessionState, iframeOrigin); } else { this.loggerService.logDebug(configuration, `CheckSession - session_state is '${sessionState}' - AuthWellKnownEndPoints is '${JSON.stringify(authWellKnownEndPoints, null, 2)}'`); this.checkSessionChangedInternal$.next(true); } } else { this.loggerService.logWarning(configuration, `CheckSession - OidcSecurityCheckSession pollServerSession checkSession IFrame does not exist: clientId : '${clientId}' - existingIframe: '${existingIframe}'`); } // after sending three messages with no response, fail. if (this.outstandingMessages > 3) { this.loggerService.logError(configuration, `CheckSession - OidcSecurityCheckSession not receiving check session response messages. Outstanding messages: '${this.outstandingMessages}'. Server unreachable?`); } this.zone.runOutsideAngular(() => { this.scheduledHeartBeatRunning = this.document?.defaultView?.setTimeout(() => this.zone.run(pollServerSessionRecur), this.heartBeatInterval) ?? null; }); }); }; pollServerSessionRecur(); } clearScheduledHeartBeat() { if (this.scheduledHeartBeatRunning !== null) { clearTimeout(this.scheduledHeartBeatRunning); this.scheduledHeartBeatRunning = null; } } messageHandler(configuration, e) { const existingIFrame = this.getExistingIframe(); const authWellKnownEndPoints = this.storagePersistenceService.read('authWellKnownEndPoints', configuration); const startsWith = !!authWellKnownEndPoints?.checkSessionIframe?.startsWith(e.origin); this.outstandingMessages = 0; if (existingIFrame && startsWith && e.source === existingIFrame.contentWindow) { if (e.data === 'error') { this.loggerService.logWarning(configuration, 'CheckSession - error from check session messageHandler'); } else if (e.data === 'changed') { this.loggerService.logDebug(configuration, `CheckSession - ${e} from check session messageHandler`); this.checkSessionReceived = true; this.eventService.fireEvent(EventTypes.CheckSessionReceived, e.data); this.checkSessionChangedInternal$.next(true); } else { this.eventService.fireEvent(EventTypes.CheckSessionReceived, e.data); this.loggerService.logDebug(configuration, `CheckSession - ${e.data} from check session messageHandler`); } } } bindMessageEventToIframe(configuration) { this.iframeMessageEventListener = this.messageHandler.bind(this, configuration); const defaultView = this.document.defaultView; if (defaultView) { defaultView.addEventListener('message', this.iframeMessageEventListener, false); } } getOrCreateIframe(configuration) { return (this.getExistingIframe() || this.iFrameService.addIFrameToWindowBody(IFRAME_FOR_CHECK_SESSION_IDENTIFIER, configuration)); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: CheckSessionService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); } static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: CheckSessionService, providedIn: 'root' }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: CheckSessionService, decorators: [{ type: Injectable, args: [{ providedIn: 'root' }] }] }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"check-session.service.js","sourceRoot":"","sources":["../../../../../projects/angular-auth-oidc-client/src/lib/iframe/check-session.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { DOCUMENT } from '@angular/common';\r\nimport { Injectable, NgZone, OnDestroy, inject } from '@angular/core';\r\nimport { BehaviorSubject, Observable, of } from 'rxjs';\r\nimport { take } from 'rxjs/operators';\r\nimport { OpenIdConfiguration } from '../config/openid-configuration';\r\nimport { LoggerService } from '../logging/logger.service';\r\nimport { EventTypes } from '../public-events/event-types';\r\nimport { PublicEventsService } from '../public-events/public-events.service';\r\nimport { StoragePersistenceService } from '../storage/storage-persistence.service';\r\nimport { IFrameService } from './existing-iframe.service';\r\n\r\nconst IFRAME_FOR_CHECK_SESSION_IDENTIFIER = 'myiFrameForCheckSession';\r\n\r\n// http://openid.net/specs/openid-connect-session-1_0-ID4.html\r\n\r\n@Injectable({ providedIn: 'root' })\r\nexport class CheckSessionService implements OnDestroy {\r\n  private readonly loggerService = inject(LoggerService);\r\n\r\n  private readonly storagePersistenceService = inject(\r\n    StoragePersistenceService\r\n  );\r\n\r\n  private readonly iFrameService = inject(IFrameService);\r\n\r\n  private readonly eventService = inject(PublicEventsService);\r\n\r\n  private readonly zone = inject(NgZone);\r\n\r\n  private readonly document = inject(DOCUMENT);\r\n\r\n  private checkSessionReceived = false;\r\n\r\n  private scheduledHeartBeatRunning: number | null = null;\r\n\r\n  private lastIFrameRefresh = 0;\r\n\r\n  private outstandingMessages = 0;\r\n\r\n  private readonly heartBeatInterval = 3000;\r\n\r\n  private readonly iframeRefreshInterval = 60000;\r\n\r\n  private readonly checkSessionChangedInternal$ = new BehaviorSubject<boolean>(\r\n    false\r\n  );\r\n\r\n  private iframeMessageEventListener?: (\r\n    this: Window,\r\n    ev: MessageEvent<any>\r\n  ) => any;\r\n\r\n  get checkSessionChanged$(): Observable<boolean> {\r\n    return this.checkSessionChangedInternal$.asObservable();\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.stop();\r\n    const windowAsDefaultView = this.document.defaultView;\r\n\r\n    if (windowAsDefaultView && this.iframeMessageEventListener) {\r\n      windowAsDefaultView.removeEventListener(\r\n        'message',\r\n        this.iframeMessageEventListener,\r\n        false\r\n      );\r\n    }\r\n  }\r\n\r\n  isCheckSessionConfigured(configuration: OpenIdConfiguration): boolean {\r\n    const { startCheckSession } = configuration;\r\n\r\n    return Boolean(startCheckSession);\r\n  }\r\n\r\n  start(configuration: OpenIdConfiguration): void {\r\n    if (!!this.scheduledHeartBeatRunning) {\r\n      return;\r\n    }\r\n\r\n    const { clientId } = configuration;\r\n\r\n    this.pollServerSession(clientId, configuration);\r\n  }\r\n\r\n  stop(): void {\r\n    if (!this.scheduledHeartBeatRunning) {\r\n      return;\r\n    }\r\n\r\n    this.clearScheduledHeartBeat();\r\n    this.checkSessionReceived = false;\r\n  }\r\n\r\n  serverStateChanged(configuration: OpenIdConfiguration): boolean {\r\n    const { startCheckSession } = configuration;\r\n\r\n    return Boolean(startCheckSession) && this.checkSessionReceived;\r\n  }\r\n\r\n  getExistingIframe(): HTMLIFrameElement | null {\r\n    return this.iFrameService.getExistingIFrame(\r\n      IFRAME_FOR_CHECK_SESSION_IDENTIFIER\r\n    );\r\n  }\r\n\r\n  private init(configuration: OpenIdConfiguration): Observable<any> {\r\n    if (this.lastIFrameRefresh + this.iframeRefreshInterval > Date.now()) {\r\n      return of(undefined);\r\n    }\r\n\r\n    const authWellKnownEndPoints = this.storagePersistenceService.read(\r\n      'authWellKnownEndPoints',\r\n      configuration\r\n    );\r\n\r\n    if (!authWellKnownEndPoints) {\r\n      this.loggerService.logWarning(\r\n        configuration,\r\n        'CheckSession - init check session: authWellKnownEndpoints is undefined. Returning.'\r\n      );\r\n\r\n      return of();\r\n    }\r\n\r\n    const existingIframe = this.getOrCreateIframe(configuration);\r\n\r\n    // https://www.w3.org/TR/2000/REC-DOM-Level-2-Events-20001113/events.html#Events-EventTarget-addEventListener\r\n    // If multiple identical EventListeners are registered on the same EventTarget with the same parameters the duplicate instances are discarded. They do not cause the EventListener to be called twice and since they are discarded they do not need to be removed with the removeEventListener method.\r\n    // this is done even if iframe exists for HMR to work, since iframe exists on service init\r\n    this.bindMessageEventToIframe(configuration);\r\n    const checkSessionIframe = authWellKnownEndPoints.checkSessionIframe;\r\n    const contentWindow = existingIframe.contentWindow;\r\n\r\n    if (!checkSessionIframe) {\r\n      this.loggerService.logWarning(\r\n        configuration,\r\n        'CheckSession - init check session: checkSessionIframe is not configured to run'\r\n      );\r\n    }\r\n\r\n    if (!contentWindow) {\r\n      this.loggerService.logWarning(\r\n        configuration,\r\n        'CheckSession - init check session: IFrame contentWindow does not exist'\r\n      );\r\n    } else {\r\n      contentWindow.location.replace(checkSessionIframe);\r\n    }\r\n\r\n    return new Observable((observer) => {\r\n      existingIframe.onload = (): void => {\r\n        this.lastIFrameRefresh = Date.now();\r\n        observer.next();\r\n        observer.complete();\r\n      };\r\n    });\r\n  }\r\n\r\n  private pollServerSession(\r\n    clientId: string | undefined,\r\n    configuration: OpenIdConfiguration\r\n  ): void {\r\n    this.outstandingMessages = 0;\r\n\r\n    const pollServerSessionRecur = (): void => {\r\n      this.init(configuration)\r\n        .pipe(take(1))\r\n        .subscribe(() => {\r\n          const existingIframe = this.getExistingIframe();\r\n\r\n          if (existingIframe && clientId) {\r\n            this.loggerService.logDebug(\r\n              configuration,\r\n              `CheckSession - clientId : '${clientId}' - existingIframe: '${existingIframe}'`\r\n            );\r\n            const sessionState = this.storagePersistenceService.read(\r\n              'session_state',\r\n              configuration\r\n            );\r\n            const authWellKnownEndPoints = this.storagePersistenceService.read(\r\n              'authWellKnownEndPoints',\r\n              configuration\r\n            );\r\n            const contentWindow = existingIframe.contentWindow;\r\n\r\n            if (\r\n              sessionState &&\r\n              authWellKnownEndPoints?.checkSessionIframe &&\r\n              contentWindow\r\n            ) {\r\n              const iframeOrigin = new URL(\r\n                authWellKnownEndPoints.checkSessionIframe\r\n              )?.origin;\r\n\r\n              this.outstandingMessages++;\r\n              contentWindow.postMessage(\r\n                clientId + ' ' + sessionState,\r\n                iframeOrigin\r\n              );\r\n            } else {\r\n              this.loggerService.logDebug(\r\n                configuration,\r\n                `CheckSession - session_state is '${sessionState}' - AuthWellKnownEndPoints is '${JSON.stringify(\r\n                  authWellKnownEndPoints,\r\n                  null,\r\n                  2\r\n                )}'`\r\n              );\r\n              this.checkSessionChangedInternal$.next(true);\r\n            }\r\n          } else {\r\n            this.loggerService.logWarning(\r\n              configuration,\r\n              `CheckSession - OidcSecurityCheckSession pollServerSession checkSession IFrame does not exist:\r\n               clientId : '${clientId}' - existingIframe: '${existingIframe}'`\r\n            );\r\n          }\r\n\r\n          // after sending three messages with no response, fail.\r\n          if (this.outstandingMessages > 3) {\r\n            this.loggerService.logError(\r\n              configuration,\r\n              `CheckSession - OidcSecurityCheckSession not receiving check session response messages.\r\n                            Outstanding messages: '${this.outstandingMessages}'. Server unreachable?`\r\n            );\r\n          }\r\n\r\n          this.zone.runOutsideAngular(() => {\r\n            this.scheduledHeartBeatRunning =\r\n              this.document?.defaultView?.setTimeout(\r\n                () => this.zone.run(pollServerSessionRecur),\r\n                this.heartBeatInterval\r\n              ) ?? null;\r\n          });\r\n        });\r\n    };\r\n\r\n    pollServerSessionRecur();\r\n  }\r\n\r\n  private clearScheduledHeartBeat(): void {\r\n    if (this.scheduledHeartBeatRunning !== null) {\r\n      clearTimeout(this.scheduledHeartBeatRunning);\r\n      this.scheduledHeartBeatRunning = null;\r\n    }\r\n  }\r\n\r\n  private messageHandler(configuration: OpenIdConfiguration, e: any): void {\r\n    const existingIFrame = this.getExistingIframe();\r\n    const authWellKnownEndPoints = this.storagePersistenceService.read(\r\n      'authWellKnownEndPoints',\r\n      configuration\r\n    );\r\n    const startsWith = !!authWellKnownEndPoints?.checkSessionIframe?.startsWith(\r\n      e.origin\r\n    );\r\n\r\n    this.outstandingMessages = 0;\r\n\r\n    if (\r\n      existingIFrame &&\r\n      startsWith &&\r\n      e.source === existingIFrame.contentWindow\r\n    ) {\r\n      if (e.data === 'error') {\r\n        this.loggerService.logWarning(\r\n          configuration,\r\n          'CheckSession - error from check session messageHandler'\r\n        );\r\n      } else if (e.data === 'changed') {\r\n        this.loggerService.logDebug(\r\n          configuration,\r\n          `CheckSession - ${e} from check session messageHandler`\r\n        );\r\n        this.checkSessionReceived = true;\r\n        this.eventService.fireEvent(EventTypes.CheckSessionReceived, e.data);\r\n        this.checkSessionChangedInternal$.next(true);\r\n      } else {\r\n        this.eventService.fireEvent(EventTypes.CheckSessionReceived, e.data);\r\n        this.loggerService.logDebug(\r\n          configuration,\r\n          `CheckSession - ${e.data} from check session messageHandler`\r\n        );\r\n      }\r\n    }\r\n  }\r\n\r\n  private bindMessageEventToIframe(configuration: OpenIdConfiguration): void {\r\n    this.iframeMessageEventListener = this.messageHandler.bind(\r\n      this,\r\n      configuration\r\n    );\r\n\r\n    const defaultView = this.document.defaultView;\r\n\r\n    if (defaultView) {\r\n      defaultView.addEventListener(\r\n        'message',\r\n        this.iframeMessageEventListener,\r\n        false\r\n      );\r\n    }\r\n  }\r\n\r\n  private getOrCreateIframe(\r\n    configuration: OpenIdConfiguration\r\n  ): HTMLIFrameElement {\r\n    return (\r\n      this.getExistingIframe() ||\r\n      this.iFrameService.addIFrameToWindowBody(\r\n        IFRAME_FOR_CHECK_SESSION_IDENTIFIER,\r\n        configuration\r\n      )\r\n    );\r\n  }\r\n}\r\n"]}