@huntsman-cancer-institute/authentication 15.5.2 → 15.6.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/authentication.service.d.ts +10 -2
- package/esm2020/authentication.service.mjs +47 -4
- package/esm2020/index.mjs +2 -2
- package/fesm2015/huntsman-cancer-institute-authentication.mjs +47 -4
- package/fesm2015/huntsman-cancer-institute-authentication.mjs.map +1 -1
- package/fesm2020/huntsman-cancer-institute-authentication.mjs +47 -4
- package/fesm2020/huntsman-cancer-institute-authentication.mjs.map +1 -1
- package/index.d.ts +1 -1
- package/package.json +1 -1
|
@@ -20,6 +20,8 @@ export declare let AUTHENTICATION_ROUTE: InjectionToken<string>;
|
|
|
20
20
|
export declare let AUTHENTICATION_MAX_INACTIVITY_MINUTES: InjectionToken<number>;
|
|
21
21
|
export declare let AUTHENTICATION_USER_COUNTDOWN_SECONDS: InjectionToken<number>;
|
|
22
22
|
export declare let AUTHENTICATION_IDP_INACTIVITY_MINUTES: InjectionToken<number>;
|
|
23
|
+
export declare let AUTHENTICATION_USE_UI_EVENTS_FOR_ACTIVITY: InjectionToken<boolean>;
|
|
24
|
+
export declare let AUTHENTICATION_APP_KEEP_ALIVE_ENDPOINT: InjectionToken<string>;
|
|
23
25
|
/**
|
|
24
26
|
* @since 1.0.0
|
|
25
27
|
*/
|
|
@@ -37,6 +39,8 @@ export declare class AuthenticationService {
|
|
|
37
39
|
private _maxInactivity;
|
|
38
40
|
private _userCountdownSeconds;
|
|
39
41
|
private _idpInactivityMinutes;
|
|
42
|
+
private _useUIEventsForActivity;
|
|
43
|
+
private _keepAliveEndpoint;
|
|
40
44
|
private locationStrategy;
|
|
41
45
|
/**
|
|
42
46
|
* The generic error message used when a server error is thrown without a status.
|
|
@@ -63,9 +67,10 @@ export declare class AuthenticationService {
|
|
|
63
67
|
private _refreshSubscription;
|
|
64
68
|
private _lastUserInteraction;
|
|
65
69
|
private _maxInactivityMinutes;
|
|
70
|
+
private _uiEvents;
|
|
66
71
|
private baseUrl;
|
|
67
72
|
private contextRoot;
|
|
68
|
-
constructor(_http: HttpClient, _router: Router, _localStorageService: CoolLocalStorage, _jwtHelper: JwtHelperService, authenticationProvider: AuthenticationProvider, _authenticationRoute: string, _logoutPath: string, _tokenEndpoint: string, _serverUrl: string, _directEndpoint: string, _maxInactivity: number, _userCountdownSeconds: number, _idpInactivityMinutes: number, locationStrategy: LocationStrategy);
|
|
73
|
+
constructor(_http: HttpClient, _router: Router, _localStorageService: CoolLocalStorage, _jwtHelper: JwtHelperService, authenticationProvider: AuthenticationProvider, _authenticationRoute: string, _logoutPath: string, _tokenEndpoint: string, _serverUrl: string, _directEndpoint: string, _maxInactivity: number, _userCountdownSeconds: number, _idpInactivityMinutes: number, _useUIEventsForActivity: boolean, _keepAliveEndpoint: string, locationStrategy: LocationStrategy);
|
|
69
74
|
getBaseUrl(): string;
|
|
70
75
|
getContextRoot(): string;
|
|
71
76
|
getHeaders(req: HttpRequest<any>): HttpHeaders;
|
|
@@ -81,6 +86,7 @@ export declare class AuthenticationService {
|
|
|
81
86
|
set redirectUrl(redirectUrl: string);
|
|
82
87
|
get redirectUrl(): string;
|
|
83
88
|
requestAccessToken(redirectOnSuccess: boolean): void;
|
|
89
|
+
makeKeepAliveRequest(): void;
|
|
84
90
|
/**
|
|
85
91
|
* Verifies whether or not a current user session exists.
|
|
86
92
|
*
|
|
@@ -90,6 +96,7 @@ export declare class AuthenticationService {
|
|
|
90
96
|
isAboutToTimeOut(): Observable<boolean>;
|
|
91
97
|
getTimeoutStart(): number;
|
|
92
98
|
tokenLocation(): string;
|
|
99
|
+
keepAliveLocation(): string;
|
|
93
100
|
directLoginLocation(): string;
|
|
94
101
|
logoutLocation(): string;
|
|
95
102
|
/**
|
|
@@ -110,6 +117,7 @@ export declare class AuthenticationService {
|
|
|
110
117
|
storeToken(token: string): void;
|
|
111
118
|
proceedIfAuthenticated(): boolean;
|
|
112
119
|
validateToken(token: string): boolean;
|
|
120
|
+
subscribeToUIActivity(): void;
|
|
113
121
|
subscribeToTokenRefresh(token: any): void;
|
|
114
122
|
unsubscribeFromTokenRefresh(): void;
|
|
115
123
|
getMaxViewPermission(): "view" | "viewident" | "viewlimited";
|
|
@@ -118,6 +126,6 @@ export declare class AuthenticationService {
|
|
|
118
126
|
private refreshTokenIfUserIsActive;
|
|
119
127
|
private hasValidConfig;
|
|
120
128
|
private handleError;
|
|
121
|
-
static ɵfac: i0.ɵɵFactoryDeclaration<AuthenticationService, [null, null, null, null, null, null, null, null, { optional: true; }, { optional: true; }, { optional: true; }, { optional: true; }, { optional: true; }, { optional: true; }]>;
|
|
129
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<AuthenticationService, [null, null, null, null, null, null, null, null, { optional: true; }, { optional: true; }, { optional: true; }, { optional: true; }, { optional: true; }, { optional: true; }, { optional: true; }, { optional: true; }]>;
|
|
122
130
|
static ɵprov: i0.ɵɵInjectableDeclaration<AuthenticationService>;
|
|
123
131
|
}
|
|
@@ -5,7 +5,7 @@ import { Injectable, InjectionToken, Inject, Optional, isDevMode } from "@angula
|
|
|
5
5
|
import { LocationStrategy } from "@angular/common";
|
|
6
6
|
import { Router } from "@angular/router";
|
|
7
7
|
import { HttpClient, HttpHeaders } from "@angular/common/http";
|
|
8
|
-
import { interval, BehaviorSubject, throwError } from "rxjs";
|
|
8
|
+
import { interval, BehaviorSubject, throwError, fromEvent } from "rxjs";
|
|
9
9
|
import { catchError, map } from "rxjs/operators";
|
|
10
10
|
import { JwtHelperService } from "@auth0/angular-jwt";
|
|
11
11
|
import { AuthenticationProvider } from "./authentication.provider";
|
|
@@ -30,11 +30,13 @@ export let AUTHENTICATION_ROUTE = new InjectionToken("authentication_route");
|
|
|
30
30
|
export let AUTHENTICATION_MAX_INACTIVITY_MINUTES = new InjectionToken("authentication_max_inactivity");
|
|
31
31
|
export let AUTHENTICATION_USER_COUNTDOWN_SECONDS = new InjectionToken("authentication_user_countdown_seconds");
|
|
32
32
|
export let AUTHENTICATION_IDP_INACTIVITY_MINUTES = new InjectionToken("authentication_idp_inactivity_minutes");
|
|
33
|
+
export let AUTHENTICATION_USE_UI_EVENTS_FOR_ACTIVITY = new InjectionToken("authentication_use_ui_events_for_activity");
|
|
34
|
+
export let AUTHENTICATION_APP_KEEP_ALIVE_ENDPOINT = new InjectionToken("authentication_app_keep_alive_endpoint");
|
|
33
35
|
/**
|
|
34
36
|
* @since 1.0.0
|
|
35
37
|
*/
|
|
36
38
|
export class AuthenticationService {
|
|
37
|
-
constructor(_http, _router, _localStorageService, _jwtHelper, authenticationProvider, _authenticationRoute, _logoutPath, _tokenEndpoint, _serverUrl, _directEndpoint, _maxInactivity, _userCountdownSeconds, _idpInactivityMinutes, locationStrategy) {
|
|
39
|
+
constructor(_http, _router, _localStorageService, _jwtHelper, authenticationProvider, _authenticationRoute, _logoutPath, _tokenEndpoint, _serverUrl, _directEndpoint, _maxInactivity, _userCountdownSeconds, _idpInactivityMinutes, _useUIEventsForActivity, _keepAliveEndpoint, locationStrategy) {
|
|
38
40
|
this._http = _http;
|
|
39
41
|
this._router = _router;
|
|
40
42
|
this._localStorageService = _localStorageService;
|
|
@@ -48,6 +50,8 @@ export class AuthenticationService {
|
|
|
48
50
|
this._maxInactivity = _maxInactivity;
|
|
49
51
|
this._userCountdownSeconds = _userCountdownSeconds;
|
|
50
52
|
this._idpInactivityMinutes = _idpInactivityMinutes;
|
|
53
|
+
this._useUIEventsForActivity = _useUIEventsForActivity;
|
|
54
|
+
this._keepAliveEndpoint = _keepAliveEndpoint;
|
|
51
55
|
this.locationStrategy = locationStrategy;
|
|
52
56
|
this.userCountdownSeconds = 60;
|
|
53
57
|
this.idpInactivityMinutes = 5;
|
|
@@ -58,6 +62,7 @@ export class AuthenticationService {
|
|
|
58
62
|
this._isAuthenticatedSubject = new BehaviorSubject(false);
|
|
59
63
|
this._userIsAboutToTimeOut = new BehaviorSubject(false);
|
|
60
64
|
this._maxInactivityMinutes = 120;
|
|
65
|
+
this._uiEvents = ['keydown', 'click', 'wheel', 'mousemove'];
|
|
61
66
|
this.contextRoot = "";
|
|
62
67
|
if (isDevMode()) {
|
|
63
68
|
console.debug("window.location.href: " + window.location.href);
|
|
@@ -81,6 +86,10 @@ export class AuthenticationService {
|
|
|
81
86
|
if (_idpInactivityMinutes) {
|
|
82
87
|
this.idpInactivityMinutes = _idpInactivityMinutes;
|
|
83
88
|
}
|
|
89
|
+
//Subscribe to UI events for user activity
|
|
90
|
+
if (_useUIEventsForActivity) {
|
|
91
|
+
this.subscribeToUIActivity();
|
|
92
|
+
}
|
|
84
93
|
this.hasValidConfig();
|
|
85
94
|
//There could be a non-expired token in local storage.
|
|
86
95
|
let token = this.authenticationProvider.authToken;
|
|
@@ -150,6 +159,9 @@ export class AuthenticationService {
|
|
|
150
159
|
this.logout(true);
|
|
151
160
|
});
|
|
152
161
|
}
|
|
162
|
+
makeKeepAliveRequest() {
|
|
163
|
+
this._http.get(this.keepAliveLocation(), { withCredentials: true }).subscribe();
|
|
164
|
+
}
|
|
153
165
|
/**
|
|
154
166
|
* Verifies whether or not a current user session exists.
|
|
155
167
|
*
|
|
@@ -174,6 +186,14 @@ export class AuthenticationService {
|
|
|
174
186
|
return this._tokenEndpoint;
|
|
175
187
|
}
|
|
176
188
|
}
|
|
189
|
+
keepAliveLocation() {
|
|
190
|
+
if (this._serverUrl) {
|
|
191
|
+
return this._serverUrl + this._keepAliveEndpoint;
|
|
192
|
+
}
|
|
193
|
+
else {
|
|
194
|
+
return this._keepAliveEndpoint;
|
|
195
|
+
}
|
|
196
|
+
}
|
|
177
197
|
directLoginLocation() {
|
|
178
198
|
if (this._serverUrl) {
|
|
179
199
|
return this._serverUrl + this._directEndpoint;
|
|
@@ -280,6 +300,14 @@ export class AuthenticationService {
|
|
|
280
300
|
validateToken(token) {
|
|
281
301
|
return (token && !this._jwtHelper.isTokenExpired(token));
|
|
282
302
|
}
|
|
303
|
+
subscribeToUIActivity() {
|
|
304
|
+
this._uiEvents.forEach(event => fromEvent(document, event).subscribe(_ => {
|
|
305
|
+
//Only update activity on UI event if not already about to time out. Counter intuitive, but in this case we should have 60 second count-down dialog open.
|
|
306
|
+
if (this._userIsAboutToTimeOut.getValue() !== true) {
|
|
307
|
+
this.updateUserActivity();
|
|
308
|
+
}
|
|
309
|
+
}));
|
|
310
|
+
}
|
|
283
311
|
subscribeToTokenRefresh(token) {
|
|
284
312
|
let exp = this._jwtHelper.getTokenExpirationDate(token);
|
|
285
313
|
// Use a timer to periodically check timeouts
|
|
@@ -319,7 +347,12 @@ export class AuthenticationService {
|
|
|
319
347
|
refreshTokenIfUserIsActive() {
|
|
320
348
|
//Only refresh if the user has been active
|
|
321
349
|
if (this._lastUserInteraction && ((new Date().valueOf() - this._lastUserInteraction.valueOf()) <= (this._maxInactivityMinutes * 60 * 1000))) {
|
|
350
|
+
//Refresh the JWT
|
|
322
351
|
this.requestAccessToken(false);
|
|
352
|
+
//Might as well make application keep alive requests at the same time as refreshing tokens - more efficient & probably needs same logic
|
|
353
|
+
if (this._keepAliveEndpoint) {
|
|
354
|
+
this.makeKeepAliveRequest();
|
|
355
|
+
}
|
|
323
356
|
}
|
|
324
357
|
}
|
|
325
358
|
hasValidConfig() {
|
|
@@ -346,7 +379,7 @@ AuthenticationService.SEC_GOV_CLASS_HEADER = "SecurityGovernorClass";
|
|
|
346
379
|
AuthenticationService.SEC_GOV_ID_HEADER = "SecurityGovernorId";
|
|
347
380
|
AuthenticationService.DEIDENT_HEADER = "DeidentifiedContext";
|
|
348
381
|
AuthenticationService.LIMITED_HEADER = "LimitedContext";
|
|
349
|
-
AuthenticationService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: AuthenticationService, deps: [{ token: i1.HttpClient }, { token: i2.Router }, { token: i3.CoolLocalStorage }, { token: i4.JwtHelperService }, { token: i5.AuthenticationProvider }, { token: AUTHENTICATION_ROUTE }, { token: AUTHENTICATION_LOGOUT_PATH }, { token: AUTHENTICATION_TOKEN_ENDPOINT }, { token: AUTHENTICATION_SERVER_URL, optional: true }, { token: AUTHENTICATION_DIRECT_ENDPOINT, optional: true }, { token: AUTHENTICATION_MAX_INACTIVITY_MINUTES, optional: true }, { token: AUTHENTICATION_USER_COUNTDOWN_SECONDS, optional: true }, { token: AUTHENTICATION_IDP_INACTIVITY_MINUTES, optional: true }, { token: LocationStrategy, optional: true }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
382
|
+
AuthenticationService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: AuthenticationService, deps: [{ token: i1.HttpClient }, { token: i2.Router }, { token: i3.CoolLocalStorage }, { token: i4.JwtHelperService }, { token: i5.AuthenticationProvider }, { token: AUTHENTICATION_ROUTE }, { token: AUTHENTICATION_LOGOUT_PATH }, { token: AUTHENTICATION_TOKEN_ENDPOINT }, { token: AUTHENTICATION_SERVER_URL, optional: true }, { token: AUTHENTICATION_DIRECT_ENDPOINT, optional: true }, { token: AUTHENTICATION_MAX_INACTIVITY_MINUTES, optional: true }, { token: AUTHENTICATION_USER_COUNTDOWN_SECONDS, optional: true }, { token: AUTHENTICATION_IDP_INACTIVITY_MINUTES, optional: true }, { token: AUTHENTICATION_USE_UI_EVENTS_FOR_ACTIVITY, optional: true }, { token: AUTHENTICATION_APP_KEEP_ALIVE_ENDPOINT, optional: true }, { token: LocationStrategy, optional: true }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
350
383
|
AuthenticationService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: AuthenticationService });
|
|
351
384
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: AuthenticationService, decorators: [{
|
|
352
385
|
type: Injectable
|
|
@@ -384,10 +417,20 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
|
|
|
384
417
|
}, {
|
|
385
418
|
type: Inject,
|
|
386
419
|
args: [AUTHENTICATION_IDP_INACTIVITY_MINUTES]
|
|
420
|
+
}] }, { type: undefined, decorators: [{
|
|
421
|
+
type: Optional
|
|
422
|
+
}, {
|
|
423
|
+
type: Inject,
|
|
424
|
+
args: [AUTHENTICATION_USE_UI_EVENTS_FOR_ACTIVITY]
|
|
425
|
+
}] }, { type: undefined, decorators: [{
|
|
426
|
+
type: Optional
|
|
427
|
+
}, {
|
|
428
|
+
type: Inject,
|
|
429
|
+
args: [AUTHENTICATION_APP_KEEP_ALIVE_ENDPOINT]
|
|
387
430
|
}] }, { type: i6.LocationStrategy, decorators: [{
|
|
388
431
|
type: Optional
|
|
389
432
|
}, {
|
|
390
433
|
type: Inject,
|
|
391
434
|
args: [LocationStrategy]
|
|
392
435
|
}] }]; } });
|
|
393
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"authentication.service.js","sourceRoot":"","sources":["../../../projects/authentication/src/authentication.service.ts"],"names":[],"mappings":"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","sourcesContent":["/*\r\n * Copyright (c) 2016 Huntsman Cancer Institute at the University of Utah, Confidential and Proprietary\r\n */\r\nimport {Injectable, InjectionToken, Inject, Optional, isDevMode} from \"@angular/core\";\r\nimport {LocationStrategy} from \"@angular/common\";\r\nimport {Router} from \"@angular/router\";\r\nimport {HttpClient, HttpHeaders, HttpRequest, HttpResponse} from \"@angular/common/http\";\r\n\r\nimport {interval, Observable, BehaviorSubject, Subscription, throwError, of} from \"rxjs\";\r\nimport {catchError, first, map} from \"rxjs/operators\";\r\nimport {JwtHelperService} from \"@auth0/angular-jwt\";\r\n\r\nimport {AuthenticationProvider} from \"./authentication.provider\";\r\nimport { CoolLocalStorage } from \"@angular-cool/storage\";\r\n\r\n/**\r\n * The token used for injection of the server side endpoint for the currently authenticated subject.\r\n *\r\n * @type {InjectionToken}\r\n */\r\nexport let AUTHENTICATION_SERVER_URL = new InjectionToken<string>(\"authentication_server_rest_api\");\r\nexport let AUTHENTICATION_LOGOUT_PATH = new InjectionToken<string>(\"authentication_logout_path\");\r\nexport let AUTHENTICATION_DIRECT_ENDPOINT = new InjectionToken<string>(\"authentication_direct_endpoint\");\r\nexport let AUTHENTICATION_TOKEN_ENDPOINT = new InjectionToken<string>(\"authentication_token_endpoint\");\r\nexport let AUTHENTICATION_ROUTE = new InjectionToken<string>(\"authentication_route\");\r\nexport let AUTHENTICATION_MAX_INACTIVITY_MINUTES = new InjectionToken<number>(\"authentication_max_inactivity\");\r\nexport let AUTHENTICATION_USER_COUNTDOWN_SECONDS = new InjectionToken<number>(\"authentication_user_countdown_seconds\");\r\nexport let AUTHENTICATION_IDP_INACTIVITY_MINUTES = new InjectionToken<number>(\"authentication_idp_inactivity_minutes\");\r\n\r\n/**\r\n * @since 1.0.0\r\n */\r\n@Injectable()\r\nexport class AuthenticationService {\r\n\r\n  /**\r\n   * The generic error message used when a server error is thrown without a status.\r\n   *\r\n   * @type {string}\r\n   */\r\n  public static GENERIC_ERR_MSG: string = \"Server error\";\r\n\r\n  private static CONTENT_TYPE: string = \"Content-Type\";\r\n  private static SEC_GOV_CLASS_HEADER: string = \"SecurityGovernorClass\";\r\n  private static SEC_GOV_ID_HEADER: string = \"SecurityGovernorId\";\r\n  private static DEIDENT_HEADER: string = \"DeidentifiedContext\";\r\n  private static LIMITED_HEADER: string = \"LimitedContext\";\r\n\r\n  public userCountdownSeconds: number = 60;\r\n  public idpInactivityMinutes: number = 5;\r\n\r\n  public contentType: string = \"application/json\";\r\n  public securityGovernorClass: string;\r\n  public securityGovernorId: number;\r\n  public limitedContext: boolean = false;\r\n  public deidentifiedContext: boolean = false;\r\n\r\n  private maxViewPermission: BehaviorSubject<\"view\" | \"viewident\" | \"viewlimited\"> = new BehaviorSubject<\"view\" | \"viewident\" | \"viewlimited\">(\"viewident\");\r\n  private _isAuthenticatedSubject: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\r\n  private _userIsAboutToTimeOut: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\r\n  private _redirectUrl: string;\r\n  private _refreshSubscription: Subscription;\r\n  private _lastUserInteraction: Date;\r\n  private _maxInactivityMinutes: number = 120;\r\n\r\n  private baseUrl: string;\r\n  private contextRoot: string = \"\";\r\n\r\n  constructor(private _http: HttpClient,\r\n              private _router: Router,\r\n              private _localStorageService: CoolLocalStorage,\r\n              private _jwtHelper: JwtHelperService,\r\n              private authenticationProvider: AuthenticationProvider,\r\n              @Inject(AUTHENTICATION_ROUTE) private _authenticationRoute: string,\r\n              @Inject(AUTHENTICATION_LOGOUT_PATH) private _logoutPath: string,\r\n              @Inject(AUTHENTICATION_TOKEN_ENDPOINT) private _tokenEndpoint: string,\r\n              @Optional() @Inject(AUTHENTICATION_SERVER_URL) private _serverUrl: string,\r\n              @Optional() @Inject(AUTHENTICATION_DIRECT_ENDPOINT) private _directEndpoint: string,\r\n              @Optional() @Inject(AUTHENTICATION_MAX_INACTIVITY_MINUTES) private _maxInactivity: number,\r\n              @Optional() @Inject(AUTHENTICATION_USER_COUNTDOWN_SECONDS) private _userCountdownSeconds: number,\r\n              @Optional() @Inject(AUTHENTICATION_IDP_INACTIVITY_MINUTES) private _idpInactivityMinutes: number,\r\n              @Optional() @Inject(LocationStrategy) private locationStrategy: LocationStrategy) {\r\n    if (isDevMode()) {\r\n      console.debug(\"window.location.href: \" + window.location.href);\r\n    }\r\n\r\n    if (window.location) {\r\n      let parts: string[] = window.location.href.split(\"/\");\r\n      this.baseUrl = parts[0] + \"//\" + parts[2];\r\n      if (parts.length > 3) {\r\n        this.contextRoot = parts[3];\r\n      }\r\n    }\r\n\r\n    if (this._localStorageService.getItem(\"maxViewPermission\")) {\r\n      this.maxViewPermission.next(<\"view\" | \"viewident\" | \"viewlimited\">this._localStorageService.getItem(\"maxViewPermission\"));\r\n    }\r\n\r\n    if (_maxInactivity) {\r\n      this._maxInactivityMinutes = _maxInactivity;\r\n    }\r\n\r\n    if (_userCountdownSeconds) {\r\n      this.userCountdownSeconds = _userCountdownSeconds;\r\n    }\r\n\r\n    if (_idpInactivityMinutes) {\r\n      this.idpInactivityMinutes = _idpInactivityMinutes;\r\n    }\r\n\r\n    this.hasValidConfig();\r\n\r\n    //There could be a non-expired token in local storage.\r\n    let token: string = this.authenticationProvider.authToken;\r\n    this.storeToken(token);\r\n  }\r\n\r\n  getBaseUrl(): string {\r\n    return (this.baseUrl) ? this.baseUrl : \"\";\r\n  }\r\n\r\n  getContextRoot(): string {\r\n    return this.contextRoot;\r\n  }\r\n\r\n  getHeaders(req: HttpRequest<any>): HttpHeaders {\r\n    let headers: HttpHeaders = req.headers;\r\n\r\n    //Don't set content type if already set\r\n    if (!req.headers.get(AuthenticationService.CONTENT_TYPE)) {\r\n      headers = headers.set(AuthenticationService.CONTENT_TYPE, this.contentType.toString());\r\n    }\r\n\r\n    if (headers.get(AuthenticationService.SEC_GOV_CLASS_HEADER) === \"\") {\r\n      headers = headers.delete(AuthenticationService.SEC_GOV_CLASS_HEADER);\r\n    } else if (this.securityGovernorClass && !headers.get(AuthenticationService.SEC_GOV_CLASS_HEADER)) {\r\n      headers = headers.set(AuthenticationService.SEC_GOV_CLASS_HEADER, this.securityGovernorClass);\r\n    }\r\n\r\n    if (headers.get(AuthenticationService.SEC_GOV_ID_HEADER) === \"\") {\r\n      headers = headers.delete(AuthenticationService.SEC_GOV_ID_HEADER);\r\n    } else if (this.securityGovernorId && !headers.get(AuthenticationService.SEC_GOV_ID_HEADER)) {\r\n      headers = headers.set(AuthenticationService.SEC_GOV_ID_HEADER, this.securityGovernorId.toString());\r\n    }\r\n\r\n    headers = headers.set(AuthenticationService.DEIDENT_HEADER, this.deidentifiedContext.toString());\r\n    headers = headers.set(AuthenticationService.LIMITED_HEADER, this.limitedContext.toString());\r\n\r\n    return headers;\r\n  }\r\n\r\n  get authenticationTokenKey(): string {\r\n    return this.authenticationProvider.authenticationTokenKey;\r\n  }\r\n\r\n  get authToken(): string {\r\n    return this.authenticationProvider.authToken;\r\n  }\r\n\r\n  public updateUserActivity(): void {\r\n    if (this._isAuthenticatedSubject.value) {\r\n      this._lastUserInteraction = new Date();\r\n      this._userIsAboutToTimeOut.next(false);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * A mutator for identifying the clients original request location. Setting this value will influence the end location\r\n   * navigated to by {@link #navigateToPath}.\r\n   *\r\n   * @param redirectUrl location of the users request before authentication\r\n   */\r\n  set redirectUrl(redirectUrl: string) {\r\n    this._redirectUrl = redirectUrl;\r\n  }\r\n\r\n  get redirectUrl() {\r\n    return this._redirectUrl;\r\n  }\r\n\r\n  requestAccessToken(redirectOnSuccess: boolean): void {\r\n\r\n    this._http.get(this.tokenLocation(), {withCredentials: true})\r\n      .subscribe(\r\n        (response: any) => {\r\n          this.storeToken(response.auth_token);\r\n          if (redirectOnSuccess) {\r\n            this.proceedIfAuthenticated();\r\n          }\r\n        },\r\n        (error) => {\r\n          //Token refresh failed.\r\n          this.logout(true);\r\n        }\r\n      );\r\n  }\r\n\r\n  /**\r\n   * Verifies whether or not a current user session exists.\r\n   *\r\n   * @returns {Observable<boolean>} evaluates to true if the user is authenticated, false otherwise.\r\n   */\r\n  isAuthenticated(): Observable<boolean> {\r\n    return this._isAuthenticatedSubject.asObservable();\r\n  }\r\n\r\n  isAboutToTimeOut(): Observable<boolean> {\r\n    return this._userIsAboutToTimeOut.asObservable();\r\n  }\r\n\r\n  getTimeoutStart(): number {\r\n    if (this._lastUserInteraction) {\r\n      return this._lastUserInteraction.valueOf() + (((this._maxInactivityMinutes * 60) - this.userCountdownSeconds) * 1000);\r\n    }\r\n  }\r\n\r\n  tokenLocation(): string {\r\n    if (this._serverUrl) {\r\n      return this._serverUrl + this._tokenEndpoint;\r\n    } else {\r\n      return this._tokenEndpoint;\r\n    }\r\n  }\r\n\r\n  directLoginLocation(): string {\r\n    if (this._serverUrl) {\r\n      return this._serverUrl + this._directEndpoint;\r\n    } else {\r\n      return this._directEndpoint;\r\n    }\r\n  }\r\n\r\n  logoutLocation(): string {\r\n    if (this._serverUrl) {\r\n      return this._serverUrl + this._logoutPath;\r\n    } else {\r\n      return this._logoutPath;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * A function to authenticated the user with the provided credentials. Failure results in an error that describes the\r\n   * server response (status and status message) and should be actionable by the client application.\r\n   *\r\n   * @param username of the authenticating user to verify\r\n   * @param password of the authenticating user to verify\r\n   * @returns {Observable<R>} describing the result of the login action, true or an error\r\n   */\r\n  login(_username: string, _password: string): Observable<boolean> {\r\n    return this._http.post(\r\n      this.directLoginLocation(),\r\n      {username: _username, password: _password},\r\n      {observe: \"response\"}\r\n    ).pipe(map((resp: HttpResponse<any>) => {\r\n      if (resp.status === 201) {\r\n        return true;\r\n      } else {\r\n        throw new Error(\"Authentication failed. \" + resp.status + \": \" + resp.statusText);\r\n      }\r\n    }),\r\n      catchError(this.handleError));\r\n  }\r\n\r\n\r\n  clearLogin(): Observable<Response> {\r\n    //Front-end logout\r\n    try {\r\n      this._localStorageService.removeItem(this.authenticationProvider.authenticationTokenKey);\r\n      this.unsubscribeFromTokenRefresh();\r\n      this._isAuthenticatedSubject.next(false);\r\n      this._userIsAboutToTimeOut.next(false);\r\n    } catch (Error) {\r\n    }\r\n\r\n    //Back-end logout\r\n    let headers = new HttpHeaders().set(AuthenticationService.CONTENT_TYPE, \"text/plain\");\r\n    return <Observable<Response>>this._http.get(this.logoutLocation(), {headers: headers});\r\n  }\r\n\r\n  /**\r\n   * A function to signal the termination of the current session. Invoking this function will clean up any relevant state\r\n   * related to the last active session.\r\n   */\r\n  logout(keepCurrentRoute: boolean = false): void {\r\n    //Prevent logout if already on authentication route. Doing otherwise screws up SAML\r\n    if (! this._router.routerState || this._router.routerState.snapshot.url !== this._authenticationRoute) {\r\n      this._redirectUrl = (keepCurrentRoute && this._router.routerState && this._router.routerState.snapshot) ? this._router.routerState.snapshot.url : \"\";\r\n\r\n      if (this._redirectUrl.startsWith(\"/\")) {\r\n        this._redirectUrl = this._redirectUrl.substring(1);\r\n      }\r\n\r\n      this.clearLogin().subscribe(\r\n        (response) => {\r\n          window.location.replace(this._redirectUrl);\r\n          },\r\n        (error) => {\r\n          window.location.replace(this._redirectUrl);\r\n        }\r\n      );\r\n    }\r\n  }\r\n\r\n  storeToken(token: string): void {\r\n    let valid = this.validateToken(token);\r\n\r\n    // unsubscribe from refesh before we decide wether to resubscribe\r\n    this.unsubscribeFromTokenRefresh();\r\n\r\n    if (valid) {\r\n      this._localStorageService.setItem(this.authenticationProvider.authenticationTokenKey, token);\r\n      this.subscribeToTokenRefresh(token);\r\n\r\n      //Change the BehaviorSubject if the user was not previously authenticated.\r\n      //Since other code may be subscribing to this observable, we don't want to cause new events to fire if just refreshing the JWT.\r\n      if (! this._isAuthenticatedSubject.value) {\r\n        this._isAuthenticatedSubject.next(true);\r\n      }\r\n    } else {\r\n      this._localStorageService.removeItem(this.authenticationProvider.authenticationTokenKey);\r\n      this._isAuthenticatedSubject.next(false);\r\n    }\r\n  }\r\n\r\n  proceedIfAuthenticated(): boolean {\r\n    if (isDevMode()) {\r\n      console.debug(\"AuthenticationService.proceedIfAuthenticated: \" + this._redirectUrl);\r\n    }\r\n\r\n    if (this._isAuthenticatedSubject.value) {\r\n      //Login counts as user activity, too\r\n      this.updateUserActivity();\r\n\r\n      if (this._redirectUrl && this._redirectUrl && this._redirectUrl !== \"\") {\r\n        this._router.navigateByUrl(this._redirectUrl);\r\n      } else {\r\n        this._router.navigate([\"\"]);\r\n      }\r\n\r\n      return true;\r\n    } else {\r\n      return false;\r\n    }\r\n  }\r\n\r\n  validateToken(token: string): boolean {\r\n    return (token && !this._jwtHelper.isTokenExpired(token));\r\n  }\r\n\r\n  subscribeToTokenRefresh(token: any): void {\r\n    let exp = this._jwtHelper.getTokenExpirationDate(token);\r\n\r\n    // Use a timer to periodically check timeouts\r\n    this._refreshSubscription = interval(1000)\r\n      .subscribe(() => {\r\n\r\n        // If a tab is inactive we can't know if our timer is accurate\r\n        // so when the interval hits check against timestamps\r\n        if (this._isAuthenticatedSubject.value && Date.now() > this.getTimeoutStart()) {\r\n          //Don't update the subject more than once! Doing so initializes more than one countdown timer!\r\n          if (this._userIsAboutToTimeOut.getValue() !== true) {\r\n            this._userIsAboutToTimeOut.next(true);\r\n          }\r\n        }\r\n\r\n        // check for refresh token\r\n        let msToExpiry = (exp.valueOf() - new Date().valueOf());\r\n\r\n        // Refresh 60 seconds before expiry\r\n        if (msToExpiry <= 60000) {\r\n          this.refreshTokenIfUserIsActive();\r\n        }\r\n      });\r\n  }\r\n\r\n  unsubscribeFromTokenRefresh(): void {\r\n    if (this._refreshSubscription && ! this._refreshSubscription.closed) {\r\n      this._refreshSubscription.unsubscribe();\r\n    }\r\n  }\r\n\r\n  getMaxViewPermission(): \"view\" | \"viewident\" | \"viewlimited\" {\r\n    return this.maxViewPermission.getValue();\r\n  }\r\n\r\n  getMaxViewPermissionSubject(): BehaviorSubject<\"view\" | \"viewident\" | \"viewlimited\"> {\r\n    return this.maxViewPermission;\r\n  }\r\n\r\n  setMaxViewPermission(maxViewPermission: \"view\" | \"viewident\" | \"viewlimited\"): void {\r\n    this._localStorageService.setItem(\"maxViewPermission\", maxViewPermission);\r\n    this.maxViewPermission.next(maxViewPermission);\r\n  }\r\n\r\n  private refreshTokenIfUserIsActive(): void {\r\n    //Only refresh if the user has been active\r\n    if (this._lastUserInteraction && ((new Date().valueOf() - this._lastUserInteraction.valueOf()) <= (this._maxInactivityMinutes * 60 * 1000))) {\r\n      this.requestAccessToken(false);\r\n    }\r\n  }\r\n\r\n  private hasValidConfig(): void {\r\n    if (this._tokenEndpoint == null && (this._serverUrl === null || this._logoutPath === null)) {\r\n      throw new Error(\"BUG ALERT! Invalid AuthenticationService configuration. No valid configuration for authentication endpoint(s).\");\r\n    }\r\n    if (this._localStorageService === null || this.authenticationProvider.authenticationTokenKey === null) {\r\n      throw new Error(\"BUG ALERT! Invalid AuthenticationService configuration. No valid configuration for local storage\");\r\n    }\r\n  }\r\n\r\n  private handleError(error: any) : Observable<never> {\r\n    let errMsg = (error.message) ? error.message : AuthenticationService.GENERIC_ERR_MSG;\r\n    return throwError(errMsg);\r\n  }\r\n}\r\n"]}
|
|
436
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"authentication.service.js","sourceRoot":"","sources":["../../../projects/authentication/src/authentication.service.ts"],"names":[],"mappings":"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","sourcesContent":["/*\r\n * Copyright (c) 2016 Huntsman Cancer Institute at the University of Utah, Confidential and Proprietary\r\n */\r\nimport {Injectable, InjectionToken, Inject, Optional, isDevMode} from \"@angular/core\";\r\nimport {LocationStrategy} from \"@angular/common\";\r\nimport {Router} from \"@angular/router\";\r\nimport {HttpClient, HttpHeaders, HttpRequest, HttpResponse} from \"@angular/common/http\";\r\n\r\nimport {interval, Observable, BehaviorSubject, Subscription, throwError, of, fromEvent} from \"rxjs\";\r\nimport {catchError, first, map} from \"rxjs/operators\";\r\nimport {JwtHelperService} from \"@auth0/angular-jwt\";\r\n\r\nimport {AuthenticationProvider} from \"./authentication.provider\";\r\nimport { CoolLocalStorage } from \"@angular-cool/storage\";\r\n\r\n/**\r\n * The token used for injection of the server side endpoint for the currently authenticated subject.\r\n *\r\n * @type {InjectionToken}\r\n */\r\nexport let AUTHENTICATION_SERVER_URL = new InjectionToken<string>(\"authentication_server_rest_api\");\r\nexport let AUTHENTICATION_LOGOUT_PATH = new InjectionToken<string>(\"authentication_logout_path\");\r\nexport let AUTHENTICATION_DIRECT_ENDPOINT = new InjectionToken<string>(\"authentication_direct_endpoint\");\r\nexport let AUTHENTICATION_TOKEN_ENDPOINT = new InjectionToken<string>(\"authentication_token_endpoint\");\r\nexport let AUTHENTICATION_ROUTE = new InjectionToken<string>(\"authentication_route\");\r\nexport let AUTHENTICATION_MAX_INACTIVITY_MINUTES = new InjectionToken<number>(\"authentication_max_inactivity\");\r\nexport let AUTHENTICATION_USER_COUNTDOWN_SECONDS = new InjectionToken<number>(\"authentication_user_countdown_seconds\");\r\nexport let AUTHENTICATION_IDP_INACTIVITY_MINUTES = new InjectionToken<number>(\"authentication_idp_inactivity_minutes\");\r\nexport let AUTHENTICATION_USE_UI_EVENTS_FOR_ACTIVITY = new InjectionToken<boolean>(\"authentication_use_ui_events_for_activity\");\r\nexport let AUTHENTICATION_APP_KEEP_ALIVE_ENDPOINT = new InjectionToken<string>(\"authentication_app_keep_alive_endpoint\");\r\n\r\n/**\r\n * @since 1.0.0\r\n */\r\n@Injectable()\r\nexport class AuthenticationService {\r\n\r\n  /**\r\n   * The generic error message used when a server error is thrown without a status.\r\n   *\r\n   * @type {string}\r\n   */\r\n  public static GENERIC_ERR_MSG: string = \"Server error\";\r\n\r\n  private static CONTENT_TYPE: string = \"Content-Type\";\r\n  private static SEC_GOV_CLASS_HEADER: string = \"SecurityGovernorClass\";\r\n  private static SEC_GOV_ID_HEADER: string = \"SecurityGovernorId\";\r\n  private static DEIDENT_HEADER: string = \"DeidentifiedContext\";\r\n  private static LIMITED_HEADER: string = \"LimitedContext\";\r\n\r\n  public userCountdownSeconds: number = 60;\r\n  public idpInactivityMinutes: number = 5;\r\n\r\n  public contentType: string = \"application/json\";\r\n  public securityGovernorClass: string;\r\n  public securityGovernorId: number;\r\n  public limitedContext: boolean = false;\r\n  public deidentifiedContext: boolean = false;\r\n\r\n  private maxViewPermission: BehaviorSubject<\"view\" | \"viewident\" | \"viewlimited\"> = new BehaviorSubject<\"view\" | \"viewident\" | \"viewlimited\">(\"viewident\");\r\n  private _isAuthenticatedSubject: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\r\n  private _userIsAboutToTimeOut: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\r\n  private _redirectUrl: string;\r\n  private _refreshSubscription: Subscription;\r\n  private _lastUserInteraction: Date;\r\n  private _maxInactivityMinutes: number = 120;\r\n  private _uiEvents: string[] = ['keydown', 'click', 'wheel', 'mousemove'];\r\n\r\n  private baseUrl: string;\r\n  private contextRoot: string = \"\";\r\n\r\n  constructor(private _http: HttpClient,\r\n              private _router: Router,\r\n              private _localStorageService: CoolLocalStorage,\r\n              private _jwtHelper: JwtHelperService,\r\n              private authenticationProvider: AuthenticationProvider,\r\n              @Inject(AUTHENTICATION_ROUTE) private _authenticationRoute: string,\r\n              @Inject(AUTHENTICATION_LOGOUT_PATH) private _logoutPath: string,\r\n              @Inject(AUTHENTICATION_TOKEN_ENDPOINT) private _tokenEndpoint: string,\r\n              @Optional() @Inject(AUTHENTICATION_SERVER_URL) private _serverUrl: string,\r\n              @Optional() @Inject(AUTHENTICATION_DIRECT_ENDPOINT) private _directEndpoint: string,\r\n              @Optional() @Inject(AUTHENTICATION_MAX_INACTIVITY_MINUTES) private _maxInactivity: number,\r\n              @Optional() @Inject(AUTHENTICATION_USER_COUNTDOWN_SECONDS) private _userCountdownSeconds: number,\r\n              @Optional() @Inject(AUTHENTICATION_IDP_INACTIVITY_MINUTES) private _idpInactivityMinutes: number,\r\n              @Optional() @Inject(AUTHENTICATION_USE_UI_EVENTS_FOR_ACTIVITY) private _useUIEventsForActivity: boolean,\r\n              @Optional() @Inject(AUTHENTICATION_APP_KEEP_ALIVE_ENDPOINT) private _keepAliveEndpoint: string,\r\n              @Optional() @Inject(LocationStrategy) private locationStrategy: LocationStrategy) {\r\n    if (isDevMode()) {\r\n      console.debug(\"window.location.href: \" + window.location.href);\r\n    }\r\n\r\n    if (window.location) {\r\n      let parts: string[] = window.location.href.split(\"/\");\r\n      this.baseUrl = parts[0] + \"//\" + parts[2];\r\n      if (parts.length > 3) {\r\n        this.contextRoot = parts[3];\r\n      }\r\n    }\r\n\r\n    if (this._localStorageService.getItem(\"maxViewPermission\")) {\r\n      this.maxViewPermission.next(<\"view\" | \"viewident\" | \"viewlimited\">this._localStorageService.getItem(\"maxViewPermission\"));\r\n    }\r\n\r\n    if (_maxInactivity) {\r\n      this._maxInactivityMinutes = _maxInactivity;\r\n    }\r\n\r\n    if (_userCountdownSeconds) {\r\n      this.userCountdownSeconds = _userCountdownSeconds;\r\n    }\r\n\r\n    if (_idpInactivityMinutes) {\r\n      this.idpInactivityMinutes = _idpInactivityMinutes;\r\n    }\r\n    \r\n    //Subscribe to UI events for user activity\r\n    if (_useUIEventsForActivity) {\r\n      this.subscribeToUIActivity();\r\n    }\r\n\r\n    this.hasValidConfig();\r\n\r\n    //There could be a non-expired token in local storage.\r\n    let token: string = this.authenticationProvider.authToken;\r\n    this.storeToken(token);\r\n  }\r\n\r\n  getBaseUrl(): string {\r\n    return (this.baseUrl) ? this.baseUrl : \"\";\r\n  }\r\n\r\n  getContextRoot(): string {\r\n    return this.contextRoot;\r\n  }\r\n\r\n  getHeaders(req: HttpRequest<any>): HttpHeaders {\r\n    let headers: HttpHeaders = req.headers;\r\n\r\n    //Don't set content type if already set\r\n    if (!req.headers.get(AuthenticationService.CONTENT_TYPE)) {\r\n      headers = headers.set(AuthenticationService.CONTENT_TYPE, this.contentType.toString());\r\n    }\r\n\r\n    if (headers.get(AuthenticationService.SEC_GOV_CLASS_HEADER) === \"\") {\r\n      headers = headers.delete(AuthenticationService.SEC_GOV_CLASS_HEADER);\r\n    } else if (this.securityGovernorClass && !headers.get(AuthenticationService.SEC_GOV_CLASS_HEADER)) {\r\n      headers = headers.set(AuthenticationService.SEC_GOV_CLASS_HEADER, this.securityGovernorClass);\r\n    }\r\n\r\n    if (headers.get(AuthenticationService.SEC_GOV_ID_HEADER) === \"\") {\r\n      headers = headers.delete(AuthenticationService.SEC_GOV_ID_HEADER);\r\n    } else if (this.securityGovernorId && !headers.get(AuthenticationService.SEC_GOV_ID_HEADER)) {\r\n      headers = headers.set(AuthenticationService.SEC_GOV_ID_HEADER, this.securityGovernorId.toString());\r\n    }\r\n\r\n    headers = headers.set(AuthenticationService.DEIDENT_HEADER, this.deidentifiedContext.toString());\r\n    headers = headers.set(AuthenticationService.LIMITED_HEADER, this.limitedContext.toString());\r\n\r\n    return headers;\r\n  }\r\n\r\n  get authenticationTokenKey(): string {\r\n    return this.authenticationProvider.authenticationTokenKey;\r\n  }\r\n\r\n  get authToken(): string {\r\n    return this.authenticationProvider.authToken;\r\n  }\r\n\r\n  public updateUserActivity(): void {\r\n    if (this._isAuthenticatedSubject.value) {\r\n      this._lastUserInteraction = new Date();\r\n      this._userIsAboutToTimeOut.next(false);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * A mutator for identifying the clients original request location. Setting this value will influence the end location\r\n   * navigated to by {@link #navigateToPath}.\r\n   *\r\n   * @param redirectUrl location of the users request before authentication\r\n   */\r\n  set redirectUrl(redirectUrl: string) {\r\n    this._redirectUrl = redirectUrl;\r\n  }\r\n\r\n  get redirectUrl() {\r\n    return this._redirectUrl;\r\n  }\r\n\r\n  requestAccessToken(redirectOnSuccess: boolean): void {\r\n\r\n    this._http.get(this.tokenLocation(), {withCredentials: true})\r\n      .subscribe(\r\n        (response: any) => {\r\n          this.storeToken(response.auth_token);\r\n          if (redirectOnSuccess) {\r\n            this.proceedIfAuthenticated();\r\n          }\r\n        },\r\n        (error) => {\r\n          //Token refresh failed.\r\n          this.logout(true);\r\n        }\r\n      );\r\n  }\r\n  \r\n  makeKeepAliveRequest(): void {\r\n    this._http.get(this.keepAliveLocation(), {withCredentials: true}).subscribe();\r\n  }\r\n\r\n  /**\r\n   * Verifies whether or not a current user session exists.\r\n   *\r\n   * @returns {Observable<boolean>} evaluates to true if the user is authenticated, false otherwise.\r\n   */\r\n  isAuthenticated(): Observable<boolean> {\r\n    return this._isAuthenticatedSubject.asObservable();\r\n  }\r\n\r\n  isAboutToTimeOut(): Observable<boolean> {\r\n    return this._userIsAboutToTimeOut.asObservable();\r\n  }\r\n\r\n  getTimeoutStart(): number {\r\n    if (this._lastUserInteraction) {\r\n      return this._lastUserInteraction.valueOf() + (((this._maxInactivityMinutes * 60) - this.userCountdownSeconds) * 1000);\r\n    }\r\n  }\r\n\r\n  tokenLocation(): string {\r\n    if (this._serverUrl) {\r\n      return this._serverUrl + this._tokenEndpoint;\r\n    } else {\r\n      return this._tokenEndpoint;\r\n    }\r\n  }\r\n  \r\n  keepAliveLocation(): string {\r\n    if (this._serverUrl) {\r\n      return this._serverUrl + this._keepAliveEndpoint;\r\n    } else {\r\n      return this._keepAliveEndpoint;\r\n    }\r\n  }\r\n\r\n  directLoginLocation(): string {\r\n    if (this._serverUrl) {\r\n      return this._serverUrl + this._directEndpoint;\r\n    } else {\r\n      return this._directEndpoint;\r\n    }\r\n  }\r\n\r\n  logoutLocation(): string {\r\n    if (this._serverUrl) {\r\n      return this._serverUrl + this._logoutPath;\r\n    } else {\r\n      return this._logoutPath;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * A function to authenticated the user with the provided credentials. Failure results in an error that describes the\r\n   * server response (status and status message) and should be actionable by the client application.\r\n   *\r\n   * @param username of the authenticating user to verify\r\n   * @param password of the authenticating user to verify\r\n   * @returns {Observable<R>} describing the result of the login action, true or an error\r\n   */\r\n  login(_username: string, _password: string): Observable<boolean> {\r\n    return this._http.post(\r\n      this.directLoginLocation(),\r\n      {username: _username, password: _password},\r\n      {observe: \"response\"}\r\n    ).pipe(map((resp: HttpResponse<any>) => {\r\n      if (resp.status === 201) {\r\n        return true;\r\n      } else {\r\n        throw new Error(\"Authentication failed. \" + resp.status + \": \" + resp.statusText);\r\n      }\r\n    }),\r\n      catchError(this.handleError));\r\n  }\r\n\r\n\r\n  clearLogin(): Observable<Response> {\r\n    //Front-end logout\r\n    try {\r\n      this._localStorageService.removeItem(this.authenticationProvider.authenticationTokenKey);\r\n      this.unsubscribeFromTokenRefresh();\r\n      this._isAuthenticatedSubject.next(false);\r\n      this._userIsAboutToTimeOut.next(false);\r\n    } catch (Error) {\r\n    }\r\n\r\n    //Back-end logout\r\n    let headers = new HttpHeaders().set(AuthenticationService.CONTENT_TYPE, \"text/plain\");\r\n    return <Observable<Response>>this._http.get(this.logoutLocation(), {headers: headers});\r\n  }\r\n\r\n  /**\r\n   * A function to signal the termination of the current session. Invoking this function will clean up any relevant state\r\n   * related to the last active session.\r\n   */\r\n  logout(keepCurrentRoute: boolean = false): void {\r\n    //Prevent logout if already on authentication route. Doing otherwise screws up SAML\r\n    if (! this._router.routerState || this._router.routerState.snapshot.url !== this._authenticationRoute) {\r\n      this._redirectUrl = (keepCurrentRoute && this._router.routerState && this._router.routerState.snapshot) ? this._router.routerState.snapshot.url : \"\";\r\n\r\n      if (this._redirectUrl.startsWith(\"/\")) {\r\n        this._redirectUrl = this._redirectUrl.substring(1);\r\n      }\r\n\r\n      this.clearLogin().subscribe(\r\n        (response) => {\r\n          window.location.replace(this._redirectUrl);\r\n          },\r\n        (error) => {\r\n          window.location.replace(this._redirectUrl);\r\n        }\r\n      );\r\n    }\r\n  }\r\n\r\n  storeToken(token: string): void {\r\n    let valid = this.validateToken(token);\r\n\r\n    // unsubscribe from refesh before we decide wether to resubscribe\r\n    this.unsubscribeFromTokenRefresh();\r\n\r\n    if (valid) {\r\n      this._localStorageService.setItem(this.authenticationProvider.authenticationTokenKey, token);\r\n      this.subscribeToTokenRefresh(token);\r\n\r\n      //Change the BehaviorSubject if the user was not previously authenticated.\r\n      //Since other code may be subscribing to this observable, we don't want to cause new events to fire if just refreshing the JWT.\r\n      if (! this._isAuthenticatedSubject.value) {\r\n        this._isAuthenticatedSubject.next(true);\r\n      }\r\n    } else {\r\n      this._localStorageService.removeItem(this.authenticationProvider.authenticationTokenKey);\r\n      this._isAuthenticatedSubject.next(false);\r\n    }\r\n  }\r\n\r\n  proceedIfAuthenticated(): boolean {\r\n    if (isDevMode()) {\r\n      console.debug(\"AuthenticationService.proceedIfAuthenticated: \" + this._redirectUrl);\r\n    }\r\n\r\n    if (this._isAuthenticatedSubject.value) {\r\n      //Login counts as user activity, too\r\n      this.updateUserActivity();\r\n\r\n      if (this._redirectUrl && this._redirectUrl && this._redirectUrl !== \"\") {\r\n        this._router.navigateByUrl(this._redirectUrl);\r\n      } else {\r\n        this._router.navigate([\"\"]);\r\n      }\r\n\r\n      return true;\r\n    } else {\r\n      return false;\r\n    }\r\n  }\r\n\r\n  validateToken(token: string): boolean {\r\n    return (token && !this._jwtHelper.isTokenExpired(token));\r\n  }\r\n  \r\n  subscribeToUIActivity(){\r\n    this._uiEvents.forEach(event =>\r\n      fromEvent(document, event).subscribe(_ => {\r\n        //Only update activity on UI event if not already about to time out. Counter intuitive, but in this case we should have 60 second count-down dialog open.\r\n        if (this._userIsAboutToTimeOut.getValue() !== true) {\r\n          this.updateUserActivity();\r\n        }\r\n      })\r\n    );    \r\n  }\r\n\r\n  subscribeToTokenRefresh(token: any): void {\r\n    let exp = this._jwtHelper.getTokenExpirationDate(token);\r\n\r\n    // Use a timer to periodically check timeouts\r\n    this._refreshSubscription = interval(1000)\r\n      .subscribe(() => {\r\n\r\n        // If a tab is inactive we can't know if our timer is accurate\r\n        // so when the interval hits check against timestamps\r\n        if (this._isAuthenticatedSubject.value && Date.now() > this.getTimeoutStart()) {\r\n          //Don't update the subject more than once! Doing so initializes more than one countdown timer!\r\n          if (this._userIsAboutToTimeOut.getValue() !== true) {\r\n            this._userIsAboutToTimeOut.next(true);\r\n          }\r\n        }\r\n\r\n        // check for refresh token\r\n        let msToExpiry = (exp.valueOf() - new Date().valueOf());\r\n\r\n        // Refresh 60 seconds before expiry\r\n        if (msToExpiry <= 60000) {\r\n          this.refreshTokenIfUserIsActive();\r\n        }\r\n      });\r\n  }\r\n\r\n  unsubscribeFromTokenRefresh(): void {\r\n    if (this._refreshSubscription && ! this._refreshSubscription.closed) {\r\n      this._refreshSubscription.unsubscribe();\r\n    }\r\n  }\r\n\r\n  getMaxViewPermission(): \"view\" | \"viewident\" | \"viewlimited\" {\r\n    return this.maxViewPermission.getValue();\r\n  }\r\n\r\n  getMaxViewPermissionSubject(): BehaviorSubject<\"view\" | \"viewident\" | \"viewlimited\"> {\r\n    return this.maxViewPermission;\r\n  }\r\n\r\n  setMaxViewPermission(maxViewPermission: \"view\" | \"viewident\" | \"viewlimited\"): void {\r\n    this._localStorageService.setItem(\"maxViewPermission\", maxViewPermission);\r\n    this.maxViewPermission.next(maxViewPermission);\r\n  }\r\n\r\n  private refreshTokenIfUserIsActive(): void {\r\n    //Only refresh if the user has been active\r\n    if (this._lastUserInteraction && ((new Date().valueOf() - this._lastUserInteraction.valueOf()) <= (this._maxInactivityMinutes * 60 * 1000))) {\r\n      //Refresh the JWT\r\n      this.requestAccessToken(false);\r\n      \r\n      //Might as well make application keep alive requests at the same time as refreshing tokens - more efficient & probably needs same logic\r\n      if (this._keepAliveEndpoint) {\r\n        this.makeKeepAliveRequest();\r\n      }\r\n    }\r\n  }\r\n\r\n  private hasValidConfig(): void {\r\n    if (this._tokenEndpoint == null && (this._serverUrl === null || this._logoutPath === null)) {\r\n      throw new Error(\"BUG ALERT! Invalid AuthenticationService configuration. No valid configuration for authentication endpoint(s).\");\r\n    }\r\n    if (this._localStorageService === null || this.authenticationProvider.authenticationTokenKey === null) {\r\n      throw new Error(\"BUG ALERT! Invalid AuthenticationService configuration. No valid configuration for local storage\");\r\n    }\r\n  }\r\n\r\n  private handleError(error: any) : Observable<never> {\r\n    let errMsg = (error.message) ? error.message : AuthenticationService.GENERIC_ERR_MSG;\r\n    return throwError(errMsg);\r\n  }\r\n}\r\n"]}
|
package/esm2020/index.mjs
CHANGED
|
@@ -2,7 +2,7 @@ export { AuthenticationModule } from "./authentication.module";
|
|
|
2
2
|
/**
|
|
3
3
|
* The injection tokens for service configuration.
|
|
4
4
|
*/
|
|
5
|
-
export { AUTHENTICATION_LOGOUT_PATH, AUTHENTICATION_SERVER_URL, AUTHENTICATION_TOKEN_ENDPOINT, AUTHENTICATION_DIRECT_ENDPOINT, AUTHENTICATION_ROUTE, AUTHENTICATION_MAX_INACTIVITY_MINUTES, AUTHENTICATION_USER_COUNTDOWN_SECONDS, AUTHENTICATION_IDP_INACTIVITY_MINUTES } from "./authentication.service";
|
|
5
|
+
export { AUTHENTICATION_LOGOUT_PATH, AUTHENTICATION_SERVER_URL, AUTHENTICATION_TOKEN_ENDPOINT, AUTHENTICATION_DIRECT_ENDPOINT, AUTHENTICATION_ROUTE, AUTHENTICATION_MAX_INACTIVITY_MINUTES, AUTHENTICATION_USER_COUNTDOWN_SECONDS, AUTHENTICATION_IDP_INACTIVITY_MINUTES, AUTHENTICATION_USE_UI_EVENTS_FOR_ACTIVITY, AUTHENTICATION_APP_KEEP_ALIVE_ENDPOINT } from "./authentication.service";
|
|
6
6
|
export { AUTHENTICATION_TOKEN_KEY } from "./authentication.provider";
|
|
7
7
|
export { AuthenticationComponent } from "./authentication.component";
|
|
8
8
|
export { DirectLoginComponent } from "./directlogin.component";
|
|
@@ -10,4 +10,4 @@ export { RouteGuardService } from "./route-guard.service";
|
|
|
10
10
|
export { AuthenticationService } from "./authentication.service";
|
|
11
11
|
export { TimeoutNotificationComponent } from "./timeout-notification.component";
|
|
12
12
|
export { AuthorizationInterceptor } from "./authorization.interceptor";
|
|
13
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
13
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9wcm9qZWN0cy9hdXRoZW50aWNhdGlvbi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFDLG9CQUFvQixFQUFDLE1BQU0seUJBQXlCLENBQUM7QUFFN0Q7O0dBRUc7QUFDSCxPQUFPLEVBQ0wsMEJBQTBCLEVBQzFCLHlCQUF5QixFQUN6Qiw2QkFBNkIsRUFDN0IsOEJBQThCLEVBQzlCLG9CQUFvQixFQUNwQixxQ0FBcUMsRUFDckMscUNBQXFDLEVBQ3JDLHFDQUFxQyxFQUNyQyx5Q0FBeUMsRUFDekMsc0NBQXNDLEVBQ3ZDLE1BQU0sMEJBQTBCLENBQUM7QUFFbEMsT0FBTyxFQUFDLHdCQUF3QixFQUFDLE1BQU0sMkJBQTJCLENBQUM7QUFFbkUsT0FBTyxFQUFDLHVCQUF1QixFQUFDLE1BQU0sNEJBQTRCLENBQUE7QUFDbEUsT0FBTyxFQUFDLG9CQUFvQixFQUFDLE1BQU0seUJBQXlCLENBQUE7QUFDNUQsT0FBTyxFQUFDLGlCQUFpQixFQUFDLE1BQU0sdUJBQXVCLENBQUE7QUFDdkQsT0FBTyxFQUFDLHFCQUFxQixFQUFDLE1BQU0sMEJBQTBCLENBQUE7QUFDOUQsT0FBTyxFQUFDLDRCQUE0QixFQUFDLE1BQU0sa0NBQWtDLENBQUM7QUFDOUUsT0FBTyxFQUFDLHdCQUF3QixFQUFDLE1BQU0sNkJBQTZCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQge0F1dGhlbnRpY2F0aW9uTW9kdWxlfSBmcm9tIFwiLi9hdXRoZW50aWNhdGlvbi5tb2R1bGVcIjtcclxuXHJcbi8qKlxyXG4gKiBUaGUgaW5qZWN0aW9uIHRva2VucyBmb3Igc2VydmljZSBjb25maWd1cmF0aW9uLlxyXG4gKi9cclxuZXhwb3J0IHtcclxuICBBVVRIRU5USUNBVElPTl9MT0dPVVRfUEFUSCxcclxuICBBVVRIRU5USUNBVElPTl9TRVJWRVJfVVJMLFxyXG4gIEFVVEhFTlRJQ0FUSU9OX1RPS0VOX0VORFBPSU5ULFxyXG4gIEFVVEhFTlRJQ0FUSU9OX0RJUkVDVF9FTkRQT0lOVCxcclxuICBBVVRIRU5USUNBVElPTl9ST1VURSxcclxuICBBVVRIRU5USUNBVElPTl9NQVhfSU5BQ1RJVklUWV9NSU5VVEVTLFxyXG4gIEFVVEhFTlRJQ0FUSU9OX1VTRVJfQ09VTlRET1dOX1NFQ09ORFMsXHJcbiAgQVVUSEVOVElDQVRJT05fSURQX0lOQUNUSVZJVFlfTUlOVVRFUyxcclxuICBBVVRIRU5USUNBVElPTl9VU0VfVUlfRVZFTlRTX0ZPUl9BQ1RJVklUWSxcclxuICBBVVRIRU5USUNBVElPTl9BUFBfS0VFUF9BTElWRV9FTkRQT0lOVFxyXG59IGZyb20gXCIuL2F1dGhlbnRpY2F0aW9uLnNlcnZpY2VcIjtcclxuXHJcbmV4cG9ydCB7QVVUSEVOVElDQVRJT05fVE9LRU5fS0VZfSBmcm9tIFwiLi9hdXRoZW50aWNhdGlvbi5wcm92aWRlclwiO1xyXG5cclxuZXhwb3J0IHtBdXRoZW50aWNhdGlvbkNvbXBvbmVudH0gZnJvbSBcIi4vYXV0aGVudGljYXRpb24uY29tcG9uZW50XCJcclxuZXhwb3J0IHtEaXJlY3RMb2dpbkNvbXBvbmVudH0gZnJvbSBcIi4vZGlyZWN0bG9naW4uY29tcG9uZW50XCJcclxuZXhwb3J0IHtSb3V0ZUd1YXJkU2VydmljZX0gZnJvbSBcIi4vcm91dGUtZ3VhcmQuc2VydmljZVwiXHJcbmV4cG9ydCB7QXV0aGVudGljYXRpb25TZXJ2aWNlfSBmcm9tIFwiLi9hdXRoZW50aWNhdGlvbi5zZXJ2aWNlXCJcclxuZXhwb3J0IHtUaW1lb3V0Tm90aWZpY2F0aW9uQ29tcG9uZW50fSBmcm9tIFwiLi90aW1lb3V0LW5vdGlmaWNhdGlvbi5jb21wb25lbnRcIjtcclxuZXhwb3J0IHtBdXRob3JpemF0aW9uSW50ZXJjZXB0b3J9IGZyb20gXCIuL2F1dGhvcml6YXRpb24uaW50ZXJjZXB0b3JcIjsiXX0=
|
|
@@ -12,7 +12,7 @@ import * as i1 from '@angular-cool/storage';
|
|
|
12
12
|
import { CoolStorageModule } from '@angular-cool/storage';
|
|
13
13
|
import * as i4 from '@auth0/angular-jwt';
|
|
14
14
|
import { JwtHelperService, JwtInterceptor, JWT_OPTIONS } from '@auth0/angular-jwt';
|
|
15
|
-
import { BehaviorSubject, interval, throwError, timer } from 'rxjs';
|
|
15
|
+
import { BehaviorSubject, fromEvent, interval, throwError, timer } from 'rxjs';
|
|
16
16
|
import { map, catchError, first, takeWhile } from 'rxjs/operators';
|
|
17
17
|
import * as i2$1 from '@angular/platform-browser';
|
|
18
18
|
import { trigger, state, style, transition, animate } from '@angular/animations';
|
|
@@ -67,11 +67,13 @@ let AUTHENTICATION_ROUTE = new InjectionToken("authentication_route");
|
|
|
67
67
|
let AUTHENTICATION_MAX_INACTIVITY_MINUTES = new InjectionToken("authentication_max_inactivity");
|
|
68
68
|
let AUTHENTICATION_USER_COUNTDOWN_SECONDS = new InjectionToken("authentication_user_countdown_seconds");
|
|
69
69
|
let AUTHENTICATION_IDP_INACTIVITY_MINUTES = new InjectionToken("authentication_idp_inactivity_minutes");
|
|
70
|
+
let AUTHENTICATION_USE_UI_EVENTS_FOR_ACTIVITY = new InjectionToken("authentication_use_ui_events_for_activity");
|
|
71
|
+
let AUTHENTICATION_APP_KEEP_ALIVE_ENDPOINT = new InjectionToken("authentication_app_keep_alive_endpoint");
|
|
70
72
|
/**
|
|
71
73
|
* @since 1.0.0
|
|
72
74
|
*/
|
|
73
75
|
class AuthenticationService {
|
|
74
|
-
constructor(_http, _router, _localStorageService, _jwtHelper, authenticationProvider, _authenticationRoute, _logoutPath, _tokenEndpoint, _serverUrl, _directEndpoint, _maxInactivity, _userCountdownSeconds, _idpInactivityMinutes, locationStrategy) {
|
|
76
|
+
constructor(_http, _router, _localStorageService, _jwtHelper, authenticationProvider, _authenticationRoute, _logoutPath, _tokenEndpoint, _serverUrl, _directEndpoint, _maxInactivity, _userCountdownSeconds, _idpInactivityMinutes, _useUIEventsForActivity, _keepAliveEndpoint, locationStrategy) {
|
|
75
77
|
this._http = _http;
|
|
76
78
|
this._router = _router;
|
|
77
79
|
this._localStorageService = _localStorageService;
|
|
@@ -85,6 +87,8 @@ class AuthenticationService {
|
|
|
85
87
|
this._maxInactivity = _maxInactivity;
|
|
86
88
|
this._userCountdownSeconds = _userCountdownSeconds;
|
|
87
89
|
this._idpInactivityMinutes = _idpInactivityMinutes;
|
|
90
|
+
this._useUIEventsForActivity = _useUIEventsForActivity;
|
|
91
|
+
this._keepAliveEndpoint = _keepAliveEndpoint;
|
|
88
92
|
this.locationStrategy = locationStrategy;
|
|
89
93
|
this.userCountdownSeconds = 60;
|
|
90
94
|
this.idpInactivityMinutes = 5;
|
|
@@ -95,6 +99,7 @@ class AuthenticationService {
|
|
|
95
99
|
this._isAuthenticatedSubject = new BehaviorSubject(false);
|
|
96
100
|
this._userIsAboutToTimeOut = new BehaviorSubject(false);
|
|
97
101
|
this._maxInactivityMinutes = 120;
|
|
102
|
+
this._uiEvents = ['keydown', 'click', 'wheel', 'mousemove'];
|
|
98
103
|
this.contextRoot = "";
|
|
99
104
|
if (isDevMode()) {
|
|
100
105
|
console.debug("window.location.href: " + window.location.href);
|
|
@@ -118,6 +123,10 @@ class AuthenticationService {
|
|
|
118
123
|
if (_idpInactivityMinutes) {
|
|
119
124
|
this.idpInactivityMinutes = _idpInactivityMinutes;
|
|
120
125
|
}
|
|
126
|
+
//Subscribe to UI events for user activity
|
|
127
|
+
if (_useUIEventsForActivity) {
|
|
128
|
+
this.subscribeToUIActivity();
|
|
129
|
+
}
|
|
121
130
|
this.hasValidConfig();
|
|
122
131
|
//There could be a non-expired token in local storage.
|
|
123
132
|
let token = this.authenticationProvider.authToken;
|
|
@@ -187,6 +196,9 @@ class AuthenticationService {
|
|
|
187
196
|
this.logout(true);
|
|
188
197
|
});
|
|
189
198
|
}
|
|
199
|
+
makeKeepAliveRequest() {
|
|
200
|
+
this._http.get(this.keepAliveLocation(), { withCredentials: true }).subscribe();
|
|
201
|
+
}
|
|
190
202
|
/**
|
|
191
203
|
* Verifies whether or not a current user session exists.
|
|
192
204
|
*
|
|
@@ -211,6 +223,14 @@ class AuthenticationService {
|
|
|
211
223
|
return this._tokenEndpoint;
|
|
212
224
|
}
|
|
213
225
|
}
|
|
226
|
+
keepAliveLocation() {
|
|
227
|
+
if (this._serverUrl) {
|
|
228
|
+
return this._serverUrl + this._keepAliveEndpoint;
|
|
229
|
+
}
|
|
230
|
+
else {
|
|
231
|
+
return this._keepAliveEndpoint;
|
|
232
|
+
}
|
|
233
|
+
}
|
|
214
234
|
directLoginLocation() {
|
|
215
235
|
if (this._serverUrl) {
|
|
216
236
|
return this._serverUrl + this._directEndpoint;
|
|
@@ -317,6 +337,14 @@ class AuthenticationService {
|
|
|
317
337
|
validateToken(token) {
|
|
318
338
|
return (token && !this._jwtHelper.isTokenExpired(token));
|
|
319
339
|
}
|
|
340
|
+
subscribeToUIActivity() {
|
|
341
|
+
this._uiEvents.forEach(event => fromEvent(document, event).subscribe(_ => {
|
|
342
|
+
//Only update activity on UI event if not already about to time out. Counter intuitive, but in this case we should have 60 second count-down dialog open.
|
|
343
|
+
if (this._userIsAboutToTimeOut.getValue() !== true) {
|
|
344
|
+
this.updateUserActivity();
|
|
345
|
+
}
|
|
346
|
+
}));
|
|
347
|
+
}
|
|
320
348
|
subscribeToTokenRefresh(token) {
|
|
321
349
|
let exp = this._jwtHelper.getTokenExpirationDate(token);
|
|
322
350
|
// Use a timer to periodically check timeouts
|
|
@@ -356,7 +384,12 @@ class AuthenticationService {
|
|
|
356
384
|
refreshTokenIfUserIsActive() {
|
|
357
385
|
//Only refresh if the user has been active
|
|
358
386
|
if (this._lastUserInteraction && ((new Date().valueOf() - this._lastUserInteraction.valueOf()) <= (this._maxInactivityMinutes * 60 * 1000))) {
|
|
387
|
+
//Refresh the JWT
|
|
359
388
|
this.requestAccessToken(false);
|
|
389
|
+
//Might as well make application keep alive requests at the same time as refreshing tokens - more efficient & probably needs same logic
|
|
390
|
+
if (this._keepAliveEndpoint) {
|
|
391
|
+
this.makeKeepAliveRequest();
|
|
392
|
+
}
|
|
360
393
|
}
|
|
361
394
|
}
|
|
362
395
|
hasValidConfig() {
|
|
@@ -383,7 +416,7 @@ AuthenticationService.SEC_GOV_CLASS_HEADER = "SecurityGovernorClass";
|
|
|
383
416
|
AuthenticationService.SEC_GOV_ID_HEADER = "SecurityGovernorId";
|
|
384
417
|
AuthenticationService.DEIDENT_HEADER = "DeidentifiedContext";
|
|
385
418
|
AuthenticationService.LIMITED_HEADER = "LimitedContext";
|
|
386
|
-
AuthenticationService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: AuthenticationService, deps: [{ token: i1$1.HttpClient }, { token: i2.Router }, { token: i1.CoolLocalStorage }, { token: i4.JwtHelperService }, { token: AuthenticationProvider }, { token: AUTHENTICATION_ROUTE }, { token: AUTHENTICATION_LOGOUT_PATH }, { token: AUTHENTICATION_TOKEN_ENDPOINT }, { token: AUTHENTICATION_SERVER_URL, optional: true }, { token: AUTHENTICATION_DIRECT_ENDPOINT, optional: true }, { token: AUTHENTICATION_MAX_INACTIVITY_MINUTES, optional: true }, { token: AUTHENTICATION_USER_COUNTDOWN_SECONDS, optional: true }, { token: AUTHENTICATION_IDP_INACTIVITY_MINUTES, optional: true }, { token: LocationStrategy, optional: true }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
419
|
+
AuthenticationService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: AuthenticationService, deps: [{ token: i1$1.HttpClient }, { token: i2.Router }, { token: i1.CoolLocalStorage }, { token: i4.JwtHelperService }, { token: AuthenticationProvider }, { token: AUTHENTICATION_ROUTE }, { token: AUTHENTICATION_LOGOUT_PATH }, { token: AUTHENTICATION_TOKEN_ENDPOINT }, { token: AUTHENTICATION_SERVER_URL, optional: true }, { token: AUTHENTICATION_DIRECT_ENDPOINT, optional: true }, { token: AUTHENTICATION_MAX_INACTIVITY_MINUTES, optional: true }, { token: AUTHENTICATION_USER_COUNTDOWN_SECONDS, optional: true }, { token: AUTHENTICATION_IDP_INACTIVITY_MINUTES, optional: true }, { token: AUTHENTICATION_USE_UI_EVENTS_FOR_ACTIVITY, optional: true }, { token: AUTHENTICATION_APP_KEEP_ALIVE_ENDPOINT, optional: true }, { token: LocationStrategy, optional: true }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
387
420
|
AuthenticationService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: AuthenticationService });
|
|
388
421
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: AuthenticationService, decorators: [{
|
|
389
422
|
type: Injectable
|
|
@@ -422,6 +455,16 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
|
|
|
422
455
|
}, {
|
|
423
456
|
type: Inject,
|
|
424
457
|
args: [AUTHENTICATION_IDP_INACTIVITY_MINUTES]
|
|
458
|
+
}] }, { type: undefined, decorators: [{
|
|
459
|
+
type: Optional
|
|
460
|
+
}, {
|
|
461
|
+
type: Inject,
|
|
462
|
+
args: [AUTHENTICATION_USE_UI_EVENTS_FOR_ACTIVITY]
|
|
463
|
+
}] }, { type: undefined, decorators: [{
|
|
464
|
+
type: Optional
|
|
465
|
+
}, {
|
|
466
|
+
type: Inject,
|
|
467
|
+
args: [AUTHENTICATION_APP_KEEP_ALIVE_ENDPOINT]
|
|
425
468
|
}] }, { type: i4$1.LocationStrategy, decorators: [{
|
|
426
469
|
type: Optional
|
|
427
470
|
}, {
|
|
@@ -996,5 +1039,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
|
|
|
996
1039
|
* Generated bundle index. Do not edit.
|
|
997
1040
|
*/
|
|
998
1041
|
|
|
999
|
-
export { AUTHENTICATION_DIRECT_ENDPOINT, AUTHENTICATION_IDP_INACTIVITY_MINUTES, AUTHENTICATION_LOGOUT_PATH, AUTHENTICATION_MAX_INACTIVITY_MINUTES, AUTHENTICATION_ROUTE, AUTHENTICATION_SERVER_URL, AUTHENTICATION_TOKEN_ENDPOINT, AUTHENTICATION_TOKEN_KEY, AUTHENTICATION_USER_COUNTDOWN_SECONDS, AuthenticationComponent, AuthenticationModule, AuthenticationService, AuthorizationInterceptor, DirectLoginComponent, RouteGuardService, TimeoutNotificationComponent };
|
|
1042
|
+
export { AUTHENTICATION_APP_KEEP_ALIVE_ENDPOINT, AUTHENTICATION_DIRECT_ENDPOINT, AUTHENTICATION_IDP_INACTIVITY_MINUTES, AUTHENTICATION_LOGOUT_PATH, AUTHENTICATION_MAX_INACTIVITY_MINUTES, AUTHENTICATION_ROUTE, AUTHENTICATION_SERVER_URL, AUTHENTICATION_TOKEN_ENDPOINT, AUTHENTICATION_TOKEN_KEY, AUTHENTICATION_USER_COUNTDOWN_SECONDS, AUTHENTICATION_USE_UI_EVENTS_FOR_ACTIVITY, AuthenticationComponent, AuthenticationModule, AuthenticationService, AuthorizationInterceptor, DirectLoginComponent, RouteGuardService, TimeoutNotificationComponent };
|
|
1000
1043
|
//# sourceMappingURL=huntsman-cancer-institute-authentication.mjs.map
|