@okta/okta-angular
Version:
Angular support for Okta
136 lines • 18.6 kB
JavaScript
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
/*
* Copyright (c) 2017-Present, Okta, Inc. and/or its affiliates. All rights reserved.
* The Okta software accompanied by this notice is provided pursuant to the Apache License, Version 2.0 (the "License.")
*
* You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0.
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
*
* See the License for the specific language governing permissions and limitations under the License.
*/
import { Injectable, Inject } from '@angular/core';
import { Router, NavigationStart } from '@angular/router';
import { filter } from 'rxjs/operators';
import { OKTA_AUTH } from './models/okta.config';
import * as i0 from "@angular/core";
import * as i1 from "./services/auth-config.serice";
import * as i2 from "@okta/okta-auth-js";
export class OktaAuthGuard {
constructor(oktaAuth, injector, configService) {
this.oktaAuth = oktaAuth;
this.injector = injector;
this.configService = configService;
this.updateAuthStateListener = (authState) => __awaiter(this, void 0, void 0, function* () {
const isAuthenticated = yield this.isAuthenticated(this.routeData, authState);
if (!isAuthenticated) {
this.handleLogin(this.state.url, this.routeData);
}
});
const config = this.configService.getConfig();
if (!config) {
throw new Error('Okta config is not provided');
}
this.onAuthRequired = config.onAuthRequired;
// Unsubscribe updateAuthStateListener when route change
const router = injector.get(Router);
router.events.pipe(filter((e) => e instanceof NavigationStart && this.state && this.state.url !== e.url)).subscribe(() => {
this.oktaAuth.authStateManager.unsubscribe(this.updateAuthStateListener);
});
}
canLoad(route) {
var _a;
return __awaiter(this, void 0, void 0, function* () {
this.onAuthRequired = ((_a = route.data) === null || _a === void 0 ? void 0 : _a.onAuthRequired) || this.onAuthRequired;
const isAuthenticated = yield this.isAuthenticated(route.data);
if (isAuthenticated) {
return true;
}
const router = this.injector.get(Router);
const nav = router.getCurrentNavigation();
const originalUri = nav ? nav.extractedUrl.toString() : undefined;
yield this.handleLogin(originalUri, route.data);
return false;
});
}
/**
* Gateway for protected route. Returns true if there is a valid idToken,
* otherwise it will cache the route and start the login flow.
* @param route
* @param state
*/
canActivate(route, state) {
return __awaiter(this, void 0, void 0, function* () {
// Track states for current route
this.state = state;
this.routeData = route.data;
this.onAuthRequired = route.data && route.data.onAuthRequired || this.onAuthRequired;
// Protect the route after accessing
this.oktaAuth.authStateManager.subscribe(this.updateAuthStateListener);
const isAuthenticated = yield this.isAuthenticated(route.data);
if (isAuthenticated) {
return true;
}
yield this.handleLogin(state.url, route.data);
return false;
});
}
canActivateChild(route, state) {
return __awaiter(this, void 0, void 0, function* () {
return this.canActivate(route, state);
});
}
isAuthenticated(routeData, authState) {
var _a, _b, _c;
return __awaiter(this, void 0, void 0, function* () {
const isAuthenticated = authState ? authState === null || authState === void 0 ? void 0 : authState.isAuthenticated : yield this.oktaAuth.isAuthenticated();
let res = isAuthenticated;
if ((_a = routeData === null || routeData === void 0 ? void 0 : routeData.okta) === null || _a === void 0 ? void 0 : _a.acrValues) {
if (!authState) {
authState = this.oktaAuth.authStateManager.getAuthState();
}
res = ((_b = authState === null || authState === void 0 ? void 0 : authState.idToken) === null || _b === void 0 ? void 0 : _b.claims.acr) === ((_c = routeData === null || routeData === void 0 ? void 0 : routeData.okta) === null || _c === void 0 ? void 0 : _c.acrValues);
}
return res;
});
}
handleLogin(originalUri, routeData) {
var _a;
return __awaiter(this, void 0, void 0, function* () {
// Store the current path
if (originalUri) {
this.oktaAuth.setOriginalUri(originalUri);
}
const options = {};
if ((_a = routeData === null || routeData === void 0 ? void 0 : routeData.okta) === null || _a === void 0 ? void 0 : _a.acrValues) {
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore Supports auth-js >= 7.1.0
options.acrValues = routeData.okta.acrValues;
}
if (this.onAuthRequired) {
this.onAuthRequired(this.oktaAuth, this.injector, options);
}
else {
this.oktaAuth.signInWithRedirect(options);
}
});
}
}
OktaAuthGuard.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: OktaAuthGuard, deps: [{ token: OKTA_AUTH }, { token: i0.Injector }, { token: i1.OktaAuthConfigService }], target: i0.ɵɵFactoryTarget.Injectable });
OktaAuthGuard.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: OktaAuthGuard });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: OktaAuthGuard, decorators: [{
type: Injectable
}], ctorParameters: function () { return [{ type: i2.OktaAuth, decorators: [{
type: Inject,
args: [OKTA_AUTH]
}] }, { type: i0.Injector }, { type: i1.OktaAuthConfigService }]; } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"okta.guard.js","sourceRoot":"","sources":["../../../lib/src/okta/okta.guard.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2017-Present, Okta, Inc. and/or its affiliates. All rights reserved.\n * The Okta software accompanied by this notice is provided pursuant to the Apache License, Version 2.0 (the \"License.\")\n *\n * You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0.\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS, WITHOUT\n * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n *\n * See the License for the specific language governing permissions and limitations under the License.\n */\n\nimport { Injectable, Injector, Inject } from '@angular/core';\nimport {\n  CanActivate,\n  CanActivateChild,\n  ActivatedRouteSnapshot,\n  RouterStateSnapshot,\n  Router,\n  NavigationStart, \n  Event,\n  CanLoad,\n  Route,\n  Data\n} from '@angular/router';\nimport { filter } from 'rxjs/operators';\n\nimport { OktaAuth, AuthState, TokenParams } from '@okta/okta-auth-js';\nimport { OktaAuthConfigService } from './services/auth-config.serice';\nimport { AuthRequiredFunction, OKTA_AUTH } from './models/okta.config';\n\n@Injectable()\nexport class OktaAuthGuard implements CanActivate, CanActivateChild, CanLoad {\n  private state: RouterStateSnapshot;\n  private routeData: Data;\n  private onAuthRequired?: AuthRequiredFunction;\n\n  constructor(\n    @Inject(OKTA_AUTH) private oktaAuth: OktaAuth, \n    private injector: Injector,\n    private configService: OktaAuthConfigService\n  ) {\n    const config = this.configService.getConfig();\n    if (!config) {\n      throw new Error('Okta config is not provided');\n    }\n    this.onAuthRequired = config.onAuthRequired;\n\n    // Unsubscribe updateAuthStateListener when route change\n    const router = injector.get(Router);\n    router.events.pipe(\n      filter((e: Event) => e instanceof NavigationStart && this.state && this.state.url !== e.url)\n    ).subscribe(() => {\n      this.oktaAuth.authStateManager.unsubscribe(this.updateAuthStateListener);\n    });\n  }\n\n  async canLoad(route: Route): Promise<boolean> {\n    this.onAuthRequired = route.data?.onAuthRequired || this.onAuthRequired;\n\n    const isAuthenticated = await this.isAuthenticated(route.data);\n    if (isAuthenticated) {\n      return true;\n    }\n\n    const router = this.injector.get(Router);\n    const nav = router.getCurrentNavigation();\n    const originalUri = nav ? nav.extractedUrl.toString() : undefined;\n    await this.handleLogin(originalUri, route.data);\n\n    return false;\n  }\n\n  /**\n   * Gateway for protected route. Returns true if there is a valid idToken,\n   * otherwise it will cache the route and start the login flow.\n   * @param route\n   * @param state\n   */\n  async canActivate(route: ActivatedRouteSnapshot, state: RouterStateSnapshot): Promise<boolean> {\n    // Track states for current route\n    this.state = state;\n    this.routeData = route.data;\n    this.onAuthRequired = route.data && route.data.onAuthRequired || this.onAuthRequired;\n\n    // Protect the route after accessing\n    this.oktaAuth.authStateManager.subscribe(this.updateAuthStateListener);\n    const isAuthenticated = await this.isAuthenticated(route.data);\n    if (isAuthenticated) {\n      return true;\n    }\n\n    await this.handleLogin(state.url, route.data);\n\n    return false;\n  }\n\n  async canActivateChild(\n    route: ActivatedRouteSnapshot,\n    state: RouterStateSnapshot\n  ): Promise<boolean> {\n    return this.canActivate(route, state);\n  }\n\n  private async isAuthenticated(routeData?: Data, authState?: AuthState | null) {\n    const isAuthenticated = authState ? authState?.isAuthenticated : await this.oktaAuth.isAuthenticated();\n    let res = isAuthenticated;\n    if (routeData?.okta?.acrValues) {\n      if (!authState) {\n        authState = this.oktaAuth.authStateManager.getAuthState();\n      }\n      res = authState?.idToken?.claims.acr === routeData?.okta?.acrValues;\n    }\n    return res;\n  }\n\n  private async handleLogin(originalUri?: string, routeData?: Data): Promise<void> {\n    // Store the current path\n    if (originalUri) {\n      this.oktaAuth.setOriginalUri(originalUri);\n    }\n\n    const options: TokenParams = {};\n    if (routeData?.okta?.acrValues) {\n      // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n      // @ts-ignore Supports auth-js >= 7.1.0\n      options.acrValues = routeData.okta.acrValues;\n    }\n\n    if (this.onAuthRequired) {\n      this.onAuthRequired(this.oktaAuth, this.injector, options);\n    } else {\n      this.oktaAuth.signInWithRedirect(options);\n    }\n  }\n\n  private updateAuthStateListener = async (authState: AuthState) => {\n    const isAuthenticated = await this.isAuthenticated(this.routeData, authState);\n    if (!isAuthenticated) {\n      this.handleLogin(this.state.url, this.routeData);\n    }\n  };\n\n}\n"]}