@spartacus/cdc
Version:
Customer Data Cloud Integration library for Spartacus
98 lines • 14.7 kB
JavaScript
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"]}