@huntsman-cancer-institute/authentication 16.0.1 → 17.0.1
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.component.d.ts +27 -27
- package/authentication.module.d.ts +24 -24
- package/authentication.provider.d.ts +16 -16
- package/authentication.service.d.ts +123 -123
- package/authorization.interceptor.d.ts +11 -11
- package/directlogin.component.d.ts +21 -21
- package/esm2022/authentication.component.mjs +112 -112
- package/esm2022/authentication.module.mjs +103 -103
- package/esm2022/authentication.provider.mjs +36 -36
- package/esm2022/authentication.service.mjs +393 -393
- package/esm2022/authorization.interceptor.mjs +79 -79
- package/esm2022/directlogin.component.mjs +47 -47
- package/esm2022/huntsman-cancer-institute-authentication.mjs +4 -4
- package/esm2022/index.mjs +12 -12
- package/esm2022/route-guard.service.mjs +22 -22
- package/esm2022/timeout-notification.component.mjs +117 -117
- package/fesm2022/huntsman-cancer-institute-authentication.mjs +831 -831
- package/fesm2022/huntsman-cancer-institute-authentication.mjs.map +1 -1
- package/index.d.ts +12 -12
- package/package.json +4 -4
- package/route-guard.service.d.ts +1 -1
- package/timeout-notification.component.d.ts +14 -14
|
@@ -1,393 +1,393 @@
|
|
|
1
|
-
/*
|
|
2
|
-
* Copyright (c) 2016 Huntsman Cancer Institute at the University of Utah, Confidential and Proprietary
|
|
3
|
-
*/
|
|
4
|
-
import { Injectable, InjectionToken, Inject, Optional, isDevMode } from "@angular/core";
|
|
5
|
-
import { LocationStrategy } from "@angular/common";
|
|
6
|
-
import { Router } from "@angular/router";
|
|
7
|
-
import { HttpClient, HttpHeaders } from "@angular/common/http";
|
|
8
|
-
import { interval, BehaviorSubject, throwError } from "rxjs";
|
|
9
|
-
import { catchError, map } from "rxjs/operators";
|
|
10
|
-
import { JwtHelperService } from "@auth0/angular-jwt";
|
|
11
|
-
import { AuthenticationProvider } from "./authentication.provider";
|
|
12
|
-
import { CoolLocalStorage } from "@angular-cool/storage";
|
|
13
|
-
import * as i0 from "@angular/core";
|
|
14
|
-
import * as i1 from "@angular/common/http";
|
|
15
|
-
import * as i2 from "@angular/router";
|
|
16
|
-
import * as i3 from "@angular-cool/storage";
|
|
17
|
-
import * as i4 from "@auth0/angular-jwt";
|
|
18
|
-
import * as i5 from "./authentication.provider";
|
|
19
|
-
import * as i6 from "@angular/common";
|
|
20
|
-
/**
|
|
21
|
-
* The token used for injection of the server side endpoint for the currently authenticated subject.
|
|
22
|
-
*
|
|
23
|
-
* @type {InjectionToken}
|
|
24
|
-
*/
|
|
25
|
-
export let AUTHENTICATION_SERVER_URL = new InjectionToken("authentication_server_rest_api");
|
|
26
|
-
export let AUTHENTICATION_LOGOUT_PATH = new InjectionToken("authentication_logout_path");
|
|
27
|
-
export let AUTHENTICATION_DIRECT_ENDPOINT = new InjectionToken("authentication_direct_endpoint");
|
|
28
|
-
export let AUTHENTICATION_TOKEN_ENDPOINT = new InjectionToken("authentication_token_endpoint");
|
|
29
|
-
export let AUTHENTICATION_ROUTE = new InjectionToken("authentication_route");
|
|
30
|
-
export let AUTHENTICATION_MAX_INACTIVITY_MINUTES = new InjectionToken("authentication_max_inactivity");
|
|
31
|
-
export let AUTHENTICATION_USER_COUNTDOWN_SECONDS = new InjectionToken("authentication_user_countdown_seconds");
|
|
32
|
-
export let AUTHENTICATION_IDP_INACTIVITY_MINUTES = new InjectionToken("authentication_idp_inactivity_minutes");
|
|
33
|
-
/**
|
|
34
|
-
* @since 1.0.0
|
|
35
|
-
*/
|
|
36
|
-
export class AuthenticationService {
|
|
37
|
-
/**
|
|
38
|
-
* The generic error message used when a server error is thrown without a status.
|
|
39
|
-
*
|
|
40
|
-
* @type {string}
|
|
41
|
-
*/
|
|
42
|
-
static { this.GENERIC_ERR_MSG = "Server error"; }
|
|
43
|
-
static { this.CONTENT_TYPE = "Content-Type"; }
|
|
44
|
-
static { this.SEC_GOV_CLASS_HEADER = "SecurityGovernorClass"; }
|
|
45
|
-
static { this.SEC_GOV_ID_HEADER = "SecurityGovernorId"; }
|
|
46
|
-
static { this.DEIDENT_HEADER = "DeidentifiedContext"; }
|
|
47
|
-
static { this.LIMITED_HEADER = "LimitedContext"; }
|
|
48
|
-
constructor(_http, _router, _localStorageService, _jwtHelper, authenticationProvider, _authenticationRoute, _logoutPath, _tokenEndpoint, _serverUrl, _directEndpoint, _maxInactivity, _userCountdownSeconds, _idpInactivityMinutes, locationStrategy) {
|
|
49
|
-
this._http = _http;
|
|
50
|
-
this._router = _router;
|
|
51
|
-
this._localStorageService = _localStorageService;
|
|
52
|
-
this._jwtHelper = _jwtHelper;
|
|
53
|
-
this.authenticationProvider = authenticationProvider;
|
|
54
|
-
this._authenticationRoute = _authenticationRoute;
|
|
55
|
-
this._logoutPath = _logoutPath;
|
|
56
|
-
this._tokenEndpoint = _tokenEndpoint;
|
|
57
|
-
this._serverUrl = _serverUrl;
|
|
58
|
-
this._directEndpoint = _directEndpoint;
|
|
59
|
-
this._maxInactivity = _maxInactivity;
|
|
60
|
-
this._userCountdownSeconds = _userCountdownSeconds;
|
|
61
|
-
this._idpInactivityMinutes = _idpInactivityMinutes;
|
|
62
|
-
this.locationStrategy = locationStrategy;
|
|
63
|
-
this.userCountdownSeconds = 60;
|
|
64
|
-
this.idpInactivityMinutes = 5;
|
|
65
|
-
this.contentType = "application/json";
|
|
66
|
-
this.limitedContext = false;
|
|
67
|
-
this.deidentifiedContext = false;
|
|
68
|
-
this.maxViewPermission = new BehaviorSubject("viewident");
|
|
69
|
-
this._isAuthenticatedSubject = new BehaviorSubject(false);
|
|
70
|
-
this._userIsAboutToTimeOut = new BehaviorSubject(false);
|
|
71
|
-
this._maxInactivityMinutes = 120;
|
|
72
|
-
this.contextRoot = "";
|
|
73
|
-
if (isDevMode()) {
|
|
74
|
-
console.debug("window.location.href: " + window.location.href);
|
|
75
|
-
}
|
|
76
|
-
if (window.location) {
|
|
77
|
-
let parts = window.location.href.split("/");
|
|
78
|
-
this.baseUrl = parts[0] + "//" + parts[2];
|
|
79
|
-
if (parts.length > 3) {
|
|
80
|
-
this.contextRoot = parts[3];
|
|
81
|
-
}
|
|
82
|
-
}
|
|
83
|
-
if (this._localStorageService.getItem("maxViewPermission")) {
|
|
84
|
-
this.maxViewPermission.next(this._localStorageService.getItem("maxViewPermission"));
|
|
85
|
-
}
|
|
86
|
-
if (_maxInactivity) {
|
|
87
|
-
this._maxInactivityMinutes = _maxInactivity;
|
|
88
|
-
}
|
|
89
|
-
if (_userCountdownSeconds) {
|
|
90
|
-
this.userCountdownSeconds = _userCountdownSeconds;
|
|
91
|
-
}
|
|
92
|
-
if (_idpInactivityMinutes) {
|
|
93
|
-
this.idpInactivityMinutes = _idpInactivityMinutes;
|
|
94
|
-
}
|
|
95
|
-
this.hasValidConfig();
|
|
96
|
-
//There could be a non-expired token in local storage.
|
|
97
|
-
let token = this.authenticationProvider.authToken;
|
|
98
|
-
this.storeToken(token);
|
|
99
|
-
}
|
|
100
|
-
getBaseUrl() {
|
|
101
|
-
return (this.baseUrl) ? this.baseUrl : "";
|
|
102
|
-
}
|
|
103
|
-
getContextRoot() {
|
|
104
|
-
return this.contextRoot;
|
|
105
|
-
}
|
|
106
|
-
getHeaders(req) {
|
|
107
|
-
let headers = req.headers;
|
|
108
|
-
//Don't set content type if already set
|
|
109
|
-
if (!req.headers.get(AuthenticationService.CONTENT_TYPE)) {
|
|
110
|
-
headers = headers.set(AuthenticationService.CONTENT_TYPE, this.contentType.toString());
|
|
111
|
-
}
|
|
112
|
-
if (headers.get(AuthenticationService.SEC_GOV_CLASS_HEADER) === "") {
|
|
113
|
-
headers = headers.delete(AuthenticationService.SEC_GOV_CLASS_HEADER);
|
|
114
|
-
}
|
|
115
|
-
else if (this.securityGovernorClass && !headers.get(AuthenticationService.SEC_GOV_CLASS_HEADER)) {
|
|
116
|
-
headers = headers.set(AuthenticationService.SEC_GOV_CLASS_HEADER, this.securityGovernorClass);
|
|
117
|
-
}
|
|
118
|
-
if (headers.get(AuthenticationService.SEC_GOV_ID_HEADER) === "") {
|
|
119
|
-
headers = headers.delete(AuthenticationService.SEC_GOV_ID_HEADER);
|
|
120
|
-
}
|
|
121
|
-
else if (this.securityGovernorId && !headers.get(AuthenticationService.SEC_GOV_ID_HEADER)) {
|
|
122
|
-
headers = headers.set(AuthenticationService.SEC_GOV_ID_HEADER, this.securityGovernorId.toString());
|
|
123
|
-
}
|
|
124
|
-
headers = headers.set(AuthenticationService.DEIDENT_HEADER, this.deidentifiedContext.toString());
|
|
125
|
-
headers = headers.set(AuthenticationService.LIMITED_HEADER, this.limitedContext.toString());
|
|
126
|
-
return headers;
|
|
127
|
-
}
|
|
128
|
-
get authenticationTokenKey() {
|
|
129
|
-
return this.authenticationProvider.authenticationTokenKey;
|
|
130
|
-
}
|
|
131
|
-
get authToken() {
|
|
132
|
-
return this.authenticationProvider.authToken;
|
|
133
|
-
}
|
|
134
|
-
updateUserActivity() {
|
|
135
|
-
if (this._isAuthenticatedSubject.value) {
|
|
136
|
-
this._lastUserInteraction = new Date();
|
|
137
|
-
this._userIsAboutToTimeOut.next(false);
|
|
138
|
-
}
|
|
139
|
-
}
|
|
140
|
-
/**
|
|
141
|
-
* A mutator for identifying the clients original request location. Setting this value will influence the end location
|
|
142
|
-
* navigated to by {@link #navigateToPath}.
|
|
143
|
-
*
|
|
144
|
-
* @param redirectUrl location of the users request before authentication
|
|
145
|
-
*/
|
|
146
|
-
set redirectUrl(redirectUrl) {
|
|
147
|
-
this._redirectUrl = redirectUrl;
|
|
148
|
-
}
|
|
149
|
-
get redirectUrl() {
|
|
150
|
-
return this._redirectUrl;
|
|
151
|
-
}
|
|
152
|
-
requestAccessToken(redirectOnSuccess) {
|
|
153
|
-
this._http.get(this.tokenLocation(), { withCredentials: true })
|
|
154
|
-
.subscribe((response) => {
|
|
155
|
-
this.storeToken(response.auth_token);
|
|
156
|
-
if (redirectOnSuccess) {
|
|
157
|
-
this.proceedIfAuthenticated();
|
|
158
|
-
}
|
|
159
|
-
}, (error) => {
|
|
160
|
-
//Token refresh failed.
|
|
161
|
-
this.logout(true);
|
|
162
|
-
});
|
|
163
|
-
}
|
|
164
|
-
/**
|
|
165
|
-
* Verifies whether or not a current user session exists.
|
|
166
|
-
*
|
|
167
|
-
* @returns {Observable<boolean>} evaluates to true if the user is authenticated, false otherwise.
|
|
168
|
-
*/
|
|
169
|
-
isAuthenticated() {
|
|
170
|
-
return this._isAuthenticatedSubject.asObservable();
|
|
171
|
-
}
|
|
172
|
-
isAboutToTimeOut() {
|
|
173
|
-
return this._userIsAboutToTimeOut.asObservable();
|
|
174
|
-
}
|
|
175
|
-
getTimeoutStart() {
|
|
176
|
-
if (this._lastUserInteraction) {
|
|
177
|
-
return this._lastUserInteraction.valueOf() + (((this._maxInactivityMinutes * 60) - this.userCountdownSeconds) * 1000);
|
|
178
|
-
}
|
|
179
|
-
}
|
|
180
|
-
tokenLocation() {
|
|
181
|
-
if (this._serverUrl) {
|
|
182
|
-
return this._serverUrl + this._tokenEndpoint;
|
|
183
|
-
}
|
|
184
|
-
else {
|
|
185
|
-
return this._tokenEndpoint;
|
|
186
|
-
}
|
|
187
|
-
}
|
|
188
|
-
directLoginLocation() {
|
|
189
|
-
if (this._serverUrl) {
|
|
190
|
-
return this._serverUrl + this._directEndpoint;
|
|
191
|
-
}
|
|
192
|
-
else {
|
|
193
|
-
return this._directEndpoint;
|
|
194
|
-
}
|
|
195
|
-
}
|
|
196
|
-
logoutLocation() {
|
|
197
|
-
if (this._serverUrl) {
|
|
198
|
-
return this._serverUrl + this._logoutPath;
|
|
199
|
-
}
|
|
200
|
-
else {
|
|
201
|
-
return this._logoutPath;
|
|
202
|
-
}
|
|
203
|
-
}
|
|
204
|
-
/**
|
|
205
|
-
* A function to authenticated the user with the provided credentials. Failure results in an error that describes the
|
|
206
|
-
* server response (status and status message) and should be actionable by the client application.
|
|
207
|
-
*
|
|
208
|
-
* @param username of the authenticating user to verify
|
|
209
|
-
* @param password of the authenticating user to verify
|
|
210
|
-
* @returns {Observable<R>} describing the result of the login action, true or an error
|
|
211
|
-
*/
|
|
212
|
-
login(_username, _password) {
|
|
213
|
-
return this._http.post(this.directLoginLocation(), { username: _username, password: _password }, { observe: "response" }).pipe(map((resp) => {
|
|
214
|
-
if (resp.status === 201) {
|
|
215
|
-
return true;
|
|
216
|
-
}
|
|
217
|
-
else {
|
|
218
|
-
throw new Error("Authentication failed. " + resp.status + ": " + resp.statusText);
|
|
219
|
-
}
|
|
220
|
-
}), catchError(this.handleError));
|
|
221
|
-
}
|
|
222
|
-
clearLogin() {
|
|
223
|
-
//Front-end logout
|
|
224
|
-
try {
|
|
225
|
-
this._localStorageService.removeItem(this.authenticationProvider.authenticationTokenKey);
|
|
226
|
-
this.unsubscribeFromTokenRefresh();
|
|
227
|
-
this._isAuthenticatedSubject.next(false);
|
|
228
|
-
this._userIsAboutToTimeOut.next(false);
|
|
229
|
-
}
|
|
230
|
-
catch (Error) {
|
|
231
|
-
}
|
|
232
|
-
//Back-end logout
|
|
233
|
-
let headers = new HttpHeaders().set(AuthenticationService.CONTENT_TYPE, "text/plain");
|
|
234
|
-
return this._http.get(this.logoutLocation(), { headers: headers });
|
|
235
|
-
}
|
|
236
|
-
/**
|
|
237
|
-
* A function to signal the termination of the current session. Invoking this function will clean up any relevant state
|
|
238
|
-
* related to the last active session.
|
|
239
|
-
*/
|
|
240
|
-
logout(keepCurrentRoute = false) {
|
|
241
|
-
//Prevent logout if already on authentication route. Doing otherwise screws up SAML
|
|
242
|
-
if (!this._router.routerState || this._router.routerState.snapshot.url !== this._authenticationRoute) {
|
|
243
|
-
this._redirectUrl = (keepCurrentRoute && this._router.routerState && this._router.routerState.snapshot) ? this._router.routerState.snapshot.url : "";
|
|
244
|
-
if (this._redirectUrl.startsWith("/")) {
|
|
245
|
-
this._redirectUrl = this._redirectUrl.substring(1);
|
|
246
|
-
}
|
|
247
|
-
this.clearLogin().subscribe((response) => {
|
|
248
|
-
window.location.replace(this._redirectUrl);
|
|
249
|
-
}, (error) => {
|
|
250
|
-
window.location.replace(this._redirectUrl);
|
|
251
|
-
});
|
|
252
|
-
}
|
|
253
|
-
}
|
|
254
|
-
storeToken(token) {
|
|
255
|
-
let valid = this.validateToken(token);
|
|
256
|
-
// unsubscribe from refesh before we decide wether to resubscribe
|
|
257
|
-
this.unsubscribeFromTokenRefresh();
|
|
258
|
-
if (valid) {
|
|
259
|
-
this._localStorageService.setItem(this.authenticationProvider.authenticationTokenKey, token);
|
|
260
|
-
this.subscribeToTokenRefresh(token);
|
|
261
|
-
//Change the BehaviorSubject if the user was not previously authenticated.
|
|
262
|
-
//Since other code may be subscribing to this observable, we don't want to cause new events to fire if just refreshing the JWT.
|
|
263
|
-
if (!this._isAuthenticatedSubject.value) {
|
|
264
|
-
this._isAuthenticatedSubject.next(true);
|
|
265
|
-
}
|
|
266
|
-
}
|
|
267
|
-
else {
|
|
268
|
-
this._localStorageService.removeItem(this.authenticationProvider.authenticationTokenKey);
|
|
269
|
-
this._isAuthenticatedSubject.next(false);
|
|
270
|
-
}
|
|
271
|
-
}
|
|
272
|
-
proceedIfAuthenticated() {
|
|
273
|
-
if (isDevMode()) {
|
|
274
|
-
console.debug("AuthenticationService.proceedIfAuthenticated: " + this._redirectUrl);
|
|
275
|
-
}
|
|
276
|
-
if (this._isAuthenticatedSubject.value) {
|
|
277
|
-
//Login counts as user activity, too
|
|
278
|
-
this.updateUserActivity();
|
|
279
|
-
if (this._redirectUrl && this._redirectUrl && this._redirectUrl !== "") {
|
|
280
|
-
this._router.navigateByUrl(this._redirectUrl);
|
|
281
|
-
}
|
|
282
|
-
else {
|
|
283
|
-
this._router.navigate([""]);
|
|
284
|
-
}
|
|
285
|
-
return true;
|
|
286
|
-
}
|
|
287
|
-
else {
|
|
288
|
-
return false;
|
|
289
|
-
}
|
|
290
|
-
}
|
|
291
|
-
validateToken(token) {
|
|
292
|
-
return (token && !this._jwtHelper.isTokenExpired(token));
|
|
293
|
-
}
|
|
294
|
-
subscribeToTokenRefresh(token) {
|
|
295
|
-
let exp = this._jwtHelper.getTokenExpirationDate(token);
|
|
296
|
-
// Use a timer to periodically check timeouts
|
|
297
|
-
this._refreshSubscription = interval(1000)
|
|
298
|
-
.subscribe(() => {
|
|
299
|
-
// If a tab is inactive we can't know if our timer is accurate
|
|
300
|
-
// so when the interval hits check against timestamps
|
|
301
|
-
if (this._isAuthenticatedSubject.value && Date.now() > this.getTimeoutStart()) {
|
|
302
|
-
//Don't update the subject more than once! Doing so initializes more than one countdown timer!
|
|
303
|
-
if (this._userIsAboutToTimeOut.getValue() !== true) {
|
|
304
|
-
this._userIsAboutToTimeOut.next(true);
|
|
305
|
-
}
|
|
306
|
-
}
|
|
307
|
-
// check for refresh token
|
|
308
|
-
let msToExpiry = (exp.valueOf() - new Date().valueOf());
|
|
309
|
-
// Refresh 60 seconds before expiry
|
|
310
|
-
if (msToExpiry <= 60000) {
|
|
311
|
-
this.refreshTokenIfUserIsActive();
|
|
312
|
-
}
|
|
313
|
-
});
|
|
314
|
-
}
|
|
315
|
-
unsubscribeFromTokenRefresh() {
|
|
316
|
-
if (this._refreshSubscription && !this._refreshSubscription.closed) {
|
|
317
|
-
this._refreshSubscription.unsubscribe();
|
|
318
|
-
}
|
|
319
|
-
}
|
|
320
|
-
getMaxViewPermission() {
|
|
321
|
-
return this.maxViewPermission.getValue();
|
|
322
|
-
}
|
|
323
|
-
getMaxViewPermissionSubject() {
|
|
324
|
-
return this.maxViewPermission;
|
|
325
|
-
}
|
|
326
|
-
setMaxViewPermission(maxViewPermission) {
|
|
327
|
-
this._localStorageService.setItem("maxViewPermission", maxViewPermission);
|
|
328
|
-
this.maxViewPermission.next(maxViewPermission);
|
|
329
|
-
}
|
|
330
|
-
refreshTokenIfUserIsActive() {
|
|
331
|
-
//Only refresh if the user has been active
|
|
332
|
-
if (this._lastUserInteraction && ((new Date().valueOf() - this._lastUserInteraction.valueOf()) <= (this._maxInactivityMinutes * 60 * 1000))) {
|
|
333
|
-
this.requestAccessToken(false);
|
|
334
|
-
}
|
|
335
|
-
}
|
|
336
|
-
hasValidConfig() {
|
|
337
|
-
if (this._tokenEndpoint == null && (this._serverUrl === null || this._logoutPath === null)) {
|
|
338
|
-
throw new Error("BUG ALERT! Invalid AuthenticationService configuration. No valid configuration for authentication endpoint(s).");
|
|
339
|
-
}
|
|
340
|
-
if (this._localStorageService === null || this.authenticationProvider.authenticationTokenKey === null) {
|
|
341
|
-
throw new Error("BUG ALERT! Invalid AuthenticationService configuration. No valid configuration for local storage");
|
|
342
|
-
}
|
|
343
|
-
}
|
|
344
|
-
handleError(error) {
|
|
345
|
-
let errMsg = (error.message) ? error.message : AuthenticationService.GENERIC_ERR_MSG;
|
|
346
|
-
return throwError(errMsg);
|
|
347
|
-
}
|
|
348
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "
|
|
349
|
-
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "
|
|
350
|
-
}
|
|
351
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "
|
|
352
|
-
type: Injectable
|
|
353
|
-
}], ctorParameters:
|
|
354
|
-
type: Inject,
|
|
355
|
-
args: [AUTHENTICATION_ROUTE]
|
|
356
|
-
}] }, { type: undefined, decorators: [{
|
|
357
|
-
type: Inject,
|
|
358
|
-
args: [AUTHENTICATION_LOGOUT_PATH]
|
|
359
|
-
}] }, { type: undefined, decorators: [{
|
|
360
|
-
type: Inject,
|
|
361
|
-
args: [AUTHENTICATION_TOKEN_ENDPOINT]
|
|
362
|
-
}] }, { type: undefined, decorators: [{
|
|
363
|
-
type: Optional
|
|
364
|
-
}, {
|
|
365
|
-
type: Inject,
|
|
366
|
-
args: [AUTHENTICATION_SERVER_URL]
|
|
367
|
-
}] }, { type: undefined, decorators: [{
|
|
368
|
-
type: Optional
|
|
369
|
-
}, {
|
|
370
|
-
type: Inject,
|
|
371
|
-
args: [AUTHENTICATION_DIRECT_ENDPOINT]
|
|
372
|
-
}] }, { type: undefined, decorators: [{
|
|
373
|
-
type: Optional
|
|
374
|
-
}, {
|
|
375
|
-
type: Inject,
|
|
376
|
-
args: [AUTHENTICATION_MAX_INACTIVITY_MINUTES]
|
|
377
|
-
}] }, { type: undefined, decorators: [{
|
|
378
|
-
type: Optional
|
|
379
|
-
}, {
|
|
380
|
-
type: Inject,
|
|
381
|
-
args: [AUTHENTICATION_USER_COUNTDOWN_SECONDS]
|
|
382
|
-
}] }, { type: undefined, decorators: [{
|
|
383
|
-
type: Optional
|
|
384
|
-
}, {
|
|
385
|
-
type: Inject,
|
|
386
|
-
args: [AUTHENTICATION_IDP_INACTIVITY_MINUTES]
|
|
387
|
-
}] }, { type: i6.LocationStrategy, decorators: [{
|
|
388
|
-
type: Optional
|
|
389
|
-
}, {
|
|
390
|
-
type: Inject,
|
|
391
|
-
args: [LocationStrategy]
|
|
392
|
-
}] }]
|
|
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"]}
|
|
1
|
+
/*
|
|
2
|
+
* Copyright (c) 2016 Huntsman Cancer Institute at the University of Utah, Confidential and Proprietary
|
|
3
|
+
*/
|
|
4
|
+
import { Injectable, InjectionToken, Inject, Optional, isDevMode } from "@angular/core";
|
|
5
|
+
import { LocationStrategy } from "@angular/common";
|
|
6
|
+
import { Router } from "@angular/router";
|
|
7
|
+
import { HttpClient, HttpHeaders } from "@angular/common/http";
|
|
8
|
+
import { interval, BehaviorSubject, throwError } from "rxjs";
|
|
9
|
+
import { catchError, map } from "rxjs/operators";
|
|
10
|
+
import { JwtHelperService } from "@auth0/angular-jwt";
|
|
11
|
+
import { AuthenticationProvider } from "./authentication.provider";
|
|
12
|
+
import { CoolLocalStorage } from "@angular-cool/storage";
|
|
13
|
+
import * as i0 from "@angular/core";
|
|
14
|
+
import * as i1 from "@angular/common/http";
|
|
15
|
+
import * as i2 from "@angular/router";
|
|
16
|
+
import * as i3 from "@angular-cool/storage";
|
|
17
|
+
import * as i4 from "@auth0/angular-jwt";
|
|
18
|
+
import * as i5 from "./authentication.provider";
|
|
19
|
+
import * as i6 from "@angular/common";
|
|
20
|
+
/**
|
|
21
|
+
* The token used for injection of the server side endpoint for the currently authenticated subject.
|
|
22
|
+
*
|
|
23
|
+
* @type {InjectionToken}
|
|
24
|
+
*/
|
|
25
|
+
export let AUTHENTICATION_SERVER_URL = new InjectionToken("authentication_server_rest_api");
|
|
26
|
+
export let AUTHENTICATION_LOGOUT_PATH = new InjectionToken("authentication_logout_path");
|
|
27
|
+
export let AUTHENTICATION_DIRECT_ENDPOINT = new InjectionToken("authentication_direct_endpoint");
|
|
28
|
+
export let AUTHENTICATION_TOKEN_ENDPOINT = new InjectionToken("authentication_token_endpoint");
|
|
29
|
+
export let AUTHENTICATION_ROUTE = new InjectionToken("authentication_route");
|
|
30
|
+
export let AUTHENTICATION_MAX_INACTIVITY_MINUTES = new InjectionToken("authentication_max_inactivity");
|
|
31
|
+
export let AUTHENTICATION_USER_COUNTDOWN_SECONDS = new InjectionToken("authentication_user_countdown_seconds");
|
|
32
|
+
export let AUTHENTICATION_IDP_INACTIVITY_MINUTES = new InjectionToken("authentication_idp_inactivity_minutes");
|
|
33
|
+
/**
|
|
34
|
+
* @since 1.0.0
|
|
35
|
+
*/
|
|
36
|
+
export class AuthenticationService {
|
|
37
|
+
/**
|
|
38
|
+
* The generic error message used when a server error is thrown without a status.
|
|
39
|
+
*
|
|
40
|
+
* @type {string}
|
|
41
|
+
*/
|
|
42
|
+
static { this.GENERIC_ERR_MSG = "Server error"; }
|
|
43
|
+
static { this.CONTENT_TYPE = "Content-Type"; }
|
|
44
|
+
static { this.SEC_GOV_CLASS_HEADER = "SecurityGovernorClass"; }
|
|
45
|
+
static { this.SEC_GOV_ID_HEADER = "SecurityGovernorId"; }
|
|
46
|
+
static { this.DEIDENT_HEADER = "DeidentifiedContext"; }
|
|
47
|
+
static { this.LIMITED_HEADER = "LimitedContext"; }
|
|
48
|
+
constructor(_http, _router, _localStorageService, _jwtHelper, authenticationProvider, _authenticationRoute, _logoutPath, _tokenEndpoint, _serverUrl, _directEndpoint, _maxInactivity, _userCountdownSeconds, _idpInactivityMinutes, locationStrategy) {
|
|
49
|
+
this._http = _http;
|
|
50
|
+
this._router = _router;
|
|
51
|
+
this._localStorageService = _localStorageService;
|
|
52
|
+
this._jwtHelper = _jwtHelper;
|
|
53
|
+
this.authenticationProvider = authenticationProvider;
|
|
54
|
+
this._authenticationRoute = _authenticationRoute;
|
|
55
|
+
this._logoutPath = _logoutPath;
|
|
56
|
+
this._tokenEndpoint = _tokenEndpoint;
|
|
57
|
+
this._serverUrl = _serverUrl;
|
|
58
|
+
this._directEndpoint = _directEndpoint;
|
|
59
|
+
this._maxInactivity = _maxInactivity;
|
|
60
|
+
this._userCountdownSeconds = _userCountdownSeconds;
|
|
61
|
+
this._idpInactivityMinutes = _idpInactivityMinutes;
|
|
62
|
+
this.locationStrategy = locationStrategy;
|
|
63
|
+
this.userCountdownSeconds = 60;
|
|
64
|
+
this.idpInactivityMinutes = 5;
|
|
65
|
+
this.contentType = "application/json";
|
|
66
|
+
this.limitedContext = false;
|
|
67
|
+
this.deidentifiedContext = false;
|
|
68
|
+
this.maxViewPermission = new BehaviorSubject("viewident");
|
|
69
|
+
this._isAuthenticatedSubject = new BehaviorSubject(false);
|
|
70
|
+
this._userIsAboutToTimeOut = new BehaviorSubject(false);
|
|
71
|
+
this._maxInactivityMinutes = 120;
|
|
72
|
+
this.contextRoot = "";
|
|
73
|
+
if (isDevMode()) {
|
|
74
|
+
console.debug("window.location.href: " + window.location.href);
|
|
75
|
+
}
|
|
76
|
+
if (window.location) {
|
|
77
|
+
let parts = window.location.href.split("/");
|
|
78
|
+
this.baseUrl = parts[0] + "//" + parts[2];
|
|
79
|
+
if (parts.length > 3) {
|
|
80
|
+
this.contextRoot = parts[3];
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
if (this._localStorageService.getItem("maxViewPermission")) {
|
|
84
|
+
this.maxViewPermission.next(this._localStorageService.getItem("maxViewPermission"));
|
|
85
|
+
}
|
|
86
|
+
if (_maxInactivity) {
|
|
87
|
+
this._maxInactivityMinutes = _maxInactivity;
|
|
88
|
+
}
|
|
89
|
+
if (_userCountdownSeconds) {
|
|
90
|
+
this.userCountdownSeconds = _userCountdownSeconds;
|
|
91
|
+
}
|
|
92
|
+
if (_idpInactivityMinutes) {
|
|
93
|
+
this.idpInactivityMinutes = _idpInactivityMinutes;
|
|
94
|
+
}
|
|
95
|
+
this.hasValidConfig();
|
|
96
|
+
//There could be a non-expired token in local storage.
|
|
97
|
+
let token = this.authenticationProvider.authToken;
|
|
98
|
+
this.storeToken(token);
|
|
99
|
+
}
|
|
100
|
+
getBaseUrl() {
|
|
101
|
+
return (this.baseUrl) ? this.baseUrl : "";
|
|
102
|
+
}
|
|
103
|
+
getContextRoot() {
|
|
104
|
+
return this.contextRoot;
|
|
105
|
+
}
|
|
106
|
+
getHeaders(req) {
|
|
107
|
+
let headers = req.headers;
|
|
108
|
+
//Don't set content type if already set
|
|
109
|
+
if (!req.headers.get(AuthenticationService.CONTENT_TYPE)) {
|
|
110
|
+
headers = headers.set(AuthenticationService.CONTENT_TYPE, this.contentType.toString());
|
|
111
|
+
}
|
|
112
|
+
if (headers.get(AuthenticationService.SEC_GOV_CLASS_HEADER) === "") {
|
|
113
|
+
headers = headers.delete(AuthenticationService.SEC_GOV_CLASS_HEADER);
|
|
114
|
+
}
|
|
115
|
+
else if (this.securityGovernorClass && !headers.get(AuthenticationService.SEC_GOV_CLASS_HEADER)) {
|
|
116
|
+
headers = headers.set(AuthenticationService.SEC_GOV_CLASS_HEADER, this.securityGovernorClass);
|
|
117
|
+
}
|
|
118
|
+
if (headers.get(AuthenticationService.SEC_GOV_ID_HEADER) === "") {
|
|
119
|
+
headers = headers.delete(AuthenticationService.SEC_GOV_ID_HEADER);
|
|
120
|
+
}
|
|
121
|
+
else if (this.securityGovernorId && !headers.get(AuthenticationService.SEC_GOV_ID_HEADER)) {
|
|
122
|
+
headers = headers.set(AuthenticationService.SEC_GOV_ID_HEADER, this.securityGovernorId.toString());
|
|
123
|
+
}
|
|
124
|
+
headers = headers.set(AuthenticationService.DEIDENT_HEADER, this.deidentifiedContext.toString());
|
|
125
|
+
headers = headers.set(AuthenticationService.LIMITED_HEADER, this.limitedContext.toString());
|
|
126
|
+
return headers;
|
|
127
|
+
}
|
|
128
|
+
get authenticationTokenKey() {
|
|
129
|
+
return this.authenticationProvider.authenticationTokenKey;
|
|
130
|
+
}
|
|
131
|
+
get authToken() {
|
|
132
|
+
return this.authenticationProvider.authToken;
|
|
133
|
+
}
|
|
134
|
+
updateUserActivity() {
|
|
135
|
+
if (this._isAuthenticatedSubject.value) {
|
|
136
|
+
this._lastUserInteraction = new Date();
|
|
137
|
+
this._userIsAboutToTimeOut.next(false);
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
/**
|
|
141
|
+
* A mutator for identifying the clients original request location. Setting this value will influence the end location
|
|
142
|
+
* navigated to by {@link #navigateToPath}.
|
|
143
|
+
*
|
|
144
|
+
* @param redirectUrl location of the users request before authentication
|
|
145
|
+
*/
|
|
146
|
+
set redirectUrl(redirectUrl) {
|
|
147
|
+
this._redirectUrl = redirectUrl;
|
|
148
|
+
}
|
|
149
|
+
get redirectUrl() {
|
|
150
|
+
return this._redirectUrl;
|
|
151
|
+
}
|
|
152
|
+
requestAccessToken(redirectOnSuccess) {
|
|
153
|
+
this._http.get(this.tokenLocation(), { withCredentials: true })
|
|
154
|
+
.subscribe((response) => {
|
|
155
|
+
this.storeToken(response.auth_token);
|
|
156
|
+
if (redirectOnSuccess) {
|
|
157
|
+
this.proceedIfAuthenticated();
|
|
158
|
+
}
|
|
159
|
+
}, (error) => {
|
|
160
|
+
//Token refresh failed.
|
|
161
|
+
this.logout(true);
|
|
162
|
+
});
|
|
163
|
+
}
|
|
164
|
+
/**
|
|
165
|
+
* Verifies whether or not a current user session exists.
|
|
166
|
+
*
|
|
167
|
+
* @returns {Observable<boolean>} evaluates to true if the user is authenticated, false otherwise.
|
|
168
|
+
*/
|
|
169
|
+
isAuthenticated() {
|
|
170
|
+
return this._isAuthenticatedSubject.asObservable();
|
|
171
|
+
}
|
|
172
|
+
isAboutToTimeOut() {
|
|
173
|
+
return this._userIsAboutToTimeOut.asObservable();
|
|
174
|
+
}
|
|
175
|
+
getTimeoutStart() {
|
|
176
|
+
if (this._lastUserInteraction) {
|
|
177
|
+
return this._lastUserInteraction.valueOf() + (((this._maxInactivityMinutes * 60) - this.userCountdownSeconds) * 1000);
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
tokenLocation() {
|
|
181
|
+
if (this._serverUrl) {
|
|
182
|
+
return this._serverUrl + this._tokenEndpoint;
|
|
183
|
+
}
|
|
184
|
+
else {
|
|
185
|
+
return this._tokenEndpoint;
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
directLoginLocation() {
|
|
189
|
+
if (this._serverUrl) {
|
|
190
|
+
return this._serverUrl + this._directEndpoint;
|
|
191
|
+
}
|
|
192
|
+
else {
|
|
193
|
+
return this._directEndpoint;
|
|
194
|
+
}
|
|
195
|
+
}
|
|
196
|
+
logoutLocation() {
|
|
197
|
+
if (this._serverUrl) {
|
|
198
|
+
return this._serverUrl + this._logoutPath;
|
|
199
|
+
}
|
|
200
|
+
else {
|
|
201
|
+
return this._logoutPath;
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
/**
|
|
205
|
+
* A function to authenticated the user with the provided credentials. Failure results in an error that describes the
|
|
206
|
+
* server response (status and status message) and should be actionable by the client application.
|
|
207
|
+
*
|
|
208
|
+
* @param username of the authenticating user to verify
|
|
209
|
+
* @param password of the authenticating user to verify
|
|
210
|
+
* @returns {Observable<R>} describing the result of the login action, true or an error
|
|
211
|
+
*/
|
|
212
|
+
login(_username, _password) {
|
|
213
|
+
return this._http.post(this.directLoginLocation(), { username: _username, password: _password }, { observe: "response" }).pipe(map((resp) => {
|
|
214
|
+
if (resp.status === 201) {
|
|
215
|
+
return true;
|
|
216
|
+
}
|
|
217
|
+
else {
|
|
218
|
+
throw new Error("Authentication failed. " + resp.status + ": " + resp.statusText);
|
|
219
|
+
}
|
|
220
|
+
}), catchError(this.handleError));
|
|
221
|
+
}
|
|
222
|
+
clearLogin() {
|
|
223
|
+
//Front-end logout
|
|
224
|
+
try {
|
|
225
|
+
this._localStorageService.removeItem(this.authenticationProvider.authenticationTokenKey);
|
|
226
|
+
this.unsubscribeFromTokenRefresh();
|
|
227
|
+
this._isAuthenticatedSubject.next(false);
|
|
228
|
+
this._userIsAboutToTimeOut.next(false);
|
|
229
|
+
}
|
|
230
|
+
catch (Error) {
|
|
231
|
+
}
|
|
232
|
+
//Back-end logout
|
|
233
|
+
let headers = new HttpHeaders().set(AuthenticationService.CONTENT_TYPE, "text/plain");
|
|
234
|
+
return this._http.get(this.logoutLocation(), { headers: headers });
|
|
235
|
+
}
|
|
236
|
+
/**
|
|
237
|
+
* A function to signal the termination of the current session. Invoking this function will clean up any relevant state
|
|
238
|
+
* related to the last active session.
|
|
239
|
+
*/
|
|
240
|
+
logout(keepCurrentRoute = false) {
|
|
241
|
+
//Prevent logout if already on authentication route. Doing otherwise screws up SAML
|
|
242
|
+
if (!this._router.routerState || this._router.routerState.snapshot.url !== this._authenticationRoute) {
|
|
243
|
+
this._redirectUrl = (keepCurrentRoute && this._router.routerState && this._router.routerState.snapshot) ? this._router.routerState.snapshot.url : "";
|
|
244
|
+
if (this._redirectUrl.startsWith("/")) {
|
|
245
|
+
this._redirectUrl = this._redirectUrl.substring(1);
|
|
246
|
+
}
|
|
247
|
+
this.clearLogin().subscribe((response) => {
|
|
248
|
+
window.location.replace(this._redirectUrl);
|
|
249
|
+
}, (error) => {
|
|
250
|
+
window.location.replace(this._redirectUrl);
|
|
251
|
+
});
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
storeToken(token) {
|
|
255
|
+
let valid = this.validateToken(token);
|
|
256
|
+
// unsubscribe from refesh before we decide wether to resubscribe
|
|
257
|
+
this.unsubscribeFromTokenRefresh();
|
|
258
|
+
if (valid) {
|
|
259
|
+
this._localStorageService.setItem(this.authenticationProvider.authenticationTokenKey, token);
|
|
260
|
+
this.subscribeToTokenRefresh(token);
|
|
261
|
+
//Change the BehaviorSubject if the user was not previously authenticated.
|
|
262
|
+
//Since other code may be subscribing to this observable, we don't want to cause new events to fire if just refreshing the JWT.
|
|
263
|
+
if (!this._isAuthenticatedSubject.value) {
|
|
264
|
+
this._isAuthenticatedSubject.next(true);
|
|
265
|
+
}
|
|
266
|
+
}
|
|
267
|
+
else {
|
|
268
|
+
this._localStorageService.removeItem(this.authenticationProvider.authenticationTokenKey);
|
|
269
|
+
this._isAuthenticatedSubject.next(false);
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
proceedIfAuthenticated() {
|
|
273
|
+
if (isDevMode()) {
|
|
274
|
+
console.debug("AuthenticationService.proceedIfAuthenticated: " + this._redirectUrl);
|
|
275
|
+
}
|
|
276
|
+
if (this._isAuthenticatedSubject.value) {
|
|
277
|
+
//Login counts as user activity, too
|
|
278
|
+
this.updateUserActivity();
|
|
279
|
+
if (this._redirectUrl && this._redirectUrl && this._redirectUrl !== "") {
|
|
280
|
+
this._router.navigateByUrl(this._redirectUrl);
|
|
281
|
+
}
|
|
282
|
+
else {
|
|
283
|
+
this._router.navigate([""]);
|
|
284
|
+
}
|
|
285
|
+
return true;
|
|
286
|
+
}
|
|
287
|
+
else {
|
|
288
|
+
return false;
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
validateToken(token) {
|
|
292
|
+
return (token && !this._jwtHelper.isTokenExpired(token));
|
|
293
|
+
}
|
|
294
|
+
subscribeToTokenRefresh(token) {
|
|
295
|
+
let exp = this._jwtHelper.getTokenExpirationDate(token);
|
|
296
|
+
// Use a timer to periodically check timeouts
|
|
297
|
+
this._refreshSubscription = interval(1000)
|
|
298
|
+
.subscribe(() => {
|
|
299
|
+
// If a tab is inactive we can't know if our timer is accurate
|
|
300
|
+
// so when the interval hits check against timestamps
|
|
301
|
+
if (this._isAuthenticatedSubject.value && Date.now() > this.getTimeoutStart()) {
|
|
302
|
+
//Don't update the subject more than once! Doing so initializes more than one countdown timer!
|
|
303
|
+
if (this._userIsAboutToTimeOut.getValue() !== true) {
|
|
304
|
+
this._userIsAboutToTimeOut.next(true);
|
|
305
|
+
}
|
|
306
|
+
}
|
|
307
|
+
// check for refresh token
|
|
308
|
+
let msToExpiry = (exp.valueOf() - new Date().valueOf());
|
|
309
|
+
// Refresh 60 seconds before expiry
|
|
310
|
+
if (msToExpiry <= 60000) {
|
|
311
|
+
this.refreshTokenIfUserIsActive();
|
|
312
|
+
}
|
|
313
|
+
});
|
|
314
|
+
}
|
|
315
|
+
unsubscribeFromTokenRefresh() {
|
|
316
|
+
if (this._refreshSubscription && !this._refreshSubscription.closed) {
|
|
317
|
+
this._refreshSubscription.unsubscribe();
|
|
318
|
+
}
|
|
319
|
+
}
|
|
320
|
+
getMaxViewPermission() {
|
|
321
|
+
return this.maxViewPermission.getValue();
|
|
322
|
+
}
|
|
323
|
+
getMaxViewPermissionSubject() {
|
|
324
|
+
return this.maxViewPermission;
|
|
325
|
+
}
|
|
326
|
+
setMaxViewPermission(maxViewPermission) {
|
|
327
|
+
this._localStorageService.setItem("maxViewPermission", maxViewPermission);
|
|
328
|
+
this.maxViewPermission.next(maxViewPermission);
|
|
329
|
+
}
|
|
330
|
+
refreshTokenIfUserIsActive() {
|
|
331
|
+
//Only refresh if the user has been active
|
|
332
|
+
if (this._lastUserInteraction && ((new Date().valueOf() - this._lastUserInteraction.valueOf()) <= (this._maxInactivityMinutes * 60 * 1000))) {
|
|
333
|
+
this.requestAccessToken(false);
|
|
334
|
+
}
|
|
335
|
+
}
|
|
336
|
+
hasValidConfig() {
|
|
337
|
+
if (this._tokenEndpoint == null && (this._serverUrl === null || this._logoutPath === null)) {
|
|
338
|
+
throw new Error("BUG ALERT! Invalid AuthenticationService configuration. No valid configuration for authentication endpoint(s).");
|
|
339
|
+
}
|
|
340
|
+
if (this._localStorageService === null || this.authenticationProvider.authenticationTokenKey === null) {
|
|
341
|
+
throw new Error("BUG ALERT! Invalid AuthenticationService configuration. No valid configuration for local storage");
|
|
342
|
+
}
|
|
343
|
+
}
|
|
344
|
+
handleError(error) {
|
|
345
|
+
let errMsg = (error.message) ? error.message : AuthenticationService.GENERIC_ERR_MSG;
|
|
346
|
+
return throwError(errMsg);
|
|
347
|
+
}
|
|
348
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.2", 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 }); }
|
|
349
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: AuthenticationService }); }
|
|
350
|
+
}
|
|
351
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: AuthenticationService, decorators: [{
|
|
352
|
+
type: Injectable
|
|
353
|
+
}], ctorParameters: () => [{ type: i1.HttpClient }, { type: i2.Router }, { type: i3.CoolLocalStorage }, { type: i4.JwtHelperService }, { type: i5.AuthenticationProvider }, { type: undefined, decorators: [{
|
|
354
|
+
type: Inject,
|
|
355
|
+
args: [AUTHENTICATION_ROUTE]
|
|
356
|
+
}] }, { type: undefined, decorators: [{
|
|
357
|
+
type: Inject,
|
|
358
|
+
args: [AUTHENTICATION_LOGOUT_PATH]
|
|
359
|
+
}] }, { type: undefined, decorators: [{
|
|
360
|
+
type: Inject,
|
|
361
|
+
args: [AUTHENTICATION_TOKEN_ENDPOINT]
|
|
362
|
+
}] }, { type: undefined, decorators: [{
|
|
363
|
+
type: Optional
|
|
364
|
+
}, {
|
|
365
|
+
type: Inject,
|
|
366
|
+
args: [AUTHENTICATION_SERVER_URL]
|
|
367
|
+
}] }, { type: undefined, decorators: [{
|
|
368
|
+
type: Optional
|
|
369
|
+
}, {
|
|
370
|
+
type: Inject,
|
|
371
|
+
args: [AUTHENTICATION_DIRECT_ENDPOINT]
|
|
372
|
+
}] }, { type: undefined, decorators: [{
|
|
373
|
+
type: Optional
|
|
374
|
+
}, {
|
|
375
|
+
type: Inject,
|
|
376
|
+
args: [AUTHENTICATION_MAX_INACTIVITY_MINUTES]
|
|
377
|
+
}] }, { type: undefined, decorators: [{
|
|
378
|
+
type: Optional
|
|
379
|
+
}, {
|
|
380
|
+
type: Inject,
|
|
381
|
+
args: [AUTHENTICATION_USER_COUNTDOWN_SECONDS]
|
|
382
|
+
}] }, { type: undefined, decorators: [{
|
|
383
|
+
type: Optional
|
|
384
|
+
}, {
|
|
385
|
+
type: Inject,
|
|
386
|
+
args: [AUTHENTICATION_IDP_INACTIVITY_MINUTES]
|
|
387
|
+
}] }, { type: i6.LocationStrategy, decorators: [{
|
|
388
|
+
type: Optional
|
|
389
|
+
}, {
|
|
390
|
+
type: Inject,
|
|
391
|
+
args: [LocationStrategy]
|
|
392
|
+
}] }] });
|
|
393
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"authentication.service.js","sourceRoot":"","sources":["../../../projects/authentication/src/authentication.service.ts"],"names":[],"mappings":"AAAA;;GAEG;AACH,OAAO,EAAC,UAAU,EAAE,cAAc,EAAE,MAAM,EAAE,QAAQ,EAAE,SAAS,EAAC,MAAM,eAAe,CAAC;AACtF,OAAO,EAAC,gBAAgB,EAAC,MAAM,iBAAiB,CAAC;AACjD,OAAO,EAAC,MAAM,EAAC,MAAM,iBAAiB,CAAC;AACvC,OAAO,EAAC,UAAU,EAAE,WAAW,EAA4B,MAAM,sBAAsB,CAAC;AAExF,OAAO,EAAC,QAAQ,EAAc,eAAe,EAAgB,UAAU,EAAK,MAAM,MAAM,CAAC;AACzF,OAAO,EAAC,UAAU,EAAS,GAAG,EAAC,MAAM,gBAAgB,CAAC;AACtD,OAAO,EAAC,gBAAgB,EAAC,MAAM,oBAAoB,CAAC;AAEpD,OAAO,EAAC,sBAAsB,EAAC,MAAM,2BAA2B,CAAC;AACjE,OAAO,EAAE,gBAAgB,EAAE,MAAM,uBAAuB,CAAC;;;;;;;;AAEzD;;;;GAIG;AACH,MAAM,CAAC,IAAI,yBAAyB,GAAG,IAAI,cAAc,CAAS,gCAAgC,CAAC,CAAC;AACpG,MAAM,CAAC,IAAI,0BAA0B,GAAG,IAAI,cAAc,CAAS,4BAA4B,CAAC,CAAC;AACjG,MAAM,CAAC,IAAI,8BAA8B,GAAG,IAAI,cAAc,CAAS,gCAAgC,CAAC,CAAC;AACzG,MAAM,CAAC,IAAI,6BAA6B,GAAG,IAAI,cAAc,CAAS,+BAA+B,CAAC,CAAC;AACvG,MAAM,CAAC,IAAI,oBAAoB,GAAG,IAAI,cAAc,CAAS,sBAAsB,CAAC,CAAC;AACrF,MAAM,CAAC,IAAI,qCAAqC,GAAG,IAAI,cAAc,CAAS,+BAA+B,CAAC,CAAC;AAC/G,MAAM,CAAC,IAAI,qCAAqC,GAAG,IAAI,cAAc,CAAS,uCAAuC,CAAC,CAAC;AACvH,MAAM,CAAC,IAAI,qCAAqC,GAAG,IAAI,cAAc,CAAS,uCAAuC,CAAC,CAAC;AAEvH;;GAEG;AAEH,MAAM,OAAO,qBAAqB;IAEhC;;;;OAIG;aACW,oBAAe,GAAW,cAAc,AAAzB,CAA0B;aAExC,iBAAY,GAAW,cAAc,AAAzB,CAA0B;aACtC,yBAAoB,GAAW,uBAAuB,AAAlC,CAAmC;aACvD,sBAAiB,GAAW,oBAAoB,AAA/B,CAAgC;aACjD,mBAAc,GAAW,qBAAqB,AAAhC,CAAiC;aAC/C,mBAAc,GAAW,gBAAgB,AAA3B,CAA4B;IAsBzD,YAAoB,KAAiB,EACjB,OAAe,EACf,oBAAsC,EACtC,UAA4B,EAC5B,sBAA8C,EAChB,oBAA4B,EACtB,WAAmB,EAChB,cAAsB,EACd,UAAkB,EACb,eAAuB,EAChB,cAAsB,EACtB,qBAA6B,EAC7B,qBAA6B,EAClD,gBAAkC;QAbxE,UAAK,GAAL,KAAK,CAAY;QACjB,YAAO,GAAP,OAAO,CAAQ;QACf,yBAAoB,GAApB,oBAAoB,CAAkB;QACtC,eAAU,GAAV,UAAU,CAAkB;QAC5B,2BAAsB,GAAtB,sBAAsB,CAAwB;QAChB,yBAAoB,GAApB,oBAAoB,CAAQ;QACtB,gBAAW,GAAX,WAAW,CAAQ;QAChB,mBAAc,GAAd,cAAc,CAAQ;QACd,eAAU,GAAV,UAAU,CAAQ;QACb,oBAAe,GAAf,eAAe,CAAQ;QAChB,mBAAc,GAAd,cAAc,CAAQ;QACtB,0BAAqB,GAArB,qBAAqB,CAAQ;QAC7B,0BAAqB,GAArB,qBAAqB,CAAQ;QAClD,qBAAgB,GAAhB,gBAAgB,CAAkB;QAjCrF,yBAAoB,GAAW,EAAE,CAAC;QAClC,yBAAoB,GAAW,CAAC,CAAC;QAEjC,gBAAW,GAAW,kBAAkB,CAAC;QAGzC,mBAAc,GAAY,KAAK,CAAC;QAChC,wBAAmB,GAAY,KAAK,CAAC;QAEpC,sBAAiB,GAA0D,IAAI,eAAe,CAAuC,WAAW,CAAC,CAAC;QAClJ,4BAAuB,GAA6B,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QACxF,0BAAqB,GAA6B,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QAItF,0BAAqB,GAAW,GAAG,CAAC;QAGpC,gBAAW,GAAW,EAAE,CAAC;QAgB/B,IAAI,SAAS,EAAE,EAAE,CAAC;YAChB,OAAO,CAAC,KAAK,CAAC,wBAAwB,GAAG,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;QACjE,CAAC;QAED,IAAI,MAAM,CAAC,QAAQ,EAAE,CAAC;YACpB,IAAI,KAAK,GAAa,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YACtD,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC,CAAC,CAAC,GAAG,IAAI,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC;YAC1C,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACrB,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC;YAC9B,CAAC;QACH,CAAC;QAED,IAAI,IAAI,CAAC,oBAAoB,CAAC,OAAO,CAAC,mBAAmB,CAAC,EAAE,CAAC;YAC3D,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAuC,IAAI,CAAC,oBAAoB,CAAC,OAAO,CAAC,mBAAmB,CAAC,CAAC,CAAC;QAC5H,CAAC;QAED,IAAI,cAAc,EAAE,CAAC;YACnB,IAAI,CAAC,qBAAqB,GAAG,cAAc,CAAC;QAC9C,CAAC;QAED,IAAI,qBAAqB,EAAE,CAAC;YAC1B,IAAI,CAAC,oBAAoB,GAAG,qBAAqB,CAAC;QACpD,CAAC;QAED,IAAI,qBAAqB,EAAE,CAAC;YAC1B,IAAI,CAAC,oBAAoB,GAAG,qBAAqB,CAAC;QACpD,CAAC;QAED,IAAI,CAAC,cAAc,EAAE,CAAC;QAEtB,sDAAsD;QACtD,IAAI,KAAK,GAAW,IAAI,CAAC,sBAAsB,CAAC,SAAS,CAAC;QAC1D,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;IACzB,CAAC;IAED,UAAU;QACR,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC;IAC5C,CAAC;IAED,cAAc;QACZ,OAAO,IAAI,CAAC,WAAW,CAAC;IAC1B,CAAC;IAED,UAAU,CAAC,GAAqB;QAC9B,IAAI,OAAO,GAAgB,GAAG,CAAC,OAAO,CAAC;QAEvC,uCAAuC;QACvC,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,YAAY,CAAC,EAAE,CAAC;YACzD,OAAO,GAAG,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,YAAY,EAAE,IAAI,CAAC,WAAW,CAAC,QAAQ,EAAE,CAAC,CAAC;QACzF,CAAC;QAED,IAAI,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,oBAAoB,CAAC,KAAK,EAAE,EAAE,CAAC;YACnE,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,qBAAqB,CAAC,oBAAoB,CAAC,CAAC;QACvE,CAAC;aAAM,IAAI,IAAI,CAAC,qBAAqB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,oBAAoB,CAAC,EAAE,CAAC;YAClG,OAAO,GAAG,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,oBAAoB,EAAE,IAAI,CAAC,qBAAqB,CAAC,CAAC;QAChG,CAAC;QAED,IAAI,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,iBAAiB,CAAC,KAAK,EAAE,EAAE,CAAC;YAChE,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,qBAAqB,CAAC,iBAAiB,CAAC,CAAC;QACpE,CAAC;aAAM,IAAI,IAAI,CAAC,kBAAkB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,iBAAiB,CAAC,EAAE,CAAC;YAC5F,OAAO,GAAG,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,iBAAiB,EAAE,IAAI,CAAC,kBAAkB,CAAC,QAAQ,EAAE,CAAC,CAAC;QACrG,CAAC;QAED,OAAO,GAAG,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,cAAc,EAAE,IAAI,CAAC,mBAAmB,CAAC,QAAQ,EAAE,CAAC,CAAC;QACjG,OAAO,GAAG,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,cAAc,EAAE,IAAI,CAAC,cAAc,CAAC,QAAQ,EAAE,CAAC,CAAC;QAE5F,OAAO,OAAO,CAAC;IACjB,CAAC;IAED,IAAI,sBAAsB;QACxB,OAAO,IAAI,CAAC,sBAAsB,CAAC,sBAAsB,CAAC;IAC5D,CAAC;IAED,IAAI,SAAS;QACX,OAAO,IAAI,CAAC,sBAAsB,CAAC,SAAS,CAAC;IAC/C,CAAC;IAEM,kBAAkB;QACvB,IAAI,IAAI,CAAC,uBAAuB,CAAC,KAAK,EAAE,CAAC;YACvC,IAAI,CAAC,oBAAoB,GAAG,IAAI,IAAI,EAAE,CAAC;YACvC,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACzC,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACH,IAAI,WAAW,CAAC,WAAmB;QACjC,IAAI,CAAC,YAAY,GAAG,WAAW,CAAC;IAClC,CAAC;IAED,IAAI,WAAW;QACb,OAAO,IAAI,CAAC,YAAY,CAAC;IAC3B,CAAC;IAED,kBAAkB,CAAC,iBAA0B;QAE3C,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,aAAa,EAAE,EAAE,EAAC,eAAe,EAAE,IAAI,EAAC,CAAC;aAC1D,SAAS,CACR,CAAC,QAAa,EAAE,EAAE;YAChB,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;YACrC,IAAI,iBAAiB,EAAE,CAAC;gBACtB,IAAI,CAAC,sBAAsB,EAAE,CAAC;YAChC,CAAC;QACH,CAAC,EACD,CAAC,KAAK,EAAE,EAAE;YACR,uBAAuB;YACvB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QACpB,CAAC,CACF,CAAC;IACN,CAAC;IAED;;;;OAIG;IACH,eAAe;QACb,OAAO,IAAI,CAAC,uBAAuB,CAAC,YAAY,EAAE,CAAC;IACrD,CAAC;IAED,gBAAgB;QACd,OAAO,IAAI,CAAC,qBAAqB,CAAC,YAAY,EAAE,CAAC;IACnD,CAAC;IAED,eAAe;QACb,IAAI,IAAI,CAAC,oBAAoB,EAAE,CAAC;YAC9B,OAAO,IAAI,CAAC,oBAAoB,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,qBAAqB,GAAG,EAAE,CAAC,GAAG,IAAI,CAAC,oBAAoB,CAAC,GAAG,IAAI,CAAC,CAAC;QACxH,CAAC;IACH,CAAC;IAED,aAAa;QACX,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YACpB,OAAO,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,cAAc,CAAC;QAC/C,CAAC;aAAM,CAAC;YACN,OAAO,IAAI,CAAC,cAAc,CAAC;QAC7B,CAAC;IACH,CAAC;IAED,mBAAmB;QACjB,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YACpB,OAAO,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,eAAe,CAAC;QAChD,CAAC;aAAM,CAAC;YACN,OAAO,IAAI,CAAC,eAAe,CAAC;QAC9B,CAAC;IACH,CAAC;IAED,cAAc;QACZ,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YACpB,OAAO,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,WAAW,CAAC;QAC5C,CAAC;aAAM,CAAC;YACN,OAAO,IAAI,CAAC,WAAW,CAAC;QAC1B,CAAC;IACH,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,SAAiB,EAAE,SAAiB;QACxC,OAAO,IAAI,CAAC,KAAK,CAAC,IAAI,CACpB,IAAI,CAAC,mBAAmB,EAAE,EAC1B,EAAC,QAAQ,EAAE,SAAS,EAAE,QAAQ,EAAE,SAAS,EAAC,EAC1C,EAAC,OAAO,EAAE,UAAU,EAAC,CACtB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAuB,EAAE,EAAE;YACrC,IAAI,IAAI,CAAC,MAAM,KAAK,GAAG,EAAE,CAAC;gBACxB,OAAO,IAAI,CAAC;YACd,CAAC;iBAAM,CAAC;gBACN,MAAM,IAAI,KAAK,CAAC,yBAAyB,GAAG,IAAI,CAAC,MAAM,GAAG,IAAI,GAAG,IAAI,CAAC,UAAU,CAAC,CAAC;YACpF,CAAC;QACH,CAAC,CAAC,EACA,UAAU,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC;IAClC,CAAC;IAGD,UAAU;QACR,kBAAkB;QAClB,IAAI,CAAC;YACH,IAAI,CAAC,oBAAoB,CAAC,UAAU,CAAC,IAAI,CAAC,sBAAsB,CAAC,sBAAsB,CAAC,CAAC;YACzF,IAAI,CAAC,2BAA2B,EAAE,CAAC;YACnC,IAAI,CAAC,uBAAuB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YACzC,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACzC,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;QACjB,CAAC;QAED,iBAAiB;QACjB,IAAI,OAAO,GAAG,IAAI,WAAW,EAAE,CAAC,GAAG,CAAC,qBAAqB,CAAC,YAAY,EAAE,YAAY,CAAC,CAAC;QACtF,OAA6B,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,cAAc,EAAE,EAAE,EAAC,OAAO,EAAE,OAAO,EAAC,CAAC,CAAC;IACzF,CAAC;IAED;;;OAGG;IACH,MAAM,CAAC,mBAA4B,KAAK;QACtC,mFAAmF;QACnF,IAAI,CAAE,IAAI,CAAC,OAAO,CAAC,WAAW,IAAI,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,QAAQ,CAAC,GAAG,KAAK,IAAI,CAAC,oBAAoB,EAAE,CAAC;YACtG,IAAI,CAAC,YAAY,GAAG,CAAC,gBAAgB,IAAI,IAAI,CAAC,OAAO,CAAC,WAAW,IAAI,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;YAErJ,IAAI,IAAI,CAAC,YAAY,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;gBACtC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC;YACrD,CAAC;YAED,IAAI,CAAC,UAAU,EAAE,CAAC,SAAS,CACzB,CAAC,QAAQ,EAAE,EAAE;gBACX,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YAC3C,CAAC,EACH,CAAC,KAAK,EAAE,EAAE;gBACR,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YAC7C,CAAC,CACF,CAAC;QACJ,CAAC;IACH,CAAC;IAED,UAAU,CAAC,KAAa;QACtB,IAAI,KAAK,GAAG,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;QAEtC,iEAAiE;QACjE,IAAI,CAAC,2BAA2B,EAAE,CAAC;QAEnC,IAAI,KAAK,EAAE,CAAC;YACV,IAAI,CAAC,oBAAoB,CAAC,OAAO,CAAC,IAAI,CAAC,sBAAsB,CAAC,sBAAsB,EAAE,KAAK,CAAC,CAAC;YAC7F,IAAI,CAAC,uBAAuB,CAAC,KAAK,CAAC,CAAC;YAEpC,0EAA0E;YAC1E,+HAA+H;YAC/H,IAAI,CAAE,IAAI,CAAC,uBAAuB,CAAC,KAAK,EAAE,CAAC;gBACzC,IAAI,CAAC,uBAAuB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC1C,CAAC;QACH,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,oBAAoB,CAAC,UAAU,CAAC,IAAI,CAAC,sBAAsB,CAAC,sBAAsB,CAAC,CAAC;YACzF,IAAI,CAAC,uBAAuB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC3C,CAAC;IACH,CAAC;IAED,sBAAsB;QACpB,IAAI,SAAS,EAAE,EAAE,CAAC;YAChB,OAAO,CAAC,KAAK,CAAC,gDAAgD,GAAG,IAAI,CAAC,YAAY,CAAC,CAAC;QACtF,CAAC;QAED,IAAI,IAAI,CAAC,uBAAuB,CAAC,KAAK,EAAE,CAAC;YACvC,oCAAoC;YACpC,IAAI,CAAC,kBAAkB,EAAE,CAAC;YAE1B,IAAI,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,YAAY,KAAK,EAAE,EAAE,CAAC;gBACvE,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YAChD,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;YAC9B,CAAC;YAED,OAAO,IAAI,CAAC;QACd,CAAC;aAAM,CAAC;YACN,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IAED,aAAa,CAAC,KAAa;QACzB,OAAO,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC,CAAC;IAC3D,CAAC;IAED,uBAAuB,CAAC,KAAU;QAChC,IAAI,GAAG,GAAG,IAAI,CAAC,UAAU,CAAC,sBAAsB,CAAC,KAAK,CAAC,CAAC;QAExD,6CAA6C;QAC7C,IAAI,CAAC,oBAAoB,GAAG,QAAQ,CAAC,IAAI,CAAC;aACvC,SAAS,CAAC,GAAG,EAAE;YAEd,8DAA8D;YAC9D,qDAAqD;YACrD,IAAI,IAAI,CAAC,uBAAuB,CAAC,KAAK,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,eAAe,EAAE,EAAE,CAAC;gBAC9E,8FAA8F;gBAC9F,IAAI,IAAI,CAAC,qBAAqB,CAAC,QAAQ,EAAE,KAAK,IAAI,EAAE,CAAC;oBACnD,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBACxC,CAAC;YACH,CAAC;YAED,0BAA0B;YAC1B,IAAI,UAAU,GAAG,CAAC,GAAG,CAAC,OAAO,EAAE,GAAG,IAAI,IAAI,EAAE,CAAC,OAAO,EAAE,CAAC,CAAC;YAExD,mCAAmC;YACnC,IAAI,UAAU,IAAI,KAAK,EAAE,CAAC;gBACxB,IAAI,CAAC,0BAA0B,EAAE,CAAC;YACpC,CAAC;QACH,CAAC,CAAC,CAAC;IACP,CAAC;IAED,2BAA2B;QACzB,IAAI,IAAI,CAAC,oBAAoB,IAAI,CAAE,IAAI,CAAC,oBAAoB,CAAC,MAAM,EAAE,CAAC;YACpE,IAAI,CAAC,oBAAoB,CAAC,WAAW,EAAE,CAAC;QAC1C,CAAC;IACH,CAAC;IAED,oBAAoB;QAClB,OAAO,IAAI,CAAC,iBAAiB,CAAC,QAAQ,EAAE,CAAC;IAC3C,CAAC;IAED,2BAA2B;QACzB,OAAO,IAAI,CAAC,iBAAiB,CAAC;IAChC,CAAC;IAED,oBAAoB,CAAC,iBAAuD;QAC1E,IAAI,CAAC,oBAAoB,CAAC,OAAO,CAAC,mBAAmB,EAAE,iBAAiB,CAAC,CAAC;QAC1E,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;IACjD,CAAC;IAEO,0BAA0B;QAChC,0CAA0C;QAC1C,IAAI,IAAI,CAAC,oBAAoB,IAAI,CAAC,CAAC,IAAI,IAAI,EAAE,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC,oBAAoB,CAAC,OAAO,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,qBAAqB,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC,EAAE,CAAC;YAC5I,IAAI,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC;QACjC,CAAC;IACH,CAAC;IAEO,cAAc;QACpB,IAAI,IAAI,CAAC,cAAc,IAAI,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,KAAK,IAAI,IAAI,IAAI,CAAC,WAAW,KAAK,IAAI,CAAC,EAAE,CAAC;YAC3F,MAAM,IAAI,KAAK,CAAC,gHAAgH,CAAC,CAAC;QACpI,CAAC;QACD,IAAI,IAAI,CAAC,oBAAoB,KAAK,IAAI,IAAI,IAAI,CAAC,sBAAsB,CAAC,sBAAsB,KAAK,IAAI,EAAE,CAAC;YACtG,MAAM,IAAI,KAAK,CAAC,kGAAkG,CAAC,CAAC;QACtH,CAAC;IACH,CAAC;IAEO,WAAW,CAAC,KAAU;QAC5B,IAAI,MAAM,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,qBAAqB,CAAC,eAAe,CAAC;QACrF,OAAO,UAAU,CAAC,MAAM,CAAC,CAAC;IAC5B,CAAC;8GA5XU,qBAAqB,wKAwCZ,oBAAoB,aACpB,0BAA0B,aAC1B,6BAA6B,aACjB,yBAAyB,6BACzB,8BAA8B,6BAC9B,qCAAqC,6BACrC,qCAAqC,6BACrC,qCAAqC,6BACrC,gBAAgB;kHAhDrC,qBAAqB;;2FAArB,qBAAqB;kBADjC,UAAU;;0BAyCI,MAAM;2BAAC,oBAAoB;;0BAC3B,MAAM;2BAAC,0BAA0B;;0BACjC,MAAM;2BAAC,6BAA6B;;0BACpC,QAAQ;;0BAAI,MAAM;2BAAC,yBAAyB;;0BAC5C,QAAQ;;0BAAI,MAAM;2BAAC,8BAA8B;;0BACjD,QAAQ;;0BAAI,MAAM;2BAAC,qCAAqC;;0BACxD,QAAQ;;0BAAI,MAAM;2BAAC,qCAAqC;;0BACxD,QAAQ;;0BAAI,MAAM;2BAAC,qCAAqC;;0BACxD,QAAQ;;0BAAI,MAAM;2BAAC,gBAAgB","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"]}
|