UNPKG

@terminus/ngx-tools

Version:

[![CircleCI][circle-badge]][circle-link] [![codecov][codecov-badge]][codecov-project] [![semantic-release][semantic-release-badge]][semantic-release] [![MIT License][license-image]][license-url] <br> [![NPM version][npm-version-image]][npm-url] [![Github

152 lines 24.3 kB
import { __decorate, __param, __read } from "tslib"; import { Inject, Injectable, InjectionToken, Optional, } from '@angular/core'; import { Actions, Effect, ofType, } from '@ngrx/effects'; import { Store } from '@ngrx/store'; import { TsCookieService } from '@terminus/ngx-tools/browser'; import { merge, of, Scheduler, timer, } from 'rxjs'; import { async } from 'rxjs/internal/scheduler/async'; import { delay, filter, flatMap, map, mergeMap, take, withLatestFrom, } from 'rxjs/operators'; import { jwtDecode } from '../jwt-decode/jwt-decode'; import * as JwtTokenProviderActions from './actions'; import { getJwtTokenRoot, getTokens, } from './selectors'; import { INITIAL_TOKEN_NAME } from './tokens'; import { SCHEDULER } from './utilities/retry-with-escalation'; export var SECONDS_BEFORE_EXPIRATION_TO_NOTIFY = new InjectionToken('wait time'); var SECONDS_IN_MINUTE = 60; var DEFAULT_MINUTES_BEFORE_EXPIRATION_TO_NOTIFY = 5; var DEFAULT_SECONDS_BEFORE_EXPIRATION_TO_NOTIFY = DEFAULT_MINUTES_BEFORE_EXPIRATION_TO_NOTIFY * SECONDS_IN_MINUTE; var CLEANUP_DELAY = 100; var TOKENS_EXPIRED_DELAY = 10; var MS_IN_SECONDS = 1000; var JwtTokenProviderEffects = /** @class */ (function () { function JwtTokenProviderEffects(actions$, // eslint-disable-next-line @typescript-eslint/no-explicit-any store, cookieService, initialTokenName, scheduler, timeToWaitBeforeExpiration) { var _this = this; this.actions$ = actions$; this.store = store; this.cookieService = cookieService; this.initialTokenName = initialTokenName; this.scheduler = scheduler; this.timeToWaitBeforeExpiration = timeToWaitBeforeExpiration; this.initializationCleanup$ = of(true) .pipe(delay(CLEANUP_DELAY, this.scheduler || async), withLatestFrom(this.store.select(getTokens())), map(function (_a) { var _b = __read(_a, 2), _ = _b[0], tokens = _b[1]; return tokens; }), take(1), flatMap(function (tokens) { var actions = []; for (var tokenName in tokens) { if (tokens.hasOwnProperty(tokenName)) { var token = tokens[tokenName]; if (token) { actions.push(new JwtTokenProviderActions.StoreToken({ tokenName: tokenName, token: token, })); } } } return actions; })); this.allTokensExpired$ = this.actions$ .pipe(ofType(JwtTokenProviderActions.ActionTypes.TokenExpired), delay(TOKENS_EXPIRED_DELAY, this.scheduler || async), withLatestFrom(this.store.select(getTokens())), map(function (_a) { var _b = __read(_a, 2), _ = _b[0], tokens = _b[1]; return tokens; }), filter(function (tokens) { return Object.keys(tokens).length === 0; }), map(function (tokens) { return new JwtTokenProviderActions.AllTokensExpired(); })); this.notifyOfTokenExpiration$ = this.actions$ .pipe(ofType(JwtTokenProviderActions.ActionTypes.StoreToken), // eslint-disable-next-line max-len map(function (action) { return [action, jwtDecode(action.token)]; }), filter(function (a) { return a[1].exp !== undefined; }), mergeMap(function (_a) { var _b = __read(_a, 2), action = _b[0], claims = _b[1]; var currentEpoch = Math.ceil((new Date()).getTime() / MS_IN_SECONDS); if (claims.exp > currentEpoch) { var expiresIn = claims.exp - currentEpoch; var expirationBuffer = _this.timeToWaitBeforeExpiration || DEFAULT_SECONDS_BEFORE_EXPIRATION_TO_NOTIFY; var expirationNearIn = 0; if (expiresIn < expirationBuffer) { expirationNearIn = 1; } else { expirationNearIn = expiresIn - expirationBuffer; } return merge(_this.buildDelayedExpirationObservable(expirationNearIn * MS_IN_SECONDS, action, false), _this.buildDelayedExpirationObservable(expiresIn * MS_IN_SECONDS, action, true)); } return of(new JwtTokenProviderActions.TokenExpired({ tokenName: action.tokenName, token: action.token, })); })); this.initialCookieLoader$ = function (_a) { var _b = (_a === void 0 ? {} : _a).currentState, currentState = _b === void 0 ? _this.store.select(getJwtTokenRoot()) : _b; return of(true).pipe(take(1), withLatestFrom(currentState), filter(function (_a) { var _b = __read(_a, 2), _ = _b[0], state = _b[1]; return !!(state && state.jwtTokens.initialTokenStatus === 'uninitialized'); }), mergeMap(function (_a) { var _b = __read(_a, 2), a = _b[0], _ = _b[1]; var cookie = _this.cookieService.get('jwt_cookie'); if (cookie.length > 0) { return [ new JwtTokenProviderActions.InitialTokenExtracted(cookie), new JwtTokenProviderActions.StoreToken({ tokenName: _this.initialTokenName, token: cookie, isDefaultToken: true, }), ]; } return [ new JwtTokenProviderActions.InitialTokenExtracted(cookie), ]; })); }; } /* * This next function is being excluded from coverage due the complexities of testing the `delay` function. * In order to test as much as possible, each piece has been separated into smaller testable functions. */ JwtTokenProviderEffects.prototype.buildDelayedExpirationObservable = function (emitTime, action, expired) { var outputActionArgs = { tokenName: action.tokenName, token: action.token, }; return timer(emitTime, this.scheduler || async).pipe(take(1), map(function () { return (expired ? new JwtTokenProviderActions.TokenExpired(outputActionArgs) : new JwtTokenProviderActions.TokenNearingExpiration(outputActionArgs)); })); }; JwtTokenProviderEffects.ctorParameters = function () { return [ { type: Actions }, { type: Store }, { type: TsCookieService }, { type: String, decorators: [{ type: Inject, args: [INITIAL_TOKEN_NAME,] }] }, { type: Scheduler, decorators: [{ type: Optional }, { type: Inject, args: [SCHEDULER,] }] }, { type: Number, decorators: [{ type: Optional }, { type: Inject, args: [SECONDS_BEFORE_EXPIRATION_TO_NOTIFY,] }] } ]; }; __decorate([ Effect() ], JwtTokenProviderEffects.prototype, "initializationCleanup$", void 0); __decorate([ Effect() ], JwtTokenProviderEffects.prototype, "allTokensExpired$", void 0); __decorate([ Effect() ], JwtTokenProviderEffects.prototype, "notifyOfTokenExpiration$", void 0); __decorate([ Effect() ], JwtTokenProviderEffects.prototype, "initialCookieLoader$", void 0); JwtTokenProviderEffects = __decorate([ Injectable(), __param(3, Inject(INITIAL_TOKEN_NAME)), __param(4, Optional()), __param(4, Inject(SCHEDULER)) // TODO: Scheduler is marked as deprecated to stop others from using although it is not technically deprecated from // what I can tell. The 'correct' path would be to create our own class extending `SchedulerLike`. // https://github.com/GetTerminus/ngx-tools/issues/287 // eslint-disable-next-line deprecation/deprecation , __param(5, Optional()), __param(5, Inject(SECONDS_BEFORE_EXPIRATION_TO_NOTIFY)) ], JwtTokenProviderEffects); return JwtTokenProviderEffects; }()); export { JwtTokenProviderEffects }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"effects.js","sourceRoot":"ng://@terminus/ngx-tools/jwt/","sources":["jwt-token-managment/effects.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Inject,\n  Injectable,\n  InjectionToken,\n  Optional,\n} from '@angular/core';\nimport {\n  Actions,\n  Effect,\n  ofType,\n} from '@ngrx/effects';\nimport { Store } from '@ngrx/store';\nimport { TsCookieService } from '@terminus/ngx-tools/browser';\nimport {\n  merge,\n  of,\n  Scheduler,\n  timer,\n} from 'rxjs';\nimport { async } from 'rxjs/internal/scheduler/async';\nimport {\n  delay,\n  filter,\n  flatMap,\n  map,\n  mergeMap,\n  take,\n  withLatestFrom,\n} from 'rxjs/operators';\n\nimport { jwtDecode } from '../jwt-decode/jwt-decode';\n\nimport * as JwtTokenProviderActions from './actions';\nimport {\n  getJwtTokenRoot,\n  getTokens,\n} from './selectors';\nimport { INITIAL_TOKEN_NAME } from './tokens';\nimport { SCHEDULER } from './utilities/retry-with-escalation';\n\n\nexport interface Claims { exp: number; }\n\nexport interface MinimalClaimMap {\n  [id: string]: Claims;\n}\n\nexport const SECONDS_BEFORE_EXPIRATION_TO_NOTIFY = new InjectionToken<number>('wait time');\n\nconst SECONDS_IN_MINUTE = 60;\nconst DEFAULT_MINUTES_BEFORE_EXPIRATION_TO_NOTIFY = 5;\nconst DEFAULT_SECONDS_BEFORE_EXPIRATION_TO_NOTIFY = DEFAULT_MINUTES_BEFORE_EXPIRATION_TO_NOTIFY * SECONDS_IN_MINUTE;\nconst CLEANUP_DELAY = 100;\nconst TOKENS_EXPIRED_DELAY = 10;\nconst MS_IN_SECONDS = 1000;\n\ntype PartialClaimTuple = [\n  JwtTokenProviderActions.StoreToken<MinimalClaimMap>,\n  Partial<Claims>\n];\n\ntype FullClaimsTuple = [\n  JwtTokenProviderActions.StoreToken<MinimalClaimMap>,\n  Claims\n];\n\n\n@Injectable()\nexport class JwtTokenProviderEffects {\n\n  constructor(\n    private actions$: Actions<JwtTokenProviderActions.Actions<MinimalClaimMap>>,\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    private store: Store<any>,\n    private cookieService: TsCookieService,\n\n    @Inject(INITIAL_TOKEN_NAME)\n    private initialTokenName: string,\n\n    @Optional()\n    @Inject(SCHEDULER)\n    // TODO: Scheduler is marked as deprecated to stop others from using although it is not technically deprecated from\n    // what I can tell. The 'correct' path would be to create our own class extending `SchedulerLike`.\n    // https://github.com/GetTerminus/ngx-tools/issues/287\n    // eslint-disable-next-line deprecation/deprecation\n    private scheduler: Scheduler,\n\n    @Optional()\n    @Inject(SECONDS_BEFORE_EXPIRATION_TO_NOTIFY)\n    private timeToWaitBeforeExpiration: number,\n  ) {}\n\n\n  @Effect()\n  public initializationCleanup$ = of(true)\n    .pipe(\n      delay(CLEANUP_DELAY, this.scheduler || async),\n      withLatestFrom(\n        this.store.select(getTokens<MinimalClaimMap>()),\n      ),\n      map(([_, tokens]) => tokens),\n      take(1),\n      flatMap(tokens => {\n        const actions: JwtTokenProviderActions.StoreToken<MinimalClaimMap>[] = [];\n\n        for (const tokenName in tokens) {\n          if (tokens.hasOwnProperty(tokenName)) {\n            const token = tokens[tokenName];\n            if (token) {\n              actions.push(\n                new JwtTokenProviderActions.StoreToken({\n                  tokenName,\n                  token,\n                }),\n              );\n            }\n          }\n        }\n        return actions;\n      }),\n    )\n  ;\n\n\n  @Effect()\n  public allTokensExpired$ = this.actions$\n    .pipe(\n      ofType<never>(JwtTokenProviderActions.ActionTypes.TokenExpired),\n      delay(TOKENS_EXPIRED_DELAY, this.scheduler || async),\n      withLatestFrom(\n        this.store.select(getTokens<MinimalClaimMap>()),\n      ),\n      map(([_, tokens]) => tokens),\n      filter(tokens => Object.keys(tokens).length === 0),\n      map(tokens => new JwtTokenProviderActions.AllTokensExpired()),\n    )\n  ;\n\n\n  @Effect()\n  public notifyOfTokenExpiration$ = this.actions$\n    .pipe(\n      ofType<JwtTokenProviderActions.StoreToken<MinimalClaimMap>>(JwtTokenProviderActions.ActionTypes.StoreToken),\n      // eslint-disable-next-line max-len\n      map((action: JwtTokenProviderActions.StoreToken<MinimalClaimMap>): PartialClaimTuple => [action, jwtDecode<Partial<Claims>>(action.token)]),\n      filter((a: PartialClaimTuple): a is FullClaimsTuple => a[1].exp !== undefined),\n      mergeMap(([action, claims]) => {\n        const currentEpoch = Math.ceil((new Date()).getTime() / MS_IN_SECONDS);\n\n        if (claims.exp > currentEpoch) {\n          const expiresIn = claims.exp - currentEpoch;\n          const expirationBuffer = this.timeToWaitBeforeExpiration || DEFAULT_SECONDS_BEFORE_EXPIRATION_TO_NOTIFY;\n          let expirationNearIn = 0;\n\n          if (expiresIn < expirationBuffer) {\n            expirationNearIn = 1;\n          } else {\n            expirationNearIn = expiresIn - expirationBuffer;\n          }\n\n          return merge(\n            this.buildDelayedExpirationObservable(expirationNearIn * MS_IN_SECONDS, action, false),\n            this.buildDelayedExpirationObservable(expiresIn * MS_IN_SECONDS, action, true),\n          );\n        }\n        return of(new JwtTokenProviderActions.TokenExpired<MinimalClaimMap>({\n          tokenName: action.tokenName,\n          token: action.token,\n        }));\n\n      }),\n    )\n  ;\n\n\n  @Effect()\n  public initialCookieLoader$ = ({\n    currentState = this.store.select(getJwtTokenRoot()),\n  } = {}) => of(true).pipe(\n    take(1),\n    withLatestFrom(currentState),\n    filter(([_, state]) => !!(state && state.jwtTokens.initialTokenStatus === 'uninitialized')),\n    mergeMap(([a, _]) => {\n      const cookie = this.cookieService.get('jwt_cookie');\n      if (cookie.length > 0) {\n        return [\n          new JwtTokenProviderActions.InitialTokenExtracted(cookie),\n          new JwtTokenProviderActions.StoreToken({\n            tokenName: this.initialTokenName,\n            token: cookie,\n            isDefaultToken: true,\n          }),\n        ];\n      }\n      return [\n        new JwtTokenProviderActions.InitialTokenExtracted(cookie),\n      ];\n\n    }),\n  );\n\n\n  /*\n   * This next function is being excluded from coverage due the complexities of testing the `delay` function.\n   * In order to test as much as possible, each piece has been separated into smaller testable functions.\n   */\n  public buildDelayedExpirationObservable(\n    emitTime: number | Date,\n    action: JwtTokenProviderActions.StoreToken<MinimalClaimMap>,\n    expired: boolean,\n  ) {\n    const outputActionArgs = {\n      tokenName: action.tokenName,\n      token: action.token,\n    };\n\n    return timer(emitTime, this.scheduler || async).pipe(\n      take(1),\n      map(() => (expired\n        ? new JwtTokenProviderActions.TokenExpired<MinimalClaimMap>(outputActionArgs)\n        : new JwtTokenProviderActions.TokenNearingExpiration<MinimalClaimMap>(outputActionArgs))),\n    );\n  }\n}\n"]}