angular-auth-oidc-client
Version:
Angular Lib for OpenID Connect & OAuth2
165 lines • 27.9 kB
JavaScript
import { DOCUMENT } from '@angular/common';
import { Inject, Injectable } from '@angular/core';
import { BehaviorSubject, Observable, of } from 'rxjs';
import { take } from 'rxjs/operators';
import { EventTypes } from '../public-events/event-types';
import * as i0 from "@angular/core";
import * as i1 from "../storage/storage-persistence.service";
import * as i2 from "../logging/logger.service";
import * as i3 from "./existing-iframe.service";
import * as i4 from "../public-events/public-events.service";
const IFRAME_FOR_CHECK_SESSION_IDENTIFIER = 'myiFrameForCheckSession';
// http://openid.net/specs/openid-connect-session-1_0-ID4.html
export class CheckSessionService {
constructor(storagePersistenceService, loggerService, iFrameService, eventService, zone, document) {
this.storagePersistenceService = storagePersistenceService;
this.loggerService = loggerService;
this.iFrameService = iFrameService;
this.eventService = eventService;
this.zone = zone;
this.document = document;
this.checkSessionReceived = false;
this.lastIFrameRefresh = 0;
this.outstandingMessages = 0;
this.heartBeatInterval = 3000;
this.iframeRefreshInterval = 60000;
this.checkSessionChangedInternal$ = new BehaviorSubject(false);
}
get checkSessionChanged$() {
return this.checkSessionChangedInternal$.asObservable();
}
isCheckSessionConfigured(configuration) {
const { startCheckSession } = configuration;
return 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 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);
const checkSessionIframe = authWellKnownEndPoints.checkSessionIframe;
if (checkSessionIframe) {
existingIframe.contentWindow.location.replace(checkSessionIframe);
}
else {
this.loggerService.logWarning(configuration, 'CheckSession - init check session: checkSessionIframe is not configured to run');
}
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);
if (sessionState && authWellKnownEndPoints?.checkSessionIframe) {
const iframeOrigin = new URL(authWellKnownEndPoints.checkSessionIframe)?.origin;
this.outstandingMessages++;
existingIframe.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 = setTimeout(() => this.zone.run(pollServerSessionRecur), this.heartBeatInterval);
});
});
};
pollServerSessionRecur();
}
clearScheduledHeartBeat() {
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) {
const iframeMessageEvent = this.messageHandler.bind(this, configuration);
this.document.defaultView.addEventListener('message', iframeMessageEvent, false);
}
getOrCreateIframe(configuration) {
const existingIframe = this.getExistingIframe();
if (!existingIframe) {
const frame = this.iFrameService.addIFrameToWindowBody(IFRAME_FOR_CHECK_SESSION_IDENTIFIER, configuration);
this.bindMessageEventToIframe(configuration);
return frame;
}
return existingIframe;
}
}
CheckSessionService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.1.0", ngImport: i0, type: CheckSessionService, deps: [{ token: i1.StoragePersistenceService }, { token: i2.LoggerService }, { token: i3.IFrameService }, { token: i4.PublicEventsService }, { token: i0.NgZone }, { token: DOCUMENT }], target: i0.ɵɵFactoryTarget.Injectable });
CheckSessionService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "14.1.0", ngImport: i0, type: CheckSessionService });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.1.0", ngImport: i0, type: CheckSessionService, decorators: [{
type: Injectable
}], ctorParameters: function () { return [{ type: i1.StoragePersistenceService }, { type: i2.LoggerService }, { type: i3.IFrameService }, { type: i4.PublicEventsService }, { type: i0.NgZone }, { type: undefined, decorators: [{
type: Inject,
args: [DOCUMENT]
}] }]; } });
//# 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":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAC3C,OAAO,EAAE,MAAM,EAAE,UAAU,EAAU,MAAM,eAAe,CAAC;AAC3D,OAAO,EAAE,eAAe,EAAE,UAAU,EAAE,EAAE,EAAE,MAAM,MAAM,CAAC;AACvD,OAAO,EAAE,IAAI,EAAE,MAAM,gBAAgB,CAAC;AAEtC,OAAO,EAAE,UAAU,EAAE,MAAM,8BAA8B,CAAC;;;;;;AAM1D,MAAM,mCAAmC,GAAG,yBAAyB,CAAC;AAEtE,8DAA8D;AAG9D,MAAM,OAAO,mBAAmB;IAmB9B,YACmB,yBAAoD,EACpD,aAA4B,EAC5B,aAA4B,EAC5B,YAAiC,EACjC,IAAY,EACM,QAAa;QAL/B,8BAAyB,GAAzB,yBAAyB,CAA2B;QACpD,kBAAa,GAAb,aAAa,CAAe;QAC5B,kBAAa,GAAb,aAAa,CAAe;QAC5B,iBAAY,GAAZ,YAAY,CAAqB;QACjC,SAAI,GAAJ,IAAI,CAAQ;QACM,aAAQ,GAAR,QAAQ,CAAK;QAxB1C,yBAAoB,GAAG,KAAK,CAAC;QAI7B,sBAAiB,GAAG,CAAC,CAAC;QAEtB,wBAAmB,GAAG,CAAC,CAAC;QAEf,sBAAiB,GAAG,IAAI,CAAC;QAEzB,0BAAqB,GAAG,KAAK,CAAC;QAE9B,iCAA4B,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;IAajF,CAAC;IAXJ,IAAI,oBAAoB;QACtB,OAAO,IAAI,CAAC,4BAA4B,CAAC,YAAY,EAAE,CAAC;IAC1D,CAAC;IAWD,wBAAwB,CAAC,aAAkC;QACzD,MAAM,EAAE,iBAAiB,EAAE,GAAG,aAAa,CAAC;QAE5C,OAAO,iBAAiB,CAAC;IAC3B,CAAC;IAED,KAAK,CAAC,aAAkC;QACtC,IAAI,CAAC,CAAC,IAAI,CAAC,yBAAyB,EAAE;YACpC,OAAO;SACR;QAED,MAAM,EAAE,QAAQ,EAAE,GAAG,aAAa,CAAC;QAEnC,IAAI,CAAC,iBAAiB,CAAC,QAAQ,EAAE,aAAa,CAAC,CAAC;IAClD,CAAC;IAED,IAAI;QACF,IAAI,CAAC,IAAI,CAAC,yBAAyB,EAAE;YACnC,OAAO;SACR;QAED,IAAI,CAAC,uBAAuB,EAAE,CAAC;QAC/B,IAAI,CAAC,oBAAoB,GAAG,KAAK,CAAC;IACpC,CAAC;IAED,kBAAkB,CAAC,aAAkC;QACnD,MAAM,EAAE,iBAAiB,EAAE,GAAG,aAAa,CAAC;QAE5C,OAAO,iBAAiB,IAAI,IAAI,CAAC,oBAAoB,CAAC;IACxD,CAAC;IAED,iBAAiB;QACf,OAAO,IAAI,CAAC,aAAa,CAAC,iBAAiB,CAAC,mCAAmC,CAAC,CAAC;IACnF,CAAC;IAEO,IAAI,CAAC,aAAkC;QAC7C,IAAI,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC,GAAG,EAAE,EAAE;YACpE,OAAO,EAAE,CAAC,SAAS,CAAC,CAAC;SACtB;QAED,MAAM,sBAAsB,GAAG,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,wBAAwB,EAAE,aAAa,CAAC,CAAC;QAE5G,IAAI,CAAC,sBAAsB,EAAE;YAC3B,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC,aAAa,EAAE,oFAAoF,CAAC,CAAC;YAEnI,OAAO,EAAE,EAAE,CAAC;SACb;QAED,MAAM,cAAc,GAAG,IAAI,CAAC,iBAAiB,CAAC,aAAa,CAAC,CAAC;QAC7D,MAAM,kBAAkB,GAAG,sBAAsB,CAAC,kBAAkB,CAAC;QAErE,IAAI,kBAAkB,EAAE;YACtB,cAAc,CAAC,aAAa,CAAC,QAAQ,CAAC,OAAO,CAAC,kBAAkB,CAAC,CAAC;SACnE;aAAM;YACL,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC,aAAa,EAAE,gFAAgF,CAAC,CAAC;SAChI;QAED,OAAO,IAAI,UAAU,CAAC,CAAC,QAAQ,EAAE,EAAE;YACjC,cAAc,CAAC,MAAM,GAAG,GAAS,EAAE;gBACjC,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;gBACpC,QAAQ,CAAC,IAAI,EAAE,CAAC;gBAChB,QAAQ,CAAC,QAAQ,EAAE,CAAC;YACtB,CAAC,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,iBAAiB,CAAC,QAAgB,EAAE,aAAkC;QAC5E,IAAI,CAAC,mBAAmB,GAAG,CAAC,CAAC;QAE7B,MAAM,sBAAsB,GAAG,GAAS,EAAE;YACxC,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC;iBACrB,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;iBACb,SAAS,CAAC,GAAG,EAAE;gBACd,MAAM,cAAc,GAAG,IAAI,CAAC,iBAAiB,EAAE,CAAC;gBAEhD,IAAI,cAAc,IAAI,QAAQ,EAAE;oBAC9B,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,aAAa,EAAE,8BAA8B,QAAQ,wBAAwB,cAAc,GAAG,CAAC,CAAC;oBAC5H,MAAM,YAAY,GAAG,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,eAAe,EAAE,aAAa,CAAC,CAAC;oBACzF,MAAM,sBAAsB,GAAG,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,wBAAwB,EAAE,aAAa,CAAC,CAAC;oBAE5G,IAAI,YAAY,IAAI,sBAAsB,EAAE,kBAAkB,EAAE;wBAC9D,MAAM,YAAY,GAAG,IAAI,GAAG,CAAC,sBAAsB,CAAC,kBAAkB,CAAC,EAAE,MAAM,CAAC;wBAEhF,IAAI,CAAC,mBAAmB,EAAE,CAAC;wBAC3B,cAAc,CAAC,aAAa,CAAC,WAAW,CAAC,QAAQ,GAAG,GAAG,GAAG,YAAY,EAAE,YAAY,CAAC,CAAC;qBACvF;yBAAM;wBACL,IAAI,CAAC,aAAa,CAAC,QAAQ,CACzB,aAAa,EACb,oCAAoC,YAAY,kCAAkC,IAAI,CAAC,SAAS,CAC9F,sBAAsB,EACtB,IAAI,EACJ,CAAC,CACF,GAAG,CACL,CAAC;wBACF,IAAI,CAAC,4BAA4B,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;qBAC9C;iBACF;qBAAM;oBACL,IAAI,CAAC,aAAa,CAAC,UAAU,CAC3B,aAAa,EACb;6BACe,QAAQ,wBAAwB,cAAc,GAAG,CACjE,CAAC;iBACH;gBAED,uDAAuD;gBACvD,IAAI,IAAI,CAAC,mBAAmB,GAAG,CAAC,EAAE;oBAChC,IAAI,CAAC,aAAa,CAAC,QAAQ,CACzB,aAAa,EACb;qDACuC,IAAI,CAAC,mBAAmB,wBAAwB,CACxF,CAAC;iBACH;gBAED,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,GAAG,EAAE;oBAC/B,IAAI,CAAC,yBAAyB,GAAG,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,sBAAsB,CAAC,EAAE,IAAI,CAAC,iBAAiB,CAAC,CAAC;gBACnH,CAAC,CAAC,CAAC;YACL,CAAC,CAAC,CAAC;QACP,CAAC,CAAC;QAEF,sBAAsB,EAAE,CAAC;IAC3B,CAAC;IAEO,uBAAuB;QAC7B,YAAY,CAAC,IAAI,CAAC,yBAAyB,CAAC,CAAC;QAC7C,IAAI,CAAC,yBAAyB,GAAG,IAAI,CAAC;IACxC,CAAC;IAEO,cAAc,CAAC,aAAkC,EAAE,CAAM;QAC/D,MAAM,cAAc,GAAG,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAChD,MAAM,sBAAsB,GAAG,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,wBAAwB,EAAE,aAAa,CAAC,CAAC;QAC5G,MAAM,UAAU,GAAG,CAAC,CAAC,sBAAsB,EAAE,kBAAkB,EAAE,UAAU,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;QAEtF,IAAI,CAAC,mBAAmB,GAAG,CAAC,CAAC;QAE7B,IAAI,cAAc,IAAI,UAAU,IAAI,CAAC,CAAC,MAAM,KAAK,cAAc,CAAC,aAAa,EAAE;YAC7E,IAAI,CAAC,CAAC,IAAI,KAAK,OAAO,EAAE;gBACtB,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC,aAAa,EAAE,wDAAwD,CAAC,CAAC;aACxG;iBAAM,IAAI,CAAC,CAAC,IAAI,KAAK,SAAS,EAAE;gBAC/B,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,aAAa,EAAE,kBAAkB,CAAC,oCAAoC,CAAC,CAAC;gBACpG,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC;gBACjC,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,UAAU,CAAC,oBAAoB,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC;gBACrE,IAAI,CAAC,4BAA4B,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;aAC9C;iBAAM;gBACL,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,UAAU,CAAC,oBAAoB,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC;gBACrE,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,aAAa,EAAE,kBAAkB,CAAC,CAAC,IAAI,oCAAoC,CAAC,CAAC;aAC1G;SACF;IACH,CAAC;IAEO,wBAAwB,CAAC,aAAkC;QACjE,MAAM,kBAAkB,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,EAAE,aAAa,CAAC,CAAC;QAEzE,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,gBAAgB,CAAC,SAAS,EAAE,kBAAkB,EAAE,KAAK,CAAC,CAAC;IACnF,CAAC;IAEO,iBAAiB,CAAC,aAAkC;QAC1D,MAAM,cAAc,GAAG,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAEhD,IAAI,CAAC,cAAc,EAAE;YACnB,MAAM,KAAK,GAAG,IAAI,CAAC,aAAa,CAAC,qBAAqB,CAAC,mCAAmC,EAAE,aAAa,CAAC,CAAC;YAE3G,IAAI,CAAC,wBAAwB,CAAC,aAAa,CAAC,CAAC;YAE7C,OAAO,KAAK,CAAC;SACd;QAED,OAAO,cAAc,CAAC;IACxB,CAAC;;gHAnMU,mBAAmB,8KAyBpB,QAAQ;oHAzBP,mBAAmB;2FAAnB,mBAAmB;kBAD/B,UAAU;;0BA0BN,MAAM;2BAAC,QAAQ","sourcesContent":["import { DOCUMENT } from '@angular/common';\r\nimport { Inject, Injectable, NgZone } from '@angular/core';\r\nimport { BehaviorSubject, Observable, of } from 'rxjs';\r\nimport { take } from 'rxjs/operators';\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 { OpenIdConfiguration } from './../config/openid-configuration';\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()\r\nexport class CheckSessionService {\r\n  private checkSessionReceived = false;\r\n\r\n  private scheduledHeartBeatRunning: any;\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>(false);\r\n\r\n  get checkSessionChanged$(): Observable<boolean> {\r\n    return this.checkSessionChangedInternal$.asObservable();\r\n  }\r\n\r\n  constructor(\r\n    private readonly storagePersistenceService: StoragePersistenceService,\r\n    private readonly loggerService: LoggerService,\r\n    private readonly iFrameService: IFrameService,\r\n    private readonly eventService: PublicEventsService,\r\n    private readonly zone: NgZone,\r\n    @Inject(DOCUMENT) private readonly document: any\r\n  ) {}\r\n\r\n  isCheckSessionConfigured(configuration: OpenIdConfiguration): boolean {\r\n    const { startCheckSession } = configuration;\r\n\r\n    return 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 startCheckSession && this.checkSessionReceived;\r\n  }\r\n\r\n  getExistingIframe(): HTMLIFrameElement {\r\n    return this.iFrameService.getExistingIFrame(IFRAME_FOR_CHECK_SESSION_IDENTIFIER);\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('authWellKnownEndPoints', configuration);\r\n\r\n    if (!authWellKnownEndPoints) {\r\n      this.loggerService.logWarning(configuration, 'CheckSession - init check session: authWellKnownEndpoints is undefined. Returning.');\r\n\r\n      return of();\r\n    }\r\n\r\n    const existingIframe = this.getOrCreateIframe(configuration);\r\n    const checkSessionIframe = authWellKnownEndPoints.checkSessionIframe;\r\n\r\n    if (checkSessionIframe) {\r\n      existingIframe.contentWindow.location.replace(checkSessionIframe);\r\n    } else {\r\n      this.loggerService.logWarning(configuration, 'CheckSession - init check session: checkSessionIframe is not configured to run');\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(clientId: string, configuration: OpenIdConfiguration): 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(configuration, `CheckSession - clientId : '${clientId}' - existingIframe: '${existingIframe}'`);\r\n            const sessionState = this.storagePersistenceService.read('session_state', configuration);\r\n            const authWellKnownEndPoints = this.storagePersistenceService.read('authWellKnownEndPoints', configuration);\r\n\r\n            if (sessionState && authWellKnownEndPoints?.checkSessionIframe) {\r\n              const iframeOrigin = new URL(authWellKnownEndPoints.checkSessionIframe)?.origin;\r\n\r\n              this.outstandingMessages++;\r\n              existingIframe.contentWindow.postMessage(clientId + ' ' + sessionState, iframeOrigin);\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 = setTimeout(() => this.zone.run(pollServerSessionRecur), this.heartBeatInterval);\r\n          });\r\n        });\r\n    };\r\n\r\n    pollServerSessionRecur();\r\n  }\r\n\r\n  private clearScheduledHeartBeat(): void {\r\n    clearTimeout(this.scheduledHeartBeatRunning);\r\n    this.scheduledHeartBeatRunning = null;\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('authWellKnownEndPoints', configuration);\r\n    const startsWith = !!authWellKnownEndPoints?.checkSessionIframe?.startsWith(e.origin);\r\n\r\n    this.outstandingMessages = 0;\r\n\r\n    if (existingIFrame && startsWith && e.source === existingIFrame.contentWindow) {\r\n      if (e.data === 'error') {\r\n        this.loggerService.logWarning(configuration, 'CheckSession - error from check session messageHandler');\r\n      } else if (e.data === 'changed') {\r\n        this.loggerService.logDebug(configuration, `CheckSession - ${e} from check session messageHandler`);\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(configuration, `CheckSession - ${e.data} from check session messageHandler`);\r\n      }\r\n    }\r\n  }\r\n\r\n  private bindMessageEventToIframe(configuration: OpenIdConfiguration): void {\r\n    const iframeMessageEvent = this.messageHandler.bind(this, configuration);\r\n\r\n    this.document.defaultView.addEventListener('message', iframeMessageEvent, false);\r\n  }\r\n\r\n  private getOrCreateIframe(configuration: OpenIdConfiguration): HTMLIFrameElement {\r\n    const existingIframe = this.getExistingIframe();\r\n\r\n    if (!existingIframe) {\r\n      const frame = this.iFrameService.addIFrameToWindowBody(IFRAME_FOR_CHECK_SESSION_IDENTIFIER, configuration);\r\n\r\n      this.bindMessageEventToIframe(configuration);\r\n\r\n      return frame;\r\n    }\r\n\r\n    return existingIframe;\r\n  }\r\n}\r\n"]}