UNPKG

@spartacus/cdc

Version:

Customer Data Cloud Integration library for Spartacus

98 lines 14.7 kB
import { Injectable } from '@angular/core'; import { TokenTarget } from '@spartacus/asm/root'; import { AuthActions, GlobalMessageType, OCC_USER_ID_CURRENT, } from '@spartacus/core'; import { combineLatest, of } from 'rxjs'; import { map, take, tap } from 'rxjs/operators'; import { CdcAuthActions } from '../store/actions/index'; import * as i0 from "@angular/core"; import * as i1 from "@ngrx/store"; import * as i2 from "@spartacus/core"; /** * Service to support custom CDC OAuth flow. */ export class CdcAuthService { constructor(store, authStorageService, userIdService, globalMessageService, authRedirectService) { this.store = store; this.authStorageService = authStorageService; this.userIdService = userIdService; this.globalMessageService = globalMessageService; this.authRedirectService = authRedirectService; } /** * Loads a new user token using custom oauth flow * * @param UID * @param UIDSignature * @param signatureTimestamp * @param idToken * @param baseSite */ loginWithCustomCdcFlow(UID, UIDSignature, signatureTimestamp, idToken, baseSite) { this.store.dispatch(new CdcAuthActions.LoadCdcUserToken({ UID: UID, UIDSignature: UIDSignature, signatureTimestamp: signatureTimestamp, idToken: idToken, baseSite: baseSite, })); } /** * Utility to differentiate between AuthStorageService and AsmAuthStorageService */ isAsmAuthStorageService(service) { return 'getTokenTarget' in service; } /** * Transform and store the token received from custom flow to library format and login user. * * @param token */ loginWithToken(token) { let stream$ = of(true); if (this.isAsmAuthStorageService(this.authStorageService)) { stream$ = combineLatest([ this.authStorageService.getToken(), this.authStorageService.getTokenTarget(), ]).pipe(take(1), map(([currentToken, tokenTarget]) => { return (!!(currentToken === null || currentToken === void 0 ? void 0 : currentToken.access_token) && tokenTarget === TokenTarget.CSAgent); }), tap((isAsmAgentLoggedIn) => { if (isAsmAgentLoggedIn) { this.globalMessageService.add({ key: 'asm.auth.agentLoggedInError', }, GlobalMessageType.MSG_TYPE_ERROR); } }), map((isAsmAgentLoggedIn) => !isAsmAgentLoggedIn)); } stream$.pipe(take(1)).subscribe((canLogin) => { if (canLogin) { // Code mostly based on auth lib we use and the way it handles token properties this.authStorageService.setItem('access_token', token.access_token); if (token.granted_scopes && Array.isArray(token.granted_scopes)) { this.authStorageService.setItem('granted_scopes', JSON.stringify(token.granted_scopes)); } this.authStorageService.setItem('access_token_stored_at', '' + Date.now()); if (token.expires_in) { const expiresInMilliseconds = token.expires_in * 1000; const now = new Date(); const expiresAt = now.getTime() + expiresInMilliseconds; this.authStorageService.setItem('expires_at', '' + expiresAt); } if (token.refresh_token) { this.authStorageService.setItem('refresh_token', token.refresh_token); } // OCC specific code this.userIdService.setUserId(OCC_USER_ID_CURRENT); this.store.dispatch(new AuthActions.Login()); // Remove any global errors and redirect user on successful login this.globalMessageService.remove(GlobalMessageType.MSG_TYPE_ERROR); this.authRedirectService.redirect(); } }); } } CdcAuthService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: CdcAuthService, deps: [{ token: i1.Store }, { token: i2.AuthStorageService }, { token: i2.UserIdService }, { token: i2.GlobalMessageService }, { token: i2.AuthRedirectService }], target: i0.ɵɵFactoryTarget.Injectable }); CdcAuthService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: CdcAuthService }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: CdcAuthService, decorators: [{ type: Injectable }], ctorParameters: function () { return [{ type: i1.Store }, { type: i2.AuthStorageService }, { type: i2.UserIdService }, { type: i2.GlobalMessageService }, { type: i2.AuthRedirectService }]; } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cdc-auth.service.js","sourceRoot":"","sources":["../../../../../../integration-libs/cdc/core/auth/facade/cdc-auth.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { Store } from '@ngrx/store';\nimport { AsmAuthStorageService, TokenTarget } from '@spartacus/asm/root';\nimport {\n  AuthActions,\n  AuthRedirectService,\n  AuthStorageService,\n  AuthToken,\n  GlobalMessageService,\n  GlobalMessageType,\n  OCC_USER_ID_CURRENT,\n  UserIdService,\n} from '@spartacus/core';\nimport { combineLatest, of } from 'rxjs';\nimport { map, take, tap } from 'rxjs/operators';\nimport { CdcAuthActions } from '../store/actions/index';\nimport { CdcAuthFacade } from '@spartacus/cdc/root';\n\n/**\n * Service to support custom CDC OAuth flow.\n */\n@Injectable()\nexport class CdcAuthService implements CdcAuthFacade {\n  constructor(\n    protected store: Store,\n    protected authStorageService: AuthStorageService,\n    protected userIdService: UserIdService,\n    protected globalMessageService: GlobalMessageService,\n    protected authRedirectService: AuthRedirectService\n  ) {}\n\n  /**\n   * Loads a new user token using custom oauth flow\n   *\n   * @param UID\n   * @param UIDSignature\n   * @param signatureTimestamp\n   * @param idToken\n   * @param baseSite\n   */\n  loginWithCustomCdcFlow(\n    UID: string,\n    UIDSignature: string,\n    signatureTimestamp: string,\n    idToken: string,\n    baseSite: string\n  ): void {\n    this.store.dispatch(\n      new CdcAuthActions.LoadCdcUserToken({\n        UID: UID,\n        UIDSignature: UIDSignature,\n        signatureTimestamp: signatureTimestamp,\n        idToken: idToken,\n        baseSite: baseSite,\n      })\n    );\n  }\n\n  /**\n   * Utility to differentiate between AuthStorageService and AsmAuthStorageService\n   */\n  private isAsmAuthStorageService(\n    service: AuthStorageService | AsmAuthStorageService\n  ): service is AsmAuthStorageService {\n    return 'getTokenTarget' in service;\n  }\n\n  /**\n   * Transform and store the token received from custom flow to library format and login user.\n   *\n   * @param token\n   */\n  loginWithToken(token: Partial<AuthToken> & { expires_in?: number }): void {\n    let stream$ = of(true);\n    if (this.isAsmAuthStorageService(this.authStorageService)) {\n      stream$ = combineLatest([\n        this.authStorageService.getToken(),\n        this.authStorageService.getTokenTarget(),\n      ]).pipe(\n        take(1),\n        map(([currentToken, tokenTarget]) => {\n          return (\n            !!currentToken?.access_token && tokenTarget === TokenTarget.CSAgent\n          );\n        }),\n        tap((isAsmAgentLoggedIn) => {\n          if (isAsmAgentLoggedIn) {\n            this.globalMessageService.add(\n              {\n                key: 'asm.auth.agentLoggedInError',\n              },\n              GlobalMessageType.MSG_TYPE_ERROR\n            );\n          }\n        }),\n        map((isAsmAgentLoggedIn) => !isAsmAgentLoggedIn)\n      );\n    }\n\n    stream$.pipe(take(1)).subscribe((canLogin) => {\n      if (canLogin) {\n        // Code mostly based on auth lib we use and the way it handles token properties\n        this.authStorageService.setItem('access_token', token.access_token);\n\n        if (token.granted_scopes && Array.isArray(token.granted_scopes)) {\n          this.authStorageService.setItem(\n            'granted_scopes',\n            JSON.stringify(token.granted_scopes)\n          );\n        }\n\n        this.authStorageService.setItem(\n          'access_token_stored_at',\n          '' + Date.now()\n        );\n\n        if (token.expires_in) {\n          const expiresInMilliseconds = token.expires_in * 1000;\n          const now = new Date();\n          const expiresAt = now.getTime() + expiresInMilliseconds;\n          this.authStorageService.setItem('expires_at', '' + expiresAt);\n        }\n\n        if (token.refresh_token) {\n          this.authStorageService.setItem('refresh_token', token.refresh_token);\n        }\n\n        // OCC specific code\n        this.userIdService.setUserId(OCC_USER_ID_CURRENT);\n\n        this.store.dispatch(new AuthActions.Login());\n\n        // Remove any global errors and redirect user on successful login\n        this.globalMessageService.remove(GlobalMessageType.MSG_TYPE_ERROR);\n        this.authRedirectService.redirect();\n      }\n    });\n  }\n}\n"]}