adb-shared 6.0.18 → 6.0.19

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.
@@ -17,6 +17,7 @@ export class AdbUserService {
17
17
  this.userSubject = new BehaviorSubject(null);
18
18
  this.user$ = this.userSubject.asObservable();
19
19
  this.lastRefreshTime = 0;
20
+ this.hasEmitted = false;
20
21
  this.config = { ...{ scope: 'roles', idleTime: 30, response_type: 'code' }, ...config };
21
22
  }
22
23
  init() {
@@ -35,6 +36,7 @@ export class AdbUserService {
35
36
  }
36
37
  logout() {
37
38
  clearInterval(this.intervalId);
39
+ this.hasEmitted = false;
38
40
  sessionStorage.clear();
39
41
  window.location.href = `${this.config.issuer}/connect/endsession?post_logout_redirect_uri=${this.config.postLogoutUrl}`;
40
42
  }
@@ -50,8 +52,11 @@ export class AdbUserService {
50
52
  console.log('token is about to timeout but we will refresh it because user has been active the last 5minutes');
51
53
  this.refreshToken();
52
54
  }
53
- const user = this.getUser(accessToken);
54
- this.userSubject.next(user);
55
+ if (!this.hasEmitted) {
56
+ const user = this.getUser(accessToken);
57
+ this.userSubject.next(user);
58
+ this.hasEmitted = true;
59
+ }
55
60
  }
56
61
  }
57
62
  else {
@@ -166,4 +171,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.8", ngImpor
166
171
  type: Inject,
167
172
  args: [ADB_USER_SERVICE_CONFIG]
168
173
  }] }, { type: i1.HttpClient }, { type: i2.Router }] });
169
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"adb-user.service.js","sourceRoot":"","sources":["../../../../../../projects/artdata-shared/src/lib/components/adb-user/adb-user.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable, OnDestroy } from '@angular/core';\r\nimport { ADB_USER_SERVICE_CONFIG, AdbUserServiceConfig } from './adb-user-service-config';\r\nimport { BehaviorSubject, Subscription } from 'rxjs';\r\nimport { HttpClient, HttpHeaders } from '@angular/common/http';\r\nimport { addSeconds, format } from 'date-fns';\r\nimport { SessionIds } from './user-constants';\r\nimport { Event, NavigationEnd, Router } from '@angular/router';\r\n\r\n\r\n@Injectable()\r\nexport class AdbUserService implements OnDestroy {\r\n  private subscriptions = new Subscription();\r\n  private userSubject = new BehaviorSubject<any>(null);\r\n  user$ = this.userSubject.asObservable();\r\n\r\n  lastRefreshTime = 0;\r\n  intervalId: any;\r\n\r\n  constructor(\r\n    @Inject(ADB_USER_SERVICE_CONFIG) public config: AdbUserServiceConfig,\r\n    private http: HttpClient,\r\n    private router: Router) {\r\n    this.config = { ...{ scope: 'roles', idleTime: 30, response_type: 'code' }, ...config };\r\n  }\r\n\r\n  init(): void {\r\n    this.router.events.subscribe((event: Event) => {\r\n      if (event instanceof NavigationEnd) {\r\n        if (event.url.includes('auth-callback')) {\r\n          return;\r\n        }\r\n        if(event.url&&event.url!=='/'){\r\n          sessionStorage.setItem(SessionIds.CURRENT_URL,event.url);\r\n        }        \r\n        this.mainLoop();\r\n        this.intervalId = setInterval(() => this.mainLoop(), 60 * 1000); // 60 seconds * 1000 milliseconds\r\n      }\r\n    });\r\n  }\r\n\r\n  logout(): void {\r\n    clearInterval(this.intervalId);\r\n    sessionStorage.clear();\r\n    window.location.href = `${this.config.issuer}/connect/endsession?post_logout_redirect_uri=${this.config.postLogoutUrl}`;\r\n  }\r\n\r\n  private mainLoop(): void {\r\n    const accessToken = sessionStorage.getItem(SessionIds.ACCESS_TOKEN);\r\n    if (accessToken) {\r\n      const timeToLive = this.getTokenTimeToLive();\r\n      if (timeToLive === 0) {\r\n        this.authorize();\r\n      } else {\r\n        if (this.getTokenTimeToLive() <= 60 && this.hasUserBeenActive() && this.canRefreshToken()) {\r\n          console.log('token is about to timeout but we will refresh it because user has been active the last 5minutes');\r\n          this.refreshToken();\r\n        }\r\n        const user = this.getUser(accessToken);\r\n        this.userSubject.next(user);\r\n      }\r\n    } else {\r\n      this.authorize();\r\n    }\r\n  }\r\n\r\n  private refreshToken() {\r\n    const body = `client_id=${encodeURIComponent(this.config.clientId)}&` +\r\n      `grant_type=${encodeURIComponent(\"refresh_token\")}&` +\r\n      `refresh_token=${encodeURIComponent(sessionStorage.getItem(SessionIds.REFRESH_TOKEN))}`;\r\n    this.http.post<any>(`${this.config.issuer}/connect/token`, body, { headers: new HttpHeaders({ 'Content-Type': 'application/x-www-form-urlencoded' }) }).subscribe({\r\n      next: response => {\r\n        this.setTokensInSession(response);\r\n      }\r\n    });\r\n    this.lastRefreshTime = Date.now();\r\n  }\r\n\r\n  setTokensInSession(params: any): void {\r\n    if (params.refresh_token) {\r\n      sessionStorage.setItem(SessionIds.REFRESH_TOKEN, params.refresh_token);\r\n    }\r\n    if (params.access_token) {\r\n      sessionStorage.setItem(SessionIds.ACCESS_TOKEN, params.access_token);\r\n    }\r\n    if (params.expires_in) {\r\n      const expireTime = addSeconds(new Date(), params.expires_in);\r\n      sessionStorage.setItem(SessionIds.TOKEN_EXPIRES, format(expireTime, 'yyyy-MM-dd HH:mm:ss'));\r\n    }\r\n  }\r\n\r\n  private async authorize() {\r\n    const codeVerifier = Array.from(crypto.getRandomValues(new Uint8Array(64)), byte =>\r\n      'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'[(byte as number) % 62]\r\n    ).join('');\r\n    sessionStorage.setItem(SessionIds.CODE_VERIFIER, codeVerifier);\r\n\r\n    const encoder = new TextEncoder();\r\n    const data = encoder.encode(codeVerifier);\r\n    const hashBuffer = await crypto.subtle.digest('SHA-256', data);\r\n    const hashArray = new Uint8Array(hashBuffer);\r\n    let codeChallenge = btoa(String.fromCharCode(...hashArray));\r\n    codeChallenge = codeChallenge.replace(/=/g, '')\r\n      .replace(/\\+/g, '-')\r\n      .replace(/\\//g, '_');\r\n\r\n    const params = {\r\n      response_type: this.config.responseType,\r\n      client_id: this.config.clientId,\r\n      redirect_uri: this.config.redirectUrl,\r\n      scope: this.config.scope,\r\n      code_challenge: codeChallenge,\r\n      code_challenge_method: 'S256'\r\n    };\r\n    const url = new URL(this.config.issuer + '/connect/authorize');\r\n    url.search = this.addQueryParams(params).toString();\r\n    window.location.href = url.toString();\r\n  }\r\n\r\n\r\n  private addQueryParams(object: any): URLSearchParams {\r\n    if (!object) {\r\n      return null;\r\n    }\r\n    const urlParams = new URLSearchParams();\r\n    for (const prop in object) {\r\n      urlParams.append(prop, object[prop]);\r\n    }\r\n    return urlParams;\r\n  }\r\n\r\n  private getTokenTimeToLive(): number {\r\n    const expires = sessionStorage.getItem(SessionIds.TOKEN_EXPIRES);\r\n    if (expires) {\r\n      const expireDate = new Date(expires);\r\n      const now = new Date();\r\n      const timeToLive = Math.floor((expireDate.getTime() - now.getTime()) / 1000);\r\n      return timeToLive < 0 ? 0 : timeToLive;\r\n    }\r\n    return 0;\r\n  }\r\n\r\n  //Cooldown for avoiding infinite refresh\r\n  private canRefreshToken(): boolean {\r\n    const now = Date.now();\r\n    if (now - this.lastRefreshTime > (2 * 60 * 1000)) {\r\n      return true;\r\n    }\r\n    return false;\r\n  }\r\n\r\n  private hasUserBeenActive(): boolean {\r\n    const lastActive = sessionStorage.getItem(SessionIds.USER_LAST_ACTIVE);\r\n    if (!lastActive) {\r\n      return false;\r\n    }\r\n    const timeSinceLastActive = Date.now() - parseInt(lastActive, 10);\r\n    const idleTimeMs = this.config.idleTime * 60 * 1000;\r\n    return timeSinceLastActive <= idleTimeMs;\r\n  }\r\n\r\n  getUser(token: string): any {\r\n    try {\r\n      const [header, payload, signature] = token.split('.');\r\n      const base64Url = payload.replace(/-/g, '+').replace(/_/g, '/');\r\n      const decodedPayload = atob(base64Url);\r\n      const parsedPayload = JSON.parse(decodedPayload);\r\n      return parsedPayload;\r\n    } catch (error) {\r\n      console.error('Error decoding JWT', error);\r\n      return null;\r\n    }\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.subscriptions.unsubscribe();\r\n  }\r\n\r\n}"]}
174
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"adb-user.service.js","sourceRoot":"","sources":["../../../../../../projects/artdata-shared/src/lib/components/adb-user/adb-user.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable, OnDestroy } from '@angular/core';\r\nimport { ADB_USER_SERVICE_CONFIG, AdbUserServiceConfig } from './adb-user-service-config';\r\nimport { BehaviorSubject, Subject, Subscription } from 'rxjs';\r\nimport { HttpClient, HttpHeaders } from '@angular/common/http';\r\nimport { addSeconds, format } from 'date-fns';\r\nimport { SessionIds } from './user-constants';\r\nimport { Event, NavigationEnd, Router } from '@angular/router';\r\n\r\n\r\n@Injectable()\r\nexport class AdbUserService implements OnDestroy {\r\n  private subscriptions = new Subscription();\r\n  private userSubject: BehaviorSubject<any> = new BehaviorSubject<any>(null);\r\n  user$ = this.userSubject.asObservable();\r\n\r\n  lastRefreshTime = 0;\r\n  intervalId: any;\r\n  hasEmitted = false;\r\n\r\n  constructor(\r\n    @Inject(ADB_USER_SERVICE_CONFIG) public config: AdbUserServiceConfig,\r\n    private http: HttpClient,\r\n    private router: Router) {\r\n    this.config = { ...{ scope: 'roles', idleTime: 30, response_type: 'code' }, ...config };\r\n  }\r\n\r\n  init(): void {\r\n    this.router.events.subscribe((event: Event) => {\r\n      if (event instanceof NavigationEnd) {\r\n        if (event.url.includes('auth-callback')) {\r\n          return;\r\n        }\r\n        if (event.url && event.url !== '/') {\r\n          sessionStorage.setItem(SessionIds.CURRENT_URL, event.url);\r\n        }\r\n        this.mainLoop();\r\n        this.intervalId = setInterval(() => this.mainLoop(), 60 * 1000); // 60 seconds * 1000 milliseconds\r\n      }\r\n    });\r\n  }\r\n\r\n  logout(): void {\r\n    clearInterval(this.intervalId);\r\n    this.hasEmitted = false;\r\n    sessionStorage.clear();\r\n    window.location.href = `${this.config.issuer}/connect/endsession?post_logout_redirect_uri=${this.config.postLogoutUrl}`;\r\n  }\r\n\r\n  private mainLoop(): void {\r\n    const accessToken = sessionStorage.getItem(SessionIds.ACCESS_TOKEN);\r\n    if (accessToken) {\r\n      const timeToLive = this.getTokenTimeToLive();\r\n      if (timeToLive === 0) {\r\n        this.authorize();\r\n      } else {\r\n        if (this.getTokenTimeToLive() <= 60 && this.hasUserBeenActive() && this.canRefreshToken()) {\r\n          console.log('token is about to timeout but we will refresh it because user has been active the last 5minutes');\r\n          this.refreshToken();\r\n        }\r\n        if(!this.hasEmitted){\r\n          const user = this.getUser(accessToken);\r\n          this.userSubject.next(user);\r\n          this.hasEmitted = true;\r\n        }\r\n      }\r\n    } else {\r\n      this.authorize();\r\n    }\r\n  }\r\n\r\n  private refreshToken() {\r\n    const body = `client_id=${encodeURIComponent(this.config.clientId)}&` +\r\n      `grant_type=${encodeURIComponent(\"refresh_token\")}&` +\r\n      `refresh_token=${encodeURIComponent(sessionStorage.getItem(SessionIds.REFRESH_TOKEN))}`;\r\n    this.http.post<any>(`${this.config.issuer}/connect/token`, body, { headers: new HttpHeaders({ 'Content-Type': 'application/x-www-form-urlencoded' }) }).subscribe({\r\n      next: response => {\r\n        this.setTokensInSession(response);\r\n      }\r\n    });\r\n    this.lastRefreshTime = Date.now();\r\n  }\r\n\r\n  setTokensInSession(params: any): void {\r\n    if (params.refresh_token) {\r\n      sessionStorage.setItem(SessionIds.REFRESH_TOKEN, params.refresh_token);\r\n    }\r\n    if (params.access_token) {\r\n      sessionStorage.setItem(SessionIds.ACCESS_TOKEN, params.access_token);\r\n    }\r\n    if (params.expires_in) {\r\n      const expireTime = addSeconds(new Date(), params.expires_in);\r\n      sessionStorage.setItem(SessionIds.TOKEN_EXPIRES, format(expireTime, 'yyyy-MM-dd HH:mm:ss'));\r\n    }\r\n  }\r\n\r\n  private async authorize() {\r\n    const codeVerifier = Array.from(crypto.getRandomValues(new Uint8Array(64)), byte =>\r\n      'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'[(byte as number) % 62]\r\n    ).join('');\r\n    sessionStorage.setItem(SessionIds.CODE_VERIFIER, codeVerifier);\r\n\r\n    const encoder = new TextEncoder();\r\n    const data = encoder.encode(codeVerifier);\r\n    const hashBuffer = await crypto.subtle.digest('SHA-256', data);\r\n    const hashArray = new Uint8Array(hashBuffer);\r\n    let codeChallenge = btoa(String.fromCharCode(...hashArray));\r\n    codeChallenge = codeChallenge.replace(/=/g, '')\r\n      .replace(/\\+/g, '-')\r\n      .replace(/\\//g, '_');\r\n\r\n    const params = {\r\n      response_type: this.config.responseType,\r\n      client_id: this.config.clientId,\r\n      redirect_uri: this.config.redirectUrl,\r\n      scope: this.config.scope,\r\n      code_challenge: codeChallenge,\r\n      code_challenge_method: 'S256'\r\n    };\r\n    const url = new URL(this.config.issuer + '/connect/authorize');\r\n    url.search = this.addQueryParams(params).toString();\r\n    window.location.href = url.toString();\r\n  }\r\n\r\n\r\n  private addQueryParams(object: any): URLSearchParams {\r\n    if (!object) {\r\n      return null;\r\n    }\r\n    const urlParams = new URLSearchParams();\r\n    for (const prop in object) {\r\n      urlParams.append(prop, object[prop]);\r\n    }\r\n    return urlParams;\r\n  }\r\n\r\n  private getTokenTimeToLive(): number {\r\n    const expires = sessionStorage.getItem(SessionIds.TOKEN_EXPIRES);\r\n    if (expires) {\r\n      const expireDate = new Date(expires);\r\n      const now = new Date();\r\n      const timeToLive = Math.floor((expireDate.getTime() - now.getTime()) / 1000);\r\n      return timeToLive < 0 ? 0 : timeToLive;\r\n    }\r\n    return 0;\r\n  }\r\n\r\n  //Cooldown for avoiding infinite refresh\r\n  private canRefreshToken(): boolean {\r\n    const now = Date.now();\r\n    if (now - this.lastRefreshTime > (2 * 60 * 1000)) {\r\n      return true;\r\n    }\r\n    return false;\r\n  }\r\n\r\n  private hasUserBeenActive(): boolean {\r\n    const lastActive = sessionStorage.getItem(SessionIds.USER_LAST_ACTIVE);\r\n    if (!lastActive) {\r\n      return false;\r\n    }\r\n    const timeSinceLastActive = Date.now() - parseInt(lastActive, 10);\r\n    const idleTimeMs = this.config.idleTime * 60 * 1000;\r\n    return timeSinceLastActive <= idleTimeMs;\r\n  }\r\n\r\n  getUser(token: string): any {\r\n    try {\r\n      const [header, payload, signature] = token.split('.');\r\n      const base64Url = payload.replace(/-/g, '+').replace(/_/g, '/');\r\n      const decodedPayload = atob(base64Url);\r\n      const parsedPayload = JSON.parse(decodedPayload);\r\n      return parsedPayload;\r\n    } catch (error) {\r\n      console.error('Error decoding JWT', error);\r\n      return null;\r\n    }\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.subscriptions.unsubscribe();\r\n  }\r\n\r\n}"]}
@@ -2046,6 +2046,7 @@ class AdbUserService {
2046
2046
  this.userSubject = new BehaviorSubject(null);
2047
2047
  this.user$ = this.userSubject.asObservable();
2048
2048
  this.lastRefreshTime = 0;
2049
+ this.hasEmitted = false;
2049
2050
  this.config = { ...{ scope: 'roles', idleTime: 30, response_type: 'code' }, ...config };
2050
2051
  }
2051
2052
  init() {
@@ -2064,6 +2065,7 @@ class AdbUserService {
2064
2065
  }
2065
2066
  logout() {
2066
2067
  clearInterval(this.intervalId);
2068
+ this.hasEmitted = false;
2067
2069
  sessionStorage.clear();
2068
2070
  window.location.href = `${this.config.issuer}/connect/endsession?post_logout_redirect_uri=${this.config.postLogoutUrl}`;
2069
2071
  }
@@ -2079,8 +2081,11 @@ class AdbUserService {
2079
2081
  console.log('token is about to timeout but we will refresh it because user has been active the last 5minutes');
2080
2082
  this.refreshToken();
2081
2083
  }
2082
- const user = this.getUser(accessToken);
2083
- this.userSubject.next(user);
2084
+ if (!this.hasEmitted) {
2085
+ const user = this.getUser(accessToken);
2086
+ this.userSubject.next(user);
2087
+ this.hasEmitted = true;
2088
+ }
2084
2089
  }
2085
2090
  }
2086
2091
  else {