UNPKG

angular-auth-oidc-client

Version:

An OpenID Connect Code Flow with PKCE,Implicit Flow client for Angular

301 lines 24.9 kB
/** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc */ import { Injectable, NgZone } from '@angular/core'; import { from, Observable, Subject } from 'rxjs'; import { take } from 'rxjs/operators'; import { ConfigurationProvider } from './auth-configuration.provider'; import { IFrameService } from './existing-iframe.service'; import { LoggerService } from './oidc.logger.service'; import { OidcSecurityCommon } from './oidc.security.common'; /** @type {?} */ var IFRAME_FOR_CHECK_SESSION_IDENTIFIER = 'myiFrameForCheckSession'; // http://openid.net/specs/openid-connect-session-1_0-ID4.html var OidcSecurityCheckSession = /** @class */ (function () { function OidcSecurityCheckSession(oidcSecurityCommon, loggerService, iFrameService, zone, configurationProvider) { this.oidcSecurityCommon = oidcSecurityCommon; this.loggerService = loggerService; this.iFrameService = iFrameService; this.zone = zone; this.configurationProvider = configurationProvider; this.lastIFrameRefresh = 0; this.outstandingMessages = 0; this.heartBeatInterval = 3000; this.iframeRefreshInterval = 60000; this._onCheckSessionChanged = new Subject(); } Object.defineProperty(OidcSecurityCheckSession.prototype, "onCheckSessionChanged", { get: /** * @return {?} */ function () { return this._onCheckSessionChanged.asObservable(); }, enumerable: true, configurable: true }); /** * @private * @return {?} */ OidcSecurityCheckSession.prototype.doesSessionExist = /** * @private * @return {?} */ function () { /** @type {?} */ var existingIFrame = this.iFrameService.getExistingIFrame(IFRAME_FOR_CHECK_SESSION_IDENTIFIER); if (!existingIFrame) { return false; } this.sessionIframe = existingIFrame; return true; }; /** * @private * @return {?} */ OidcSecurityCheckSession.prototype.init = /** * @private * @return {?} */ function () { var _this = this; if (this.lastIFrameRefresh + this.iframeRefreshInterval > Date.now()) { return from([this]); } if (!this.doesSessionExist()) { this.sessionIframe = this.iFrameService.addIFrameToWindowBody(IFRAME_FOR_CHECK_SESSION_IDENTIFIER); this.iframeMessageEvent = this.messageHandler.bind(this); window.addEventListener('message', this.iframeMessageEvent, false); } if (!this.configurationProvider.wellKnownEndpoints) { this.loggerService.logWarning('init check session: authWellKnownEndpoints is undefined. Returning.'); return; } if (this.configurationProvider.wellKnownEndpoints.check_session_iframe) { this.sessionIframe.contentWindow.location.replace(this.configurationProvider.wellKnownEndpoints.check_session_iframe); } else { this.loggerService.logWarning('init check session: authWellKnownEndpoints is undefined'); } return Observable.create((/** * @param {?} observer * @return {?} */ function (observer) { _this.sessionIframe.onload = (/** * @return {?} */ function () { _this.lastIFrameRefresh = Date.now(); observer.next(_this); observer.complete(); }); })); }; /** * @param {?} clientId * @return {?} */ OidcSecurityCheckSession.prototype.startCheckingSession = /** * @param {?} clientId * @return {?} */ function (clientId) { if (this.scheduledHeartBeat) { return; } this.pollServerSession(clientId); }; /** * @return {?} */ OidcSecurityCheckSession.prototype.stopCheckingSession = /** * @return {?} */ function () { if (!this.scheduledHeartBeat) { return; } this.clearScheduledHeartBeat(); }; /** * @private * @param {?} clientId * @return {?} */ OidcSecurityCheckSession.prototype.pollServerSession = /** * @private * @param {?} clientId * @return {?} */ function (clientId) { var _this = this; /** @type {?} */ var _pollServerSessionRecur = (/** * @return {?} */ function () { _this.init() .pipe(take(1)) .subscribe((/** * @return {?} */ function () { if (_this.sessionIframe && clientId) { _this.loggerService.logDebug(_this.sessionIframe); /** @type {?} */ var session_state = _this.oidcSecurityCommon.sessionState; if (session_state) { _this.outstandingMessages++; _this.sessionIframe.contentWindow.postMessage(clientId + ' ' + session_state, _this.configurationProvider.openIDConfiguration.stsServer); } else { _this.loggerService.logDebug('OidcSecurityCheckSession pollServerSession session_state is blank'); _this._onCheckSessionChanged.next(); } } else { _this.loggerService.logWarning('OidcSecurityCheckSession pollServerSession sessionIframe does not exist'); _this.loggerService.logDebug(clientId); _this.loggerService.logDebug(_this.sessionIframe); // this.init(); } // after sending three messages with no response, fail. if (_this.outstandingMessages > 3) { _this.loggerService.logError("OidcSecurityCheckSession not receiving check session response messages. Outstanding messages: " + _this.outstandingMessages + ". Server unreachable?"); _this._onCheckSessionChanged.next(); } _this.scheduledHeartBeat = setTimeout(_pollServerSessionRecur, _this.heartBeatInterval); })); }); this.outstandingMessages = 0; this.zone.runOutsideAngular((/** * @return {?} */ function () { _this.scheduledHeartBeat = setTimeout(_pollServerSessionRecur, _this.heartBeatInterval); })); }; /** * @private * @return {?} */ OidcSecurityCheckSession.prototype.clearScheduledHeartBeat = /** * @private * @return {?} */ function () { clearTimeout(this.scheduledHeartBeat); this.scheduledHeartBeat = null; }; /** * @private * @param {?} e * @return {?} */ OidcSecurityCheckSession.prototype.messageHandler = /** * @private * @param {?} e * @return {?} */ function (e) { this.outstandingMessages = 0; if (this.sessionIframe && e.origin === this.configurationProvider.openIDConfiguration.stsServer && e.source === this.sessionIframe.contentWindow) { if (e.data === 'error') { this.loggerService.logWarning('error from checksession messageHandler'); } else if (e.data === 'changed') { this._onCheckSessionChanged.next(); } else { this.loggerService.logDebug(e.data + ' from checksession messageHandler'); } } }; OidcSecurityCheckSession.decorators = [ { type: Injectable } ]; /** @nocollapse */ OidcSecurityCheckSession.ctorParameters = function () { return [ { type: OidcSecurityCommon }, { type: LoggerService }, { type: IFrameService }, { type: NgZone }, { type: ConfigurationProvider } ]; }; return OidcSecurityCheckSession; }()); export { OidcSecurityCheckSession }; if (false) { /** * @type {?} * @private */ OidcSecurityCheckSession.prototype.sessionIframe; /** * @type {?} * @private */ OidcSecurityCheckSession.prototype.iframeMessageEvent; /** * @type {?} * @private */ OidcSecurityCheckSession.prototype.scheduledHeartBeat; /** * @type {?} * @private */ OidcSecurityCheckSession.prototype.lastIFrameRefresh; /** * @type {?} * @private */ OidcSecurityCheckSession.prototype.outstandingMessages; /** * @type {?} * @private */ OidcSecurityCheckSession.prototype.heartBeatInterval; /** * @type {?} * @private */ OidcSecurityCheckSession.prototype.iframeRefreshInterval; /** * @type {?} * @private */ OidcSecurityCheckSession.prototype._onCheckSessionChanged; /** * @type {?} * @private */ OidcSecurityCheckSession.prototype.oidcSecurityCommon; /** * @type {?} * @private */ OidcSecurityCheckSession.prototype.loggerService; /** * @type {?} * @private */ OidcSecurityCheckSession.prototype.iFrameService; /** * @type {?} * @private */ OidcSecurityCheckSession.prototype.zone; /** * @type {?} * @private */ OidcSecurityCheckSession.prototype.configurationProvider; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"oidc.security.check-session.js","sourceRoot":"ng://angular-auth-oidc-client/","sources":["lib/services/oidc.security.check-session.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone } from '@angular/core';\nimport { from, Observable, Observer, Subject } from 'rxjs';\nimport { take } from 'rxjs/operators';\nimport { ConfigurationProvider } from './auth-configuration.provider';\nimport { IFrameService } from './existing-iframe.service';\nimport { LoggerService } from './oidc.logger.service';\nimport { OidcSecurityCommon } from './oidc.security.common';\n\nconst IFRAME_FOR_CHECK_SESSION_IDENTIFIER = 'myiFrameForCheckSession';\n\n// http://openid.net/specs/openid-connect-session-1_0-ID4.html\n\n@Injectable()\nexport class OidcSecurityCheckSession {\n    private sessionIframe: any;\n    private iframeMessageEvent: any;\n    private scheduledHeartBeat: any;\n    private lastIFrameRefresh = 0;\n    private outstandingMessages = 0;\n    private heartBeatInterval = 3000;\n    private iframeRefreshInterval = 60000;\n    private _onCheckSessionChanged = new Subject<any>();\n\n    public get onCheckSessionChanged(): Observable<any> {\n        return this._onCheckSessionChanged.asObservable();\n    }\n\n    constructor(\n        private oidcSecurityCommon: OidcSecurityCommon,\n        private loggerService: LoggerService,\n        private iFrameService: IFrameService,\n        private zone: NgZone,\n        private readonly configurationProvider: ConfigurationProvider\n    ) {}\n\n    private doesSessionExist(): boolean {\n        const existingIFrame = this.iFrameService.getExistingIFrame(IFRAME_FOR_CHECK_SESSION_IDENTIFIER);\n\n        if (!existingIFrame) {\n            return false;\n        }\n\n        this.sessionIframe = existingIFrame;\n        return true;\n    }\n\n    private init() {\n        if (this.lastIFrameRefresh + this.iframeRefreshInterval > Date.now()) {\n            return from([this]);\n        }\n\n        if (!this.doesSessionExist()) {\n            this.sessionIframe = this.iFrameService.addIFrameToWindowBody(IFRAME_FOR_CHECK_SESSION_IDENTIFIER);\n            this.iframeMessageEvent = this.messageHandler.bind(this);\n            window.addEventListener('message', this.iframeMessageEvent, false);\n        }\n\n        if (!this.configurationProvider.wellKnownEndpoints) {\n            this.loggerService.logWarning('init check session: authWellKnownEndpoints is undefined. Returning.');\n            return;\n        }\n\n        if (this.configurationProvider.wellKnownEndpoints.check_session_iframe) {\n            this.sessionIframe.contentWindow.location.replace(this.configurationProvider.wellKnownEndpoints.check_session_iframe);\n        } else {\n            this.loggerService.logWarning('init check session: authWellKnownEndpoints is undefined');\n        }\n\n        return Observable.create((observer: Observer<OidcSecurityCheckSession>) => {\n            this.sessionIframe.onload = () => {\n                this.lastIFrameRefresh = Date.now();\n                observer.next(this);\n                observer.complete();\n            };\n        });\n    }\n\n    startCheckingSession(clientId: string): void {\n        if (this.scheduledHeartBeat) {\n            return;\n        }\n\n        this.pollServerSession(clientId);\n    }\n\n    stopCheckingSession(): void {\n        if (!this.scheduledHeartBeat) {\n            return;\n        }\n\n        this.clearScheduledHeartBeat();\n    }\n\n    private pollServerSession(clientId: string) {\n        const _pollServerSessionRecur = () => {\n            this.init()\n                .pipe(take(1))\n                .subscribe(() => {\n                    if (this.sessionIframe && clientId) {\n                        this.loggerService.logDebug(this.sessionIframe);\n                        const session_state = this.oidcSecurityCommon.sessionState;\n                        if (session_state) {\n                            this.outstandingMessages++;\n                            this.sessionIframe.contentWindow.postMessage(\n                                clientId + ' ' + session_state,\n                                this.configurationProvider.openIDConfiguration.stsServer\n                            );\n                        } else {\n                            this.loggerService.logDebug('OidcSecurityCheckSession pollServerSession session_state is blank');\n                            this._onCheckSessionChanged.next();\n                        }\n                    } else {\n                        this.loggerService.logWarning('OidcSecurityCheckSession pollServerSession sessionIframe does not exist');\n                        this.loggerService.logDebug(clientId);\n                        this.loggerService.logDebug(this.sessionIframe);\n                        // this.init();\n                    }\n\n                    // after sending three messages with no response, fail.\n                    if (this.outstandingMessages > 3) {\n                        this.loggerService.logError(\n                            `OidcSecurityCheckSession not receiving check session response messages. Outstanding messages: ${\n                                this.outstandingMessages\n                            }. Server unreachable?`\n                        );\n                        this._onCheckSessionChanged.next();\n                    }\n\n                    this.scheduledHeartBeat = setTimeout(_pollServerSessionRecur, this.heartBeatInterval);\n                });\n        };\n\n        this.outstandingMessages = 0;\n\n        this.zone.runOutsideAngular(() => {\n            this.scheduledHeartBeat = setTimeout(_pollServerSessionRecur, this.heartBeatInterval);\n        });\n    }\n    private clearScheduledHeartBeat() {\n        clearTimeout(this.scheduledHeartBeat);\n        this.scheduledHeartBeat = null;\n    }\n\n    private messageHandler(e: any) {\n        this.outstandingMessages = 0;\n        if (\n            this.sessionIframe &&\n            e.origin === this.configurationProvider.openIDConfiguration.stsServer &&\n            e.source === this.sessionIframe.contentWindow\n        ) {\n            if (e.data === 'error') {\n                this.loggerService.logWarning('error from checksession messageHandler');\n            } else if (e.data === 'changed') {\n                this._onCheckSessionChanged.next();\n            } else {\n                this.loggerService.logDebug(e.data + ' from checksession messageHandler');\n            }\n        }\n    }\n}\n"]}