@odx/auth 17.2.0 → 18.0.0
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.
- package/CHANGELOG.md +99 -0
- package/fesm2022/odx-auth-plugins-service-connect.mjs +3 -3
- package/fesm2022/odx-auth-plugins-service-connect.mjs.map +1 -1
- package/fesm2022/odx-auth.mjs +41 -41
- package/fesm2022/odx-auth.mjs.map +1 -1
- package/index.d.ts +744 -15
- package/package.json +4 -8
- package/plugins/service-connect/index.d.ts +184 -8
- package/esm2022/index.mjs +0 -16
- package/esm2022/lib/auth.component.mjs +0 -50
- package/esm2022/lib/auth.config.mjs +0 -71
- package/esm2022/lib/auth.directive.mjs +0 -51
- package/esm2022/lib/auth.guard.mjs +0 -44
- package/esm2022/lib/auth.i18n.mjs +0 -13
- package/esm2022/lib/auth.interceptor.mjs +0 -23
- package/esm2022/lib/auth.logger.mjs +0 -3
- package/esm2022/lib/auth.module.mjs +0 -19
- package/esm2022/lib/auth.providers.mjs +0 -136
- package/esm2022/lib/auth.service.mjs +0 -317
- package/esm2022/lib/auth.typings.mjs +0 -2
- package/esm2022/lib/components/auth-actions/auth-actions.component.mjs +0 -40
- package/esm2022/lib/components/auth-loading-screen/auth-loading-screen.component.mjs +0 -53
- package/esm2022/lib/components/index.mjs +0 -3
- package/esm2022/lib/directives/auth-action.directive.mjs +0 -30
- package/esm2022/lib/directives/index.mjs +0 -4
- package/esm2022/lib/directives/sign-in.directive.mjs +0 -52
- package/esm2022/lib/directives/sign-out.directive.mjs +0 -50
- package/esm2022/lib/helpers/create-auth-host-url.mjs +0 -18
- package/esm2022/lib/helpers/create-inititals.mjs +0 -32
- package/esm2022/lib/helpers/handle-auth-error.mjs +0 -31
- package/esm2022/lib/helpers/handle-oauth-event.mjs +0 -13
- package/esm2022/lib/helpers/index.mjs +0 -9
- package/esm2022/lib/helpers/resolve-email.mjs +0 -34
- package/esm2022/lib/helpers/resolve-username.mjs +0 -38
- package/esm2022/lib/helpers/set-http-auth-header.mjs +0 -22
- package/esm2022/lib/helpers/user-language-loader.mjs +0 -17
- package/esm2022/lib/models/auth-environment.mjs +0 -3
- package/esm2022/lib/models/auth-plugin-manager.mjs +0 -49
- package/esm2022/lib/models/authorized-handler.mjs +0 -2
- package/esm2022/lib/models/index.mjs +0 -5
- package/esm2022/lib/models/offline-auth-error-handler.mjs +0 -7
- package/esm2022/lib/plugins/core-debug.plugin.mjs +0 -29
- package/esm2022/lib/plugins/core-identity.plugin.mjs +0 -20
- package/esm2022/lib/plugins/index.mjs +0 -4
- package/esm2022/lib/plugins/user-profile-link.plugin.mjs +0 -18
- package/esm2022/lib/unauth.guard.mjs +0 -38
- package/esm2022/odx-auth.mjs +0 -5
- package/esm2022/plugins/service-connect/index.mjs +0 -9
- package/esm2022/plugins/service-connect/lib/dtos/get-service-connect-rights-response.dto.mjs +0 -2
- package/esm2022/plugins/service-connect/lib/dtos/get-service-connect-user-response.dto.mjs +0 -2
- package/esm2022/plugins/service-connect/lib/dtos/index.mjs +0 -3
- package/esm2022/plugins/service-connect/lib/helpers/build-service-connect-url.mjs +0 -16
- package/esm2022/plugins/service-connect/lib/helpers/has-roles-or-rights-handler.mjs +0 -12
- package/esm2022/plugins/service-connect/lib/helpers/has-roles-or-rights.mjs +0 -11
- package/esm2022/plugins/service-connect/lib/helpers/index.mjs +0 -5
- package/esm2022/plugins/service-connect/lib/helpers/service-connect-plugin-factory.mjs +0 -25
- package/esm2022/plugins/service-connect/lib/service-connect-rights.directive.mjs +0 -44
- package/esm2022/plugins/service-connect/lib/service-connect-rights.guard.mjs +0 -15
- package/esm2022/plugins/service-connect/lib/service-connect-rights.plugin.mjs +0 -16
- package/esm2022/plugins/service-connect/lib/service-connect-user-language.plugin.mjs +0 -34
- package/esm2022/plugins/service-connect/lib/service-connect-user-profile.plugin.mjs +0 -28
- package/esm2022/plugins/service-connect/lib/service-connect.config.mjs +0 -17
- package/esm2022/plugins/service-connect/odx-auth-plugins-service-connect.mjs +0 -5
- package/lib/auth.component.d.ts +0 -20
- package/lib/auth.config.d.ts +0 -84
- package/lib/auth.directive.d.ts +0 -25
- package/lib/auth.guard.d.ts +0 -11
- package/lib/auth.i18n.d.ts +0 -3
- package/lib/auth.interceptor.d.ts +0 -10
- package/lib/auth.logger.d.ts +0 -3
- package/lib/auth.module.d.ts +0 -10
- package/lib/auth.providers.d.ts +0 -46
- package/lib/auth.service.d.ts +0 -206
- package/lib/auth.typings.d.ts +0 -27
- package/lib/components/auth-actions/auth-actions.component.d.ts +0 -16
- package/lib/components/auth-loading-screen/auth-loading-screen.component.d.ts +0 -24
- package/lib/components/index.d.ts +0 -2
- package/lib/directives/auth-action.directive.d.ts +0 -18
- package/lib/directives/index.d.ts +0 -3
- package/lib/directives/sign-in.directive.d.ts +0 -29
- package/lib/directives/sign-out.directive.d.ts +0 -28
- package/lib/helpers/create-auth-host-url.d.ts +0 -14
- package/lib/helpers/create-inititals.d.ts +0 -18
- package/lib/helpers/handle-auth-error.d.ts +0 -13
- package/lib/helpers/handle-oauth-event.d.ts +0 -11
- package/lib/helpers/index.d.ts +0 -8
- package/lib/helpers/resolve-email.d.ts +0 -21
- package/lib/helpers/resolve-username.d.ts +0 -22
- package/lib/helpers/set-http-auth-header.d.ts +0 -11
- package/lib/helpers/user-language-loader.d.ts +0 -11
- package/lib/models/auth-environment.d.ts +0 -6
- package/lib/models/auth-plugin-manager.d.ts +0 -21
- package/lib/models/authorized-handler.d.ts +0 -3
- package/lib/models/index.d.ts +0 -4
- package/lib/models/offline-auth-error-handler.d.ts +0 -2
- package/lib/plugins/core-debug.plugin.d.ts +0 -11
- package/lib/plugins/core-identity.plugin.d.ts +0 -18
- package/lib/plugins/index.d.ts +0 -3
- package/lib/plugins/user-profile-link.plugin.d.ts +0 -18
- package/lib/unauth.guard.d.ts +0 -11
- package/plugins/service-connect/lib/dtos/get-service-connect-rights-response.dto.d.ts +0 -3
- package/plugins/service-connect/lib/dtos/get-service-connect-user-response.dto.d.ts +0 -14
- package/plugins/service-connect/lib/dtos/index.d.ts +0 -2
- package/plugins/service-connect/lib/helpers/build-service-connect-url.d.ts +0 -10
- package/plugins/service-connect/lib/helpers/has-roles-or-rights-handler.d.ts +0 -10
- package/plugins/service-connect/lib/helpers/has-roles-or-rights.d.ts +0 -11
- package/plugins/service-connect/lib/helpers/index.d.ts +0 -4
- package/plugins/service-connect/lib/helpers/service-connect-plugin-factory.d.ts +0 -20
- package/plugins/service-connect/lib/service-connect-rights.directive.d.ts +0 -21
- package/plugins/service-connect/lib/service-connect-rights.guard.d.ts +0 -11
- package/plugins/service-connect/lib/service-connect-rights.plugin.d.ts +0 -18
- package/plugins/service-connect/lib/service-connect-user-language.plugin.d.ts +0 -24
- package/plugins/service-connect/lib/service-connect-user-profile.plugin.d.ts +0 -23
- package/plugins/service-connect/lib/service-connect.config.d.ts +0 -15
|
@@ -1,136 +0,0 @@
|
|
|
1
|
-
import { provideHttpClient, withInterceptors } from '@angular/common/http';
|
|
2
|
-
import { APP_INITIALIZER, ENVIRONMENT_INITIALIZER, InjectionToken, inject, makeEnvironmentProviders } from '@angular/core';
|
|
3
|
-
import { WindowRef } from '@odx/angular';
|
|
4
|
-
import { DynamicViewService } from '@odx/angular/cdk/dynamic-view';
|
|
5
|
-
import { buildUrl } from '@odx/angular/utils';
|
|
6
|
-
import { OAuthStorage, provideOAuthClient } from 'angular-oauth2-oidc';
|
|
7
|
-
import { tap } from 'rxjs';
|
|
8
|
-
import { ODX_AUTH_DEFAULT_SCOPES, injectAuthConfig, provideAuthConfig } from './auth.config';
|
|
9
|
-
import { authInterceptor } from './auth.interceptor';
|
|
10
|
-
import logger from './auth.logger';
|
|
11
|
-
import { AuthService } from './auth.service';
|
|
12
|
-
import { AuthLoadingScreenComponent } from './components';
|
|
13
|
-
import { createAuthHostUrl, handleAuthError } from './helpers';
|
|
14
|
-
import { offlineAuthErrorHandler } from './models';
|
|
15
|
-
import { coreDebugPlugin, coreIdentityPlugin, userProfileUrlPlugin } from './plugins';
|
|
16
|
-
export const ODX_AUTH_ERROR_HANDLERS = new InjectionToken('@odx/auth::ErrorHandlers', {
|
|
17
|
-
providedIn: 'root',
|
|
18
|
-
factory: () => {
|
|
19
|
-
const { errorHandler } = injectAuthConfig();
|
|
20
|
-
return [offlineAuthErrorHandler, errorHandler];
|
|
21
|
-
},
|
|
22
|
-
});
|
|
23
|
-
export const ODX_AUTH_PLUGINS_ERROR_HANDLER = new InjectionToken('@odx/auth::PluginsErrorHandler', {
|
|
24
|
-
providedIn: 'root',
|
|
25
|
-
factory: () => injectAuthConfig().pluginsErrorHandler ?? null,
|
|
26
|
-
});
|
|
27
|
-
export const ODX_AUTH_CORE_PLUGINS = new InjectionToken('@odx/auth::CorePlugins', {
|
|
28
|
-
providedIn: 'root',
|
|
29
|
-
factory: () => {
|
|
30
|
-
const { showDebugInformation } = injectAuthConfig();
|
|
31
|
-
const plugins = [coreIdentityPlugin, userProfileUrlPlugin];
|
|
32
|
-
if (showDebugInformation) {
|
|
33
|
-
plugins.push(coreDebugPlugin);
|
|
34
|
-
}
|
|
35
|
-
return plugins;
|
|
36
|
-
},
|
|
37
|
-
});
|
|
38
|
-
export const ODX_AUTH_PLUGINS = new InjectionToken('@odx/auth::Plugins', {
|
|
39
|
-
providedIn: 'root',
|
|
40
|
-
factory: () => {
|
|
41
|
-
const { plugins } = injectAuthConfig();
|
|
42
|
-
const corePlugins = inject(ODX_AUTH_CORE_PLUGINS);
|
|
43
|
-
return [...corePlugins, ...plugins].map((pluginFactory) => pluginFactory());
|
|
44
|
-
},
|
|
45
|
-
});
|
|
46
|
-
/**
|
|
47
|
-
* Provides a logger for the authentication module.
|
|
48
|
-
*
|
|
49
|
-
* @returns {Provider} - The provider for the logger.
|
|
50
|
-
*/
|
|
51
|
-
export function provideAuthLogger() {
|
|
52
|
-
return {
|
|
53
|
-
provide: ENVIRONMENT_INITIALIZER,
|
|
54
|
-
useFactory: () => {
|
|
55
|
-
const { showDebugInformation } = injectAuthConfig();
|
|
56
|
-
if (!showDebugInformation)
|
|
57
|
-
logger.disable();
|
|
58
|
-
return () => void 0;
|
|
59
|
-
},
|
|
60
|
-
multi: true,
|
|
61
|
-
};
|
|
62
|
-
}
|
|
63
|
-
/**
|
|
64
|
-
* Initializes the authentication error handlers.
|
|
65
|
-
*/
|
|
66
|
-
export function initializeAuthErrorHandlers() {
|
|
67
|
-
const authService = inject(AuthService);
|
|
68
|
-
const handler = handleAuthError(inject(ODX_AUTH_ERROR_HANDLERS));
|
|
69
|
-
authService.errors$.pipe(tap((error) => handler(error))).subscribe();
|
|
70
|
-
}
|
|
71
|
-
/**
|
|
72
|
-
* Initializes the authentication configuration.
|
|
73
|
-
*
|
|
74
|
-
* @returns {() => Promise<void>} - A function that returns a promise resolving when the configuration is initialized.
|
|
75
|
-
*/
|
|
76
|
-
export function initalizeAuthConfig() {
|
|
77
|
-
const { clientId, scopes, redirectPath, environment, postLogoutRedirectUrl, issuer, timeoutFactor, discoveryUrl, enableLoadingScreen } = injectAuthConfig();
|
|
78
|
-
const authService = inject(AuthService);
|
|
79
|
-
const redirectUri = buildUrl(inject(WindowRef).getOrigin(), redirectPath);
|
|
80
|
-
const scope = Array.from(new Set(ODX_AUTH_DEFAULT_SCOPES.concat(scopes ?? []))).join(' ');
|
|
81
|
-
if (enableLoadingScreen) {
|
|
82
|
-
AuthLoadingScreenComponent.initialize(authService, inject(DynamicViewService));
|
|
83
|
-
}
|
|
84
|
-
return () => authService.initialize({
|
|
85
|
-
clientId,
|
|
86
|
-
issuer: issuer ?? createAuthHostUrl(environment, 'oauth2/default'),
|
|
87
|
-
scope,
|
|
88
|
-
redirectUri,
|
|
89
|
-
postLogoutRedirectUri: postLogoutRedirectUrl,
|
|
90
|
-
preserveRequestedRoute: true,
|
|
91
|
-
strictDiscoveryDocumentValidation: !discoveryUrl,
|
|
92
|
-
responseType: 'code',
|
|
93
|
-
showDebugInformation: false,
|
|
94
|
-
timeoutFactor,
|
|
95
|
-
waitForTokenInMsec: 1000,
|
|
96
|
-
});
|
|
97
|
-
}
|
|
98
|
-
/**
|
|
99
|
-
* Provides the authentication configuration and dependencies.
|
|
100
|
-
*
|
|
101
|
-
* @param {ConfigProvider<Partial<AuthConfig>, D>} config - The configuration provider.
|
|
102
|
-
* @returns {EnvironmentProviders} The environment providers for authentication.
|
|
103
|
-
* @template D
|
|
104
|
-
*
|
|
105
|
-
* @example Provide the authentication configuration and dependencies
|
|
106
|
-
* ```ts
|
|
107
|
-
* providers: [
|
|
108
|
-
* provideAuth({
|
|
109
|
-
* useFactory: ({ environment, auth: { clientId, loadUserProfile } }: ApplicationEnvironment) => ({
|
|
110
|
-
* environment,
|
|
111
|
-
* clientId,
|
|
112
|
-
* loadUserProfile,
|
|
113
|
-
* }),
|
|
114
|
-
* deps: [APPLICATION_ENVIRONMENT],
|
|
115
|
-
* }),
|
|
116
|
-
* ],
|
|
117
|
-
* ```
|
|
118
|
-
*/
|
|
119
|
-
export function provideAuth(config) {
|
|
120
|
-
return makeEnvironmentProviders([
|
|
121
|
-
provideAuthConfig(config),
|
|
122
|
-
provideAuthLogger(),
|
|
123
|
-
provideOAuthClient({ resourceServer: { sendAccessToken: false, allowedUrls: [] } }),
|
|
124
|
-
provideHttpClient(withInterceptors([authInterceptor])),
|
|
125
|
-
{
|
|
126
|
-
provide: APP_INITIALIZER,
|
|
127
|
-
useFactory: initalizeAuthConfig,
|
|
128
|
-
multi: true,
|
|
129
|
-
},
|
|
130
|
-
{
|
|
131
|
-
provide: OAuthStorage,
|
|
132
|
-
useFactory: () => injectAuthConfig().storage ?? inject(WindowRef).nativeWindow.localStorage,
|
|
133
|
-
},
|
|
134
|
-
]);
|
|
135
|
-
}
|
|
136
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth.providers.js","sourceRoot":"","sources":["../../../../../libs/auth/src/lib/auth.providers.ts"],"names":[],"mappings":"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","sourcesContent":["import { provideHttpClient, withInterceptors } from '@angular/common/http';\nimport { APP_INITIALIZER, ENVIRONMENT_INITIALIZER, EnvironmentProviders, InjectionToken, Provider, inject, makeEnvironmentProviders } from '@angular/core';\nimport { WindowRef } from '@odx/angular';\nimport { DynamicViewService } from '@odx/angular/cdk/dynamic-view';\nimport { ConfigDependencies, ConfigProvider, buildUrl } from '@odx/angular/utils';\nimport { OAuthStorage, provideOAuthClient } from 'angular-oauth2-oidc';\nimport { tap } from 'rxjs';\nimport { AuthConfig, AuthErrorHandlerFn, ODX_AUTH_DEFAULT_SCOPES, injectAuthConfig, provideAuthConfig } from './auth.config';\nimport { authInterceptor } from './auth.interceptor';\nimport logger from './auth.logger';\nimport { AuthService } from './auth.service';\nimport { AuthLoadingScreenComponent } from './components';\nimport { createAuthHostUrl, handleAuthError } from './helpers';\nimport { AuthPlugin, AuthPluginErrorHandler, AuthPluginFactory, offlineAuthErrorHandler } from './models';\nimport { coreDebugPlugin, coreIdentityPlugin, userProfileUrlPlugin } from './plugins';\n\nexport const ODX_AUTH_ERROR_HANDLERS = new InjectionToken<AuthErrorHandlerFn[]>('@odx/auth::ErrorHandlers', {\n  providedIn: 'root',\n  factory: () => {\n    const { errorHandler } = injectAuthConfig();\n\n    return [offlineAuthErrorHandler, errorHandler];\n  },\n});\n\nexport const ODX_AUTH_PLUGINS_ERROR_HANDLER = new InjectionToken<AuthPluginErrorHandler | null>('@odx/auth::PluginsErrorHandler', {\n  providedIn: 'root',\n  factory: () => injectAuthConfig().pluginsErrorHandler ?? null,\n});\n\nexport const ODX_AUTH_CORE_PLUGINS = new InjectionToken<AuthPluginFactory[]>('@odx/auth::CorePlugins', {\n  providedIn: 'root',\n  factory: () => {\n    const { showDebugInformation } = injectAuthConfig();\n    const plugins = [coreIdentityPlugin, userProfileUrlPlugin];\n    if (showDebugInformation) {\n      plugins.push(coreDebugPlugin);\n    }\n\n    return plugins;\n  },\n});\n\nexport const ODX_AUTH_PLUGINS = new InjectionToken<AuthPlugin[]>('@odx/auth::Plugins', {\n  providedIn: 'root',\n  factory: () => {\n    const { plugins } = injectAuthConfig();\n    const corePlugins = inject(ODX_AUTH_CORE_PLUGINS);\n\n    return [...corePlugins, ...plugins].map((pluginFactory) => pluginFactory());\n  },\n});\n\n/**\n * Provides a logger for the authentication module.\n *\n * @returns {Provider} - The provider for the logger.\n */\nexport function provideAuthLogger(): Provider {\n  return {\n    provide: ENVIRONMENT_INITIALIZER,\n    useFactory: () => {\n      const { showDebugInformation } = injectAuthConfig();\n      if (!showDebugInformation) logger.disable();\n\n      return () => void 0;\n    },\n    multi: true,\n  };\n}\n\n/**\n * Initializes the authentication error handlers.\n */\nexport function initializeAuthErrorHandlers(): void {\n  const authService = inject(AuthService);\n  const handler = handleAuthError(inject(ODX_AUTH_ERROR_HANDLERS));\n\n  authService.errors$.pipe(tap((error) => handler(error))).subscribe();\n}\n\n/**\n * Initializes the authentication configuration.\n *\n * @returns {() => Promise<void>} - A function that returns a promise resolving when the configuration is initialized.\n */\nexport function initalizeAuthConfig(): () => Promise<void> {\n  const { clientId, scopes, redirectPath, environment, postLogoutRedirectUrl, issuer, timeoutFactor, discoveryUrl, enableLoadingScreen } = injectAuthConfig();\n  const authService = inject(AuthService);\n  const redirectUri = buildUrl(inject(WindowRef).getOrigin(), redirectPath);\n  const scope = Array.from(new Set(ODX_AUTH_DEFAULT_SCOPES.concat(scopes ?? []))).join(' ');\n  if (enableLoadingScreen) {\n    AuthLoadingScreenComponent.initialize(authService, inject(DynamicViewService));\n  }\n  return () =>\n    authService.initialize({\n      clientId,\n      issuer: issuer ?? createAuthHostUrl(environment, 'oauth2/default'),\n      scope,\n      redirectUri,\n      postLogoutRedirectUri: postLogoutRedirectUrl,\n      preserveRequestedRoute: true,\n      strictDiscoveryDocumentValidation: !discoveryUrl,\n      responseType: 'code',\n      showDebugInformation: false,\n      timeoutFactor,\n      waitForTokenInMsec: 1000,\n    });\n}\n\n/**\n * Provides the authentication configuration and dependencies.\n *\n * @param {ConfigProvider<Partial<AuthConfig>, D>} config - The configuration provider.\n * @returns {EnvironmentProviders} The environment providers for authentication.\n * @template D\n *\n * @example Provide the authentication configuration and dependencies\n * ```ts\n * providers: [\n *  provideAuth({\n *   useFactory: ({ environment, auth: { clientId, loadUserProfile } }: ApplicationEnvironment) => ({\n *     environment,\n *     clientId,\n *     loadUserProfile,\n *   }),\n *   deps: [APPLICATION_ENVIRONMENT],\n *  }),\n * ],\n * ```\n */\nexport function provideAuth<D extends ConfigDependencies>(config: ConfigProvider<Partial<AuthConfig>, D>): EnvironmentProviders {\n  return makeEnvironmentProviders([\n    provideAuthConfig(config),\n    provideAuthLogger(),\n    provideOAuthClient({ resourceServer: { sendAccessToken: false, allowedUrls: [] } }),\n    provideHttpClient(withInterceptors([authInterceptor])),\n    {\n      provide: APP_INITIALIZER,\n      useFactory: initalizeAuthConfig,\n      multi: true,\n    },\n    {\n      provide: OAuthStorage,\n      useFactory: () => injectAuthConfig().storage ?? inject(WindowRef).nativeWindow.localStorage,\n    },\n  ]);\n}\n"]}
|
|
@@ -1,317 +0,0 @@
|
|
|
1
|
-
import { HttpErrorResponse } from '@angular/common/http';
|
|
2
|
-
import { Injectable, inject } from '@angular/core';
|
|
3
|
-
import { Router } from '@angular/router';
|
|
4
|
-
import { WindowRef } from '@odx/angular';
|
|
5
|
-
import { OAuthErrorEvent, OAuthService } from 'angular-oauth2-oidc';
|
|
6
|
-
import { deepmerge } from 'deepmerge-ts';
|
|
7
|
-
import { BehaviorSubject, catchError, combineLatest, distinctUntilChanged, filter, fromEvent, map, merge, of, share, shareReplay, startWith, switchMap, take, tap, timeout, } from 'rxjs';
|
|
8
|
-
import { injectAuthConfig } from './auth.config';
|
|
9
|
-
import { setHttpAuthHeader } from './helpers';
|
|
10
|
-
import { AuthPluginManager } from './models';
|
|
11
|
-
import * as i0 from "@angular/core";
|
|
12
|
-
/**
|
|
13
|
-
* The `AuthService` class provides authentication functionality for an Angular application.
|
|
14
|
-
* It handles OAuth2/OIDC authentication, token management, and user identity claims.
|
|
15
|
-
*
|
|
16
|
-
* Key responsibilities include:
|
|
17
|
-
* - Initializing authentication with a provided configuration.
|
|
18
|
-
* - Managing tokens (access, refresh, and ID tokens).
|
|
19
|
-
* - Checking and emitting authentication and authorization states.
|
|
20
|
-
* - Handling user login and logout flows.
|
|
21
|
-
* - Supporting silent refresh and offline authentication scenarios.
|
|
22
|
-
* - Integrating authentication plugins via `AuthPluginManager`.
|
|
23
|
-
*
|
|
24
|
-
* @example
|
|
25
|
-
* ```typescript
|
|
26
|
-
* // Injecting the AuthService
|
|
27
|
-
* constructor(private authService: AuthService) {}
|
|
28
|
-
*
|
|
29
|
-
* // Using the AuthService to initialize authentication
|
|
30
|
-
* async ngOnInit() {
|
|
31
|
-
* const config: AuthConfig = { clientId: 'your-client-id', discoveryUrl: 'https://example.com/.well-known/openid-configuration' };
|
|
32
|
-
* await this.authService.initialize(config);
|
|
33
|
-
* }
|
|
34
|
-
* ```
|
|
35
|
-
*/
|
|
36
|
-
export class AuthService {
|
|
37
|
-
/**
|
|
38
|
-
* Emits whenever the `access_token` in local storage is updated or cleared.
|
|
39
|
-
* Provides an observable for tracking token updates.
|
|
40
|
-
*
|
|
41
|
-
* @type {Observable<StorageEvent | null>}
|
|
42
|
-
*/
|
|
43
|
-
get accessTokenUpdate$() {
|
|
44
|
-
return this.onAccessTokenUpdate$;
|
|
45
|
-
}
|
|
46
|
-
constructor() {
|
|
47
|
-
this.authConfig = injectAuthConfig();
|
|
48
|
-
this.authPluginManager = inject(AuthPluginManager);
|
|
49
|
-
this.oauthService = inject(OAuthService);
|
|
50
|
-
this.router = inject(Router);
|
|
51
|
-
this.windowRef = inject(WindowRef);
|
|
52
|
-
this.isInitialized$$ = new BehaviorSubject(false);
|
|
53
|
-
this.isRedirecting$$ = new BehaviorSubject(false);
|
|
54
|
-
this.onAccessTokenUpdate$ = fromEvent(this.windowRef.nativeWindow, 'storage').pipe(filter(({ key }) => key === 'access_token' || key === null), startWith(null), share());
|
|
55
|
-
this.silentRefreshHandler$ = this.isInitialized$$.pipe(filter(Boolean), switchMap(() => this.windowRef.isOnline$), tap((isOnline) => this.updateSilentRefresh(isOnline)));
|
|
56
|
-
this.onAuthStateChange$ = combineLatest([this.oauthService.events.pipe(startWith(null)), this.windowRef.isOnline$, this.onAccessTokenUpdate$]);
|
|
57
|
-
/**
|
|
58
|
-
* Emits `true` when the service has completed initialization.
|
|
59
|
-
* Emits `false` until the initialization is complete.
|
|
60
|
-
*
|
|
61
|
-
* @type {Observable<boolean>}
|
|
62
|
-
*/
|
|
63
|
-
this.isInitialized$ = this.isInitialized$$.pipe(filter(Boolean), distinctUntilChanged());
|
|
64
|
-
/**
|
|
65
|
-
* Emits `true` when the user is being redirected to the login page.
|
|
66
|
-
* Emits `false` when there is no redirection in progress.
|
|
67
|
-
*
|
|
68
|
-
* @type {Observable<boolean>}
|
|
69
|
-
*/
|
|
70
|
-
this.isRedirecting$ = this.isRedirecting$$.pipe(distinctUntilChanged(), shareReplay({ bufferSize: 1, refCount: true }));
|
|
71
|
-
/**
|
|
72
|
-
* Emits `true` when the application is in a loading state (e.g., during redirection or plugin initialization).
|
|
73
|
-
* Emits `false` when there is no ongoing loading activity.
|
|
74
|
-
*
|
|
75
|
-
* @type {Observable<boolean>}
|
|
76
|
-
*/
|
|
77
|
-
this.isLoading$ = merge(this.isRedirecting$, this.authPluginManager.pluginsLoading$).pipe(distinctUntilChanged());
|
|
78
|
-
/**
|
|
79
|
-
* Emits `true` when the user is authenticated.
|
|
80
|
-
* Emits `false` when the user is not authenticated.
|
|
81
|
-
*
|
|
82
|
-
* @type {Observable<boolean>}
|
|
83
|
-
*/
|
|
84
|
-
this.isAuthenticated$ = this.isInitialized$.pipe(switchMap(() => this.authPluginManager.pluginsReady$), switchMap(() => this.onAuthStateChange$), map(() => this.isAuthenticated()), distinctUntilChanged(), shareReplay({ bufferSize: 1, refCount: true }));
|
|
85
|
-
/**
|
|
86
|
-
* Emits the identity claims of the authenticated user.
|
|
87
|
-
* If the user is not authenticated, emits `null`.
|
|
88
|
-
*
|
|
89
|
-
* @type {Observable<OdxAuth.IdentityClaims | null>}
|
|
90
|
-
*/
|
|
91
|
-
this.identityClaims$ = this.isAuthenticated$.pipe(map(() => this.getIdentityClaims()), shareReplay({ bufferSize: 1, refCount: true }));
|
|
92
|
-
/**
|
|
93
|
-
* Emits OAuth error events.
|
|
94
|
-
*
|
|
95
|
-
* @type {Observable<OAuthErrorEvent>}
|
|
96
|
-
*/
|
|
97
|
-
this.errors$ = this.oauthService.events.pipe(filter((event) => event instanceof OAuthErrorEvent), share());
|
|
98
|
-
/**
|
|
99
|
-
* Emits events when an OAuth token is successfully received.
|
|
100
|
-
*
|
|
101
|
-
* @type {Observable<Event>}
|
|
102
|
-
*/
|
|
103
|
-
this.onTokenReceived$ = this.oauthService.events.pipe(filter((event) => event.type === 'token_received'), share());
|
|
104
|
-
this.runPlugins();
|
|
105
|
-
this.silentRefreshHandler$.subscribe();
|
|
106
|
-
}
|
|
107
|
-
/**
|
|
108
|
-
* Initializes the authentication service with the provided configuration.
|
|
109
|
-
*
|
|
110
|
-
* @param {AuthConfig} config - The authentication configuration object.
|
|
111
|
-
* @returns {Promise<void>} Resolves when initialization is complete.
|
|
112
|
-
*/
|
|
113
|
-
async initialize(config) {
|
|
114
|
-
this.assertAudience(config.clientId);
|
|
115
|
-
this.oauthService.configure({ ...config, openUri: this.redirectToLogin.bind(this) });
|
|
116
|
-
try {
|
|
117
|
-
await this.oauthService.loadDiscoveryDocument(this.authConfig.discoveryUrl);
|
|
118
|
-
await this.oauthService.tryLogin();
|
|
119
|
-
if (this.authConfig.refreshTokenOnInit) {
|
|
120
|
-
await this.tryRefreshToken();
|
|
121
|
-
}
|
|
122
|
-
}
|
|
123
|
-
catch (error) {
|
|
124
|
-
if (!(error instanceof HttpErrorResponse || error instanceof OAuthErrorEvent)) {
|
|
125
|
-
throw error;
|
|
126
|
-
}
|
|
127
|
-
if (error.type === 'token_refresh_error' || (error instanceof HttpErrorResponse && (error.status === 401 || error.status === 400))) {
|
|
128
|
-
this.signOut(true);
|
|
129
|
-
this.signIn(this.windowRef.location.pathname);
|
|
130
|
-
}
|
|
131
|
-
}
|
|
132
|
-
this.isInitialized$$.next(true);
|
|
133
|
-
if (this.authConfig.requireSignIn && !this.isAuthenticated()) {
|
|
134
|
-
this.signIn(this.windowRef.location.pathname);
|
|
135
|
-
}
|
|
136
|
-
await this.tryLoadUserProfile();
|
|
137
|
-
await this.routeToRequestedUrl();
|
|
138
|
-
}
|
|
139
|
-
/**
|
|
140
|
-
* Runs all authentication plugins registered in the `AuthPluginManager`.
|
|
141
|
-
*/
|
|
142
|
-
runPlugins() {
|
|
143
|
-
this.authPluginManager.runPlugins(this).subscribe();
|
|
144
|
-
}
|
|
145
|
-
/**
|
|
146
|
-
* Returns the issuer URL for the OAuth server.
|
|
147
|
-
*
|
|
148
|
-
* @returns {URL} The issuer URL.
|
|
149
|
-
*/
|
|
150
|
-
getIssuer() {
|
|
151
|
-
return new URL(this.oauthService.issuer);
|
|
152
|
-
}
|
|
153
|
-
/**
|
|
154
|
-
* Initiates the login flow for the user. Redirects to the login page.
|
|
155
|
-
*
|
|
156
|
-
* @param {string} [url] - The URL to redirect back to after login.
|
|
157
|
-
*/
|
|
158
|
-
signIn(url) {
|
|
159
|
-
this.oauthService.initLoginFlow(url);
|
|
160
|
-
}
|
|
161
|
-
/**
|
|
162
|
-
* Logs the user out and clears their tokens.
|
|
163
|
-
*
|
|
164
|
-
* @param {boolean} [noRedirect] - If `true`, no redirection occurs after logout.
|
|
165
|
-
*/
|
|
166
|
-
signOut(noRedirect) {
|
|
167
|
-
this.oauthService.revokeTokenAndLogout(noRedirect || !this.getAccessToken());
|
|
168
|
-
}
|
|
169
|
-
/**
|
|
170
|
-
* Attempts to refresh the user's tokens.
|
|
171
|
-
*
|
|
172
|
-
* @returns {Promise<TokenResponse>} Resolves with the new token response.
|
|
173
|
-
*/
|
|
174
|
-
async refreshTokens() {
|
|
175
|
-
return this.oauthService.refreshToken();
|
|
176
|
-
}
|
|
177
|
-
/**
|
|
178
|
-
* Retrieves the current access token, if available.
|
|
179
|
-
*
|
|
180
|
-
* @returns {string | null} The access token, or `null` if not available.
|
|
181
|
-
*/
|
|
182
|
-
getAccessToken() {
|
|
183
|
-
return this.oauthService.getAccessToken() ?? null;
|
|
184
|
-
}
|
|
185
|
-
/**
|
|
186
|
-
* Retrieves the current refresh token, if available.
|
|
187
|
-
*
|
|
188
|
-
* @returns {string | null} The refresh token, or `null` if not available.
|
|
189
|
-
*/
|
|
190
|
-
getRefreshToken() {
|
|
191
|
-
return this.oauthService.getRefreshToken() ?? null;
|
|
192
|
-
}
|
|
193
|
-
/**
|
|
194
|
-
* Retrieves the current ID token, if available.
|
|
195
|
-
*
|
|
196
|
-
* @returns {string | null} The ID token, or `null` if not available.
|
|
197
|
-
*/
|
|
198
|
-
getIdToken() {
|
|
199
|
-
return this.oauthService.getIdToken() ?? null;
|
|
200
|
-
}
|
|
201
|
-
/**
|
|
202
|
-
* Retrieves the identity claims of the authenticated user.
|
|
203
|
-
*
|
|
204
|
-
* @returns {OdxAuth.IdentityClaims | null} The identity claims, or `null` if not available.
|
|
205
|
-
*/
|
|
206
|
-
getIdentityClaims() {
|
|
207
|
-
if (!this.getIdToken())
|
|
208
|
-
return null;
|
|
209
|
-
return deepmerge(this.oauthService.getIdentityClaims(), this.authPluginManager.getResult());
|
|
210
|
-
}
|
|
211
|
-
/**
|
|
212
|
-
* Retrieves the raw identity claims of the authenticated user.
|
|
213
|
-
*
|
|
214
|
-
* @returns {OdxAuth.RawIdentityClaims | null} The raw identity claims, or `null` if not available.
|
|
215
|
-
*/
|
|
216
|
-
getRawIdentityClaims() {
|
|
217
|
-
if (!this.getIdToken())
|
|
218
|
-
return null;
|
|
219
|
-
return this.oauthService.getIdentityClaims();
|
|
220
|
-
}
|
|
221
|
-
/**
|
|
222
|
-
* Checks if the user is currently authenticated.
|
|
223
|
-
*
|
|
224
|
-
* @returns {boolean} `true` if authenticated, otherwise `false`.
|
|
225
|
-
*/
|
|
226
|
-
isAuthenticated() {
|
|
227
|
-
if (this.windowRef.isOnline()) {
|
|
228
|
-
return this.oauthService.hasValidAccessToken() && this.oauthService.hasValidIdToken();
|
|
229
|
-
}
|
|
230
|
-
return this.hasValidOfflineToken();
|
|
231
|
-
}
|
|
232
|
-
/**
|
|
233
|
-
* Checks if the user is authorized based on the provided handler.
|
|
234
|
-
*
|
|
235
|
-
* @param {AuthorizedHandler | null} [authorizedHandler] - A handler to determine authorization.
|
|
236
|
-
* @returns {boolean} `true` if authorized, otherwise `false`.
|
|
237
|
-
*/
|
|
238
|
-
isAuthorized(authorizedHandler) {
|
|
239
|
-
const handler = authorizedHandler ?? this.authConfig.defaultAuthorizedHandler;
|
|
240
|
-
return this.isAuthenticated() && (handler?.(this.getIdentityClaims(), this, this.router) ?? true);
|
|
241
|
-
}
|
|
242
|
-
/**
|
|
243
|
-
* Emits whether the user is authorized based on the provided handler.
|
|
244
|
-
*
|
|
245
|
-
* @param {AuthorizedHandler | null} [authorizedHandler] - A handler to determine authorization.
|
|
246
|
-
* @returns {Observable<boolean>} An observable emitting the authorization status.
|
|
247
|
-
*/
|
|
248
|
-
isAuthorized$(authorizedHandler) {
|
|
249
|
-
return this.isAuthenticated$.pipe(map(() => this.isAuthorized(authorizedHandler)));
|
|
250
|
-
}
|
|
251
|
-
/**
|
|
252
|
-
* Prepares an HTTP request by adding the access token to its headers.
|
|
253
|
-
*
|
|
254
|
-
* @param {HttpRequest<T> | Request} req - The HTTP request to prepare.
|
|
255
|
-
* @param {boolean} requireSignIn - Whether to require the user to sign in if no token is available.
|
|
256
|
-
* @returns {Observable<R>} An observable emitting the prepared request.
|
|
257
|
-
* @template R, T
|
|
258
|
-
*/
|
|
259
|
-
prepareAuthRequest$(req, requireSignIn = false) {
|
|
260
|
-
return this.waitForAccessToken$(requireSignIn).pipe(map((token) => setHttpAuthHeader(req, token)));
|
|
261
|
-
}
|
|
262
|
-
/**
|
|
263
|
-
* Waits for a valid access token to become available.
|
|
264
|
-
*
|
|
265
|
-
* @param {boolean} requireSignIn - Whether to require the user to sign in if no token is available.
|
|
266
|
-
* @returns {Observable<string | null>} An observable emitting the access token.
|
|
267
|
-
*/
|
|
268
|
-
waitForAccessToken$(requireSignIn) {
|
|
269
|
-
const accessToken$ = of(this.getAccessToken()).pipe(filter((token) => !!token && this.isAuthenticated()));
|
|
270
|
-
const waitForAccessToken$ = this.onTokenReceived$.pipe(timeout(this.authConfig.waitForTokenInMs ?? 0), catchError(async () => this.tryRefreshToken().catch(() => null)));
|
|
271
|
-
return this.isInitialized$.pipe(switchMap(() => merge(accessToken$, waitForAccessToken$)), take(1), map(() => this.getAccessToken()), tap((token) => !token && requireSignIn && this.signIn(this.windowRef.location.pathname)));
|
|
272
|
-
}
|
|
273
|
-
async routeToRequestedUrl() {
|
|
274
|
-
if (!this.oauthService.state || !this.oauthService.hasValidAccessToken())
|
|
275
|
-
return;
|
|
276
|
-
await this.router.navigateByUrl(decodeURIComponent(this.oauthService.state));
|
|
277
|
-
}
|
|
278
|
-
hasValidOfflineToken() {
|
|
279
|
-
const issuedAt = this.getIdentityClaims()?.iat ?? 0;
|
|
280
|
-
return Date.now() - issuedAt * 1000 <= this.authConfig.maxOfflineTime * 1000;
|
|
281
|
-
}
|
|
282
|
-
redirectToLogin(uri) {
|
|
283
|
-
this.windowRef.location.assign(uri);
|
|
284
|
-
this.isRedirecting$$.next(true);
|
|
285
|
-
}
|
|
286
|
-
async tryLoadUserProfile() {
|
|
287
|
-
if (!this.authConfig.loadUserProfile || !this.isAuthenticated())
|
|
288
|
-
return;
|
|
289
|
-
await this.oauthService.loadUserProfile().catch(() => null);
|
|
290
|
-
}
|
|
291
|
-
async tryRefreshToken() {
|
|
292
|
-
if (!this.getRefreshToken())
|
|
293
|
-
return;
|
|
294
|
-
await this.oauthService.refreshToken();
|
|
295
|
-
}
|
|
296
|
-
assertAudience(clientId) {
|
|
297
|
-
const tokenClientId = this.getIdentityClaims()?.aud;
|
|
298
|
-
if (!clientId || !tokenClientId || clientId === tokenClientId)
|
|
299
|
-
return;
|
|
300
|
-
this.signOut(true);
|
|
301
|
-
}
|
|
302
|
-
updateSilentRefresh(enabled) {
|
|
303
|
-
if (enabled) {
|
|
304
|
-
this.oauthService.setupAutomaticSilentRefresh();
|
|
305
|
-
}
|
|
306
|
-
else {
|
|
307
|
-
this.oauthService.stopAutomaticRefresh();
|
|
308
|
-
}
|
|
309
|
-
}
|
|
310
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AuthService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
311
|
-
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AuthService, providedIn: 'root' }); }
|
|
312
|
-
}
|
|
313
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AuthService, decorators: [{
|
|
314
|
-
type: Injectable,
|
|
315
|
-
args: [{ providedIn: 'root' }]
|
|
316
|
-
}], ctorParameters: () => [] });
|
|
317
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth.service.js","sourceRoot":"","sources":["../../../../../libs/auth/src/lib/auth.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpErrorResponse, HttpRequest } from '@angular/common/http';\nimport { Injectable, inject } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { WindowRef } from '@odx/angular';\nimport { AuthConfig, OAuthErrorEvent, OAuthService, TokenResponse } from 'angular-oauth2-oidc';\nimport { deepmerge } from 'deepmerge-ts';\nimport {\n  BehaviorSubject,\n  Observable,\n  catchError,\n  combineLatest,\n  distinctUntilChanged,\n  filter,\n  fromEvent,\n  map,\n  merge,\n  of,\n  share,\n  shareReplay,\n  startWith,\n  switchMap,\n  take,\n  tap,\n  timeout,\n} from 'rxjs';\nimport { injectAuthConfig } from './auth.config';\nimport { setHttpAuthHeader } from './helpers';\nimport { AuthPluginManager, AuthorizedHandler } from './models';\n\n/**\n * The `AuthService` class provides authentication functionality for an Angular application.\n * It handles OAuth2/OIDC authentication, token management, and user identity claims.\n *\n * Key responsibilities include:\n * - Initializing authentication with a provided configuration.\n * - Managing tokens (access, refresh, and ID tokens).\n * - Checking and emitting authentication and authorization states.\n * - Handling user login and logout flows.\n * - Supporting silent refresh and offline authentication scenarios.\n * - Integrating authentication plugins via `AuthPluginManager`.\n *\n * @example\n * ```typescript\n * // Injecting the AuthService\n * constructor(private authService: AuthService) {}\n *\n * // Using the AuthService to initialize authentication\n * async ngOnInit() {\n *   const config: AuthConfig = { clientId: 'your-client-id', discoveryUrl: 'https://example.com/.well-known/openid-configuration' };\n *   await this.authService.initialize(config);\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class AuthService {\n  private readonly authConfig = injectAuthConfig();\n  private readonly authPluginManager = inject(AuthPluginManager);\n  private readonly oauthService = inject(OAuthService);\n  private readonly router = inject(Router);\n  private readonly windowRef = inject(WindowRef);\n\n  private readonly isInitialized$$ = new BehaviorSubject(false);\n  private readonly isRedirecting$$ = new BehaviorSubject(false);\n  private readonly onAccessTokenUpdate$ = fromEvent<StorageEvent>(this.windowRef.nativeWindow, 'storage').pipe(\n    filter(({ key }) => key === 'access_token' || key === null),\n    startWith(null),\n    share(),\n  );\n  private readonly silentRefreshHandler$ = this.isInitialized$$.pipe(\n    filter(Boolean),\n    switchMap(() => this.windowRef.isOnline$),\n    tap((isOnline) => this.updateSilentRefresh(isOnline)),\n  );\n  private readonly onAuthStateChange$ = combineLatest([this.oauthService.events.pipe(startWith(null)), this.windowRef.isOnline$, this.onAccessTokenUpdate$]);\n\n  /**\n   * Emits `true` when the service has completed initialization.\n   * Emits `false` until the initialization is complete.\n   *\n   * @type {Observable<boolean>}\n   */\n  public readonly isInitialized$ = this.isInitialized$$.pipe(filter(Boolean), distinctUntilChanged());\n\n  /**\n   * Emits `true` when the user is being redirected to the login page.\n   * Emits `false` when there is no redirection in progress.\n   *\n   * @type {Observable<boolean>}\n   */\n  public readonly isRedirecting$ = this.isRedirecting$$.pipe(distinctUntilChanged(), shareReplay({ bufferSize: 1, refCount: true }));\n\n  /**\n   * Emits `true` when the application is in a loading state (e.g., during redirection or plugin initialization).\n   * Emits `false` when there is no ongoing loading activity.\n   *\n   * @type {Observable<boolean>}\n   */\n  public readonly isLoading$ = merge(this.isRedirecting$, this.authPluginManager.pluginsLoading$).pipe(distinctUntilChanged());\n\n  /**\n   * Emits `true` when the user is authenticated.\n   * Emits `false` when the user is not authenticated.\n   *\n   * @type {Observable<boolean>}\n   */\n  public readonly isAuthenticated$ = this.isInitialized$.pipe(\n    switchMap(() => this.authPluginManager.pluginsReady$),\n    switchMap(() => this.onAuthStateChange$),\n    map(() => this.isAuthenticated()),\n    distinctUntilChanged(),\n    shareReplay({ bufferSize: 1, refCount: true }),\n  );\n\n  /**\n   * Emits the identity claims of the authenticated user.\n   * If the user is not authenticated, emits `null`.\n   *\n   * @type {Observable<OdxAuth.IdentityClaims | null>}\n   */\n  public readonly identityClaims$ = this.isAuthenticated$.pipe(\n    map(() => this.getIdentityClaims()),\n    shareReplay({ bufferSize: 1, refCount: true }),\n  );\n\n  /**\n   * Emits OAuth error events.\n   *\n   * @type {Observable<OAuthErrorEvent>}\n   */\n  public readonly errors$ = this.oauthService.events.pipe(\n    filter((event): event is OAuthErrorEvent => event instanceof OAuthErrorEvent),\n    share(),\n  );\n\n  /**\n   * Emits events when an OAuth token is successfully received.\n   *\n   * @type {Observable<Event>}\n   */\n  public readonly onTokenReceived$ = this.oauthService.events.pipe(\n    filter((event) => event.type === 'token_received'),\n    share(),\n  );\n\n  /**\n   * Emits whenever the `access_token` in local storage is updated or cleared.\n   * Provides an observable for tracking token updates.\n   *\n   * @type {Observable<StorageEvent | null>}\n   */\n  public get accessTokenUpdate$(): Observable<StorageEvent | null> {\n    return this.onAccessTokenUpdate$;\n  }\n\n  constructor() {\n    this.runPlugins();\n    this.silentRefreshHandler$.subscribe();\n  }\n\n  /**\n   * Initializes the authentication service with the provided configuration.\n   *\n   * @param {AuthConfig} config - The authentication configuration object.\n   * @returns {Promise<void>} Resolves when initialization is complete.\n   */\n  public async initialize(config: AuthConfig): Promise<void> {\n    this.assertAudience(config.clientId);\n    this.oauthService.configure({ ...config, openUri: this.redirectToLogin.bind(this) });\n    try {\n      await this.oauthService.loadDiscoveryDocument(this.authConfig.discoveryUrl);\n      await this.oauthService.tryLogin();\n      if (this.authConfig.refreshTokenOnInit) {\n        await this.tryRefreshToken();\n      }\n    } catch (error) {\n      if (!(error instanceof HttpErrorResponse || error instanceof OAuthErrorEvent)) {\n        throw error;\n      }\n      if (error.type === 'token_refresh_error' || (error instanceof HttpErrorResponse && (error.status === 401 || error.status === 400))) {\n        this.signOut(true);\n        this.signIn(this.windowRef.location.pathname);\n      }\n    }\n    this.isInitialized$$.next(true);\n    if (this.authConfig.requireSignIn && !this.isAuthenticated()) {\n      this.signIn(this.windowRef.location.pathname);\n    }\n    await this.tryLoadUserProfile();\n    await this.routeToRequestedUrl();\n  }\n\n  /**\n   * Runs all authentication plugins registered in the `AuthPluginManager`.\n   */\n  public runPlugins(): void {\n    this.authPluginManager.runPlugins(this).subscribe();\n  }\n\n  /**\n   * Returns the issuer URL for the OAuth server.\n   *\n   * @returns {URL} The issuer URL.\n   */\n  public getIssuer(): URL {\n    return new URL(this.oauthService.issuer!);\n  }\n\n  /**\n   * Initiates the login flow for the user. Redirects to the login page.\n   *\n   * @param {string} [url] - The URL to redirect back to after login.\n   */\n  public signIn(url?: string): void {\n    this.oauthService.initLoginFlow(url);\n  }\n\n  /**\n   * Logs the user out and clears their tokens.\n   *\n   * @param {boolean} [noRedirect] - If `true`, no redirection occurs after logout.\n   */\n  public signOut(noRedirect?: boolean): void {\n    this.oauthService.revokeTokenAndLogout(noRedirect || !this.getAccessToken());\n  }\n\n  /**\n   * Attempts to refresh the user's tokens.\n   *\n   * @returns {Promise<TokenResponse>} Resolves with the new token response.\n   */\n  public async refreshTokens(): Promise<TokenResponse> {\n    return this.oauthService.refreshToken();\n  }\n\n  /**\n   * Retrieves the current access token, if available.\n   *\n   * @returns {string | null} The access token, or `null` if not available.\n   */\n  public getAccessToken(): string | null {\n    return this.oauthService.getAccessToken() ?? null;\n  }\n\n  /**\n   * Retrieves the current refresh token, if available.\n   *\n   * @returns {string | null} The refresh token, or `null` if not available.\n   */\n  public getRefreshToken(): string | null {\n    return this.oauthService.getRefreshToken() ?? null;\n  }\n\n  /**\n   * Retrieves the current ID token, if available.\n   *\n   * @returns {string | null} The ID token, or `null` if not available.\n   */\n  public getIdToken(): string | null {\n    return this.oauthService.getIdToken() ?? null;\n  }\n\n  /**\n   * Retrieves the identity claims of the authenticated user.\n   *\n   * @returns {OdxAuth.IdentityClaims | null} The identity claims, or `null` if not available.\n   */\n  public getIdentityClaims(): OdxAuth.IdentityClaims | null {\n    if (!this.getIdToken()) return null;\n    return deepmerge(this.oauthService.getIdentityClaims(), this.authPluginManager.getResult()) as OdxAuth.IdentityClaims;\n  }\n\n  /**\n   * Retrieves the raw identity claims of the authenticated user.\n   *\n   * @returns {OdxAuth.RawIdentityClaims | null} The raw identity claims, or `null` if not available.\n   */\n  public getRawIdentityClaims(): OdxAuth.RawIdentityClaims | null {\n    if (!this.getIdToken()) return null;\n    return this.oauthService.getIdentityClaims();\n  }\n\n  /**\n   * Checks if the user is currently authenticated.\n   *\n   * @returns {boolean} `true` if authenticated, otherwise `false`.\n   */\n  public isAuthenticated(): boolean {\n    if (this.windowRef.isOnline()) {\n      return this.oauthService.hasValidAccessToken() && this.oauthService.hasValidIdToken();\n    }\n    return this.hasValidOfflineToken();\n  }\n\n  /**\n   * Checks if the user is authorized based on the provided handler.\n   *\n   * @param {AuthorizedHandler | null} [authorizedHandler] - A handler to determine authorization.\n   * @returns {boolean} `true` if authorized, otherwise `false`.\n   */\n  public isAuthorized(authorizedHandler?: AuthorizedHandler | null): boolean {\n    const handler = authorizedHandler ?? this.authConfig.defaultAuthorizedHandler;\n\n    return this.isAuthenticated() && (handler?.(this.getIdentityClaims(), this, this.router) ?? true);\n  }\n\n  /**\n   * Emits whether the user is authorized based on the provided handler.\n   *\n   * @param {AuthorizedHandler | null} [authorizedHandler] - A handler to determine authorization.\n   * @returns {Observable<boolean>} An observable emitting the authorization status.\n   */\n  public isAuthorized$(authorizedHandler?: AuthorizedHandler | null): Observable<boolean> {\n    return this.isAuthenticated$.pipe(map(() => this.isAuthorized(authorizedHandler)));\n  }\n\n  /**\n   * Prepares an HTTP request by adding the access token to its headers.\n   *\n   * @param {HttpRequest<T> | Request} req - The HTTP request to prepare.\n   * @param {boolean} requireSignIn - Whether to require the user to sign in if no token is available.\n   * @returns {Observable<R>} An observable emitting the prepared request.\n   * @template R, T\n   */\n  public prepareAuthRequest$<R extends HttpRequest<T> | Request, T>(req: R, requireSignIn = false): Observable<R> {\n    return this.waitForAccessToken$(requireSignIn).pipe(map((token) => setHttpAuthHeader(req, token)));\n  }\n\n  /**\n   * Waits for a valid access token to become available.\n   *\n   * @param {boolean} requireSignIn - Whether to require the user to sign in if no token is available.\n   * @returns {Observable<string | null>} An observable emitting the access token.\n   */\n  public waitForAccessToken$(requireSignIn: boolean): Observable<string | null> {\n    const accessToken$ = of(this.getAccessToken()).pipe(filter((token) => !!token && this.isAuthenticated()));\n    const waitForAccessToken$ = this.onTokenReceived$.pipe(\n      timeout(this.authConfig.waitForTokenInMs ?? 0),\n      catchError(async () => this.tryRefreshToken().catch(() => null)),\n    );\n\n    return this.isInitialized$.pipe(\n      switchMap(() => merge(accessToken$, waitForAccessToken$)),\n      take(1),\n      map(() => this.getAccessToken()),\n      tap((token) => !token && requireSignIn && this.signIn(this.windowRef.location.pathname)),\n    );\n  }\n\n  private async routeToRequestedUrl(): Promise<void> {\n    if (!this.oauthService.state || !this.oauthService.hasValidAccessToken()) return;\n    await this.router.navigateByUrl(decodeURIComponent(this.oauthService.state));\n  }\n\n  private hasValidOfflineToken(): boolean {\n    const issuedAt = this.getIdentityClaims()?.iat ?? 0;\n\n    return Date.now() - issuedAt * 1000 <= this.authConfig.maxOfflineTime * 1000;\n  }\n\n  private redirectToLogin(uri: string): void {\n    this.windowRef.location.assign(uri);\n    this.isRedirecting$$.next(true);\n  }\n\n  private async tryLoadUserProfile(): Promise<void> {\n    if (!this.authConfig.loadUserProfile || !this.isAuthenticated()) return;\n    await this.oauthService.loadUserProfile().catch(() => null);\n  }\n\n  private async tryRefreshToken(): Promise<void> {\n    if (!this.getRefreshToken()) return;\n    await this.oauthService.refreshToken();\n  }\n\n  private assertAudience(clientId?: string | null): void {\n    const tokenClientId = this.getIdentityClaims()?.aud;\n    if (!clientId || !tokenClientId || clientId === tokenClientId) return;\n    this.signOut(true);\n  }\n\n  private updateSilentRefresh(enabled: boolean) {\n    if (enabled) {\n      this.oauthService.setupAutomaticSilentRefresh();\n    } else {\n      this.oauthService.stopAutomaticRefresh();\n    }\n  }\n}\n"]}
|
|
@@ -1,2 +0,0 @@
|
|
|
1
|
-
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0aC50eXBpbmdzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vbGlicy9hdXRoL3NyYy9saWIvYXV0aC50eXBpbmdzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJkZWNsYXJlIGdsb2JhbCB7XG4gIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tbmFtZXNwYWNlXG4gIG5hbWVzcGFjZSBPZHhBdXRoIHtcbiAgICB0eXBlIFJhd0lkZW50aXR5Q2xhaW1zID0gUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG5cbiAgICBpbnRlcmZhY2UgQXV0aFBsdWdpblJlc3VsdCB7XG4gICAgICBpbnN0aXR1dGlvbk5hbWU/OiBzdHJpbmc7XG4gICAgICBsYW5ndWFnZUNvZGU/OiBzdHJpbmc7XG4gICAgICBwcmVmZXJyZWRMYW5ndWFnZT86IHN0cmluZztcbiAgICB9XG5cbiAgICBpbnRlcmZhY2UgSWRlbnRpdHlDbGFpbXMgZXh0ZW5kcyBBdXRoUGx1Z2luUmVzdWx0IHtcbiAgICAgIHN1Yjogc3RyaW5nO1xuICAgICAgb2t0YWlkOiBzdHJpbmc7XG4gICAgICBpYXQ6IG51bWJlcjtcbiAgICAgIGF1ZDogc3RyaW5nO1xuXG4gICAgICBuYW1lOiBzdHJpbmc7XG4gICAgICBmaXJzdG5hbWU6IHN0cmluZztcbiAgICAgIGxhc3RuYW1lOiBzdHJpbmc7XG4gICAgICBkaXNwbGF5bmFtZTogc3RyaW5nO1xuICAgICAgcHJlZmVycmVkX3VzZXJuYW1lOiBzdHJpbmc7XG4gICAgICBlbWFpbDogc3RyaW5nO1xuICAgICAgaW5pdGlhbHM6IHN0cmluZztcbiAgICAgIG9yZ2FuaXphdGlvbjogc3RyaW5nO1xuXG4gICAgICBsb2NhbGU/OiBzdHJpbmc7XG4gICAgICB6b25laW5mbz86IHN0cmluZztcbiAgICB9XG4gIH1cbn1cblxuZXhwb3J0IHt9O1xuIl19
|
|
@@ -1,40 +0,0 @@
|
|
|
1
|
-
import { __decorate } from "tslib";
|
|
2
|
-
import { ChangeDetectionStrategy, Component, Input, ViewEncapsulation } from '@angular/core';
|
|
3
|
-
import { CoreModule } from '@odx/angular';
|
|
4
|
-
import { ListModule } from '@odx/angular/components/list';
|
|
5
|
-
import { CSSComponent } from '@odx/angular/internal';
|
|
6
|
-
import { TranslatePipe } from '@odx/angular/internal/translate';
|
|
7
|
-
import { injectElement } from '@odx/angular/utils';
|
|
8
|
-
import { SignOutDirective } from '../../directives';
|
|
9
|
-
import * as i0 from "@angular/core";
|
|
10
|
-
import * as i1 from "@angular/common";
|
|
11
|
-
import * as i2 from "@odx/angular/components/list";
|
|
12
|
-
import * as i3 from "@odx/angular/components/icon";
|
|
13
|
-
/**
|
|
14
|
-
* Displays authentication actions.
|
|
15
|
-
*/
|
|
16
|
-
let AuthActionsComponent = class AuthActionsComponent {
|
|
17
|
-
constructor() {
|
|
18
|
-
this.element = injectElement();
|
|
19
|
-
/**
|
|
20
|
-
* The identity claims.
|
|
21
|
-
*
|
|
22
|
-
* @type {OdxAuth.IdentityClaims | null}
|
|
23
|
-
* @default null
|
|
24
|
-
*/
|
|
25
|
-
this.claims = null;
|
|
26
|
-
}
|
|
27
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AuthActionsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
28
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: AuthActionsComponent, isStandalone: true, selector: "odx-auth-actions", inputs: { claims: "claims" }, ngImport: i0, template: "<odx-list>\n @if (claims?.userProfileUrl; as url) {\n <a odxListItem data-testid=\"odx-auth-user-profile-link\" [href]=\"url\" rel=\"noopener\" target=\"_blank\">\n <odx-icon name=\"user-administration\" iconSet=\"core\" odxListItemPrefix />\n {{ 'userProfileLink' | odxTranslate | async }}\n </a>\n }\n <button odxListItem odxAuthSignOut data-testid=\"odx-auth-sign-out-button\" style=\"width: 100%\">\n <odx-icon name=\"logout\" iconSet=\"core\" odxListItemPrefix />\n {{ 'signOutButtonText' | odxTranslate | async }}\n </button>\n</odx-list>\n", dependencies: [{ kind: "ngmodule", type: CoreModule }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "ngmodule", type: ListModule }, { kind: "component", type: i2.ListComponent, selector: "odx-list" }, { kind: "component", type: i2.ListItemComponent, selector: "odx-list-item, [odxListItem]", inputs: ["danger", "variant", "muted", "selected"] }, { kind: "component", type: i3.IconComponent, selector: "odx-icon", inputs: ["inline", "size", "name", "iconSet", "identifier"] }, { kind: "pipe", type: TranslatePipe, name: "odxTranslate" }, { kind: "directive", type: SignOutDirective, selector: "[odxAuthSignOut]", outputs: ["odxAuthSignOut"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
|
|
29
|
-
};
|
|
30
|
-
AuthActionsComponent = __decorate([
|
|
31
|
-
CSSComponent('auth-actions')
|
|
32
|
-
], AuthActionsComponent);
|
|
33
|
-
export { AuthActionsComponent };
|
|
34
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AuthActionsComponent, decorators: [{
|
|
35
|
-
type: Component,
|
|
36
|
-
args: [{ standalone: true, selector: 'odx-auth-actions', imports: [CoreModule, ListModule, TranslatePipe, SignOutDirective], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, template: "<odx-list>\n @if (claims?.userProfileUrl; as url) {\n <a odxListItem data-testid=\"odx-auth-user-profile-link\" [href]=\"url\" rel=\"noopener\" target=\"_blank\">\n <odx-icon name=\"user-administration\" iconSet=\"core\" odxListItemPrefix />\n {{ 'userProfileLink' | odxTranslate | async }}\n </a>\n }\n <button odxListItem odxAuthSignOut data-testid=\"odx-auth-sign-out-button\" style=\"width: 100%\">\n <odx-icon name=\"logout\" iconSet=\"core\" odxListItemPrefix />\n {{ 'signOutButtonText' | odxTranslate | async }}\n </button>\n</odx-list>\n" }]
|
|
37
|
-
}], propDecorators: { claims: [{
|
|
38
|
-
type: Input
|
|
39
|
-
}] } });
|
|
40
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0aC1hY3Rpb25zLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvYXV0aC9zcmMvbGliL2NvbXBvbmVudHMvYXV0aC1hY3Rpb25zL2F1dGgtYWN0aW9ucy5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi8uLi9saWJzL2F1dGgvc3JjL2xpYi9jb21wb25lbnRzL2F1dGgtYWN0aW9ucy9hdXRoLWFjdGlvbnMuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzdGLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxjQUFjLENBQUM7QUFDMUMsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLDhCQUE4QixDQUFDO0FBQzFELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUNyRCxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0saUNBQWlDLENBQUM7QUFDaEUsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLG9CQUFvQixDQUFDO0FBQ25ELE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLGtCQUFrQixDQUFDOzs7OztBQUVwRDs7R0FFRztBQVVJLElBQU0sb0JBQW9CLEdBQTFCLE1BQU0sb0JBQW9CO0lBQTFCO1FBQ1csWUFBTyxHQUFHLGFBQWEsRUFBRSxDQUFDO1FBRTFDOzs7OztXQUtHO1FBRUksV0FBTSxHQUFrQyxJQUFJLENBQUM7S0FDckQ7K0dBWFksb0JBQW9CO21HQUFwQixvQkFBb0IsMEdDcEJqQyw4akJBWUEsMkNER1ksVUFBVSxtRkFBRSxVQUFVLGtYQUFFLGFBQWEscURBQUUsZ0JBQWdCOztBQUt0RCxvQkFBb0I7SUFUaEMsWUFBWSxDQUFDLGNBQWMsQ0FBQztHQVNoQixvQkFBb0IsQ0FXaEM7OzRGQVhZLG9CQUFvQjtrQkFSaEMsU0FBUztpQ0FDSSxJQUFJLFlBQ04sa0JBQWtCLFdBQ25CLENBQUMsVUFBVSxFQUFFLFVBQVUsRUFBRSxhQUFhLEVBQUUsZ0JBQWdCLENBQUMsbUJBRWpELHVCQUF1QixDQUFDLE1BQU0saUJBQ2hDLGlCQUFpQixDQUFDLElBQUk7OEJBWTlCLE1BQU07c0JBRFosS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENoYW5nZURldGVjdGlvblN0cmF0ZWd5LCBDb21wb25lbnQsIElucHV0LCBWaWV3RW5jYXBzdWxhdGlvbiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29yZU1vZHVsZSB9IGZyb20gJ0BvZHgvYW5ndWxhcic7XG5pbXBvcnQgeyBMaXN0TW9kdWxlIH0gZnJvbSAnQG9keC9hbmd1bGFyL2NvbXBvbmVudHMvbGlzdCc7XG5pbXBvcnQgeyBDU1NDb21wb25lbnQgfSBmcm9tICdAb2R4L2FuZ3VsYXIvaW50ZXJuYWwnO1xuaW1wb3J0IHsgVHJhbnNsYXRlUGlwZSB9IGZyb20gJ0BvZHgvYW5ndWxhci9pbnRlcm5hbC90cmFuc2xhdGUnO1xuaW1wb3J0IHsgaW5qZWN0RWxlbWVudCB9IGZyb20gJ0BvZHgvYW5ndWxhci91dGlscyc7XG5pbXBvcnQgeyBTaWduT3V0RGlyZWN0aXZlIH0gZnJvbSAnLi4vLi4vZGlyZWN0aXZlcyc7XG5cbi8qKlxuICogRGlzcGxheXMgYXV0aGVudGljYXRpb24gYWN0aW9ucy5cbiAqL1xuQENTU0NvbXBvbmVudCgnYXV0aC1hY3Rpb25zJylcbkBDb21wb25lbnQoe1xuICBzdGFuZGFsb25lOiB0cnVlLFxuICBzZWxlY3RvcjogJ29keC1hdXRoLWFjdGlvbnMnLFxuICBpbXBvcnRzOiBbQ29yZU1vZHVsZSwgTGlzdE1vZHVsZSwgVHJhbnNsYXRlUGlwZSwgU2lnbk91dERpcmVjdGl2ZV0sXG4gIHRlbXBsYXRlVXJsOiAnLi9hdXRoLWFjdGlvbnMuY29tcG9uZW50Lmh0bWwnLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbiAgZW5jYXBzdWxhdGlvbjogVmlld0VuY2Fwc3VsYXRpb24uTm9uZSxcbn0pXG5leHBvcnQgY2xhc3MgQXV0aEFjdGlvbnNDb21wb25lbnQge1xuICBwdWJsaWMgcmVhZG9ubHkgZWxlbWVudCA9IGluamVjdEVsZW1lbnQoKTtcblxuICAvKipcbiAgICogVGhlIGlkZW50aXR5IGNsYWltcy5cbiAgICpcbiAgICogQHR5cGUge09keEF1dGguSWRlbnRpdHlDbGFpbXMgfCBudWxsfVxuICAgKiBAZGVmYXVsdCBudWxsXG4gICAqL1xuICBASW5wdXQoKVxuICBwdWJsaWMgY2xhaW1zOiBPZHhBdXRoLklkZW50aXR5Q2xhaW1zIHwgbnVsbCA9IG51bGw7XG59XG4iLCI8b2R4LWxpc3Q+XG4gIEBpZiAoY2xhaW1zPy51c2VyUHJvZmlsZVVybDsgYXMgdXJsKSB7XG4gICAgPGEgb2R4TGlzdEl0ZW0gZGF0YS10ZXN0aWQ9XCJvZHgtYXV0aC11c2VyLXByb2ZpbGUtbGlua1wiIFtocmVmXT1cInVybFwiIHJlbD1cIm5vb3BlbmVyXCIgdGFyZ2V0PVwiX2JsYW5rXCI+XG4gICAgICA8b2R4LWljb24gbmFtZT1cInVzZXItYWRtaW5pc3RyYXRpb25cIiBpY29uU2V0PVwiY29yZVwiIG9keExpc3RJdGVtUHJlZml4IC8+XG4gICAgICB7eyAndXNlclByb2ZpbGVMaW5rJyB8IG9keFRyYW5zbGF0ZSB8IGFzeW5jIH19XG4gICAgPC9hPlxuICB9XG4gIDxidXR0b24gb2R4TGlzdEl0ZW0gb2R4QXV0aFNpZ25PdXQgZGF0YS10ZXN0aWQ9XCJvZHgtYXV0aC1zaWduLW91dC1idXR0b25cIiBzdHlsZT1cIndpZHRoOiAxMDAlXCI+XG4gICAgPG9keC1pY29uIG5hbWU9XCJsb2dvdXRcIiBpY29uU2V0PVwiY29yZVwiIG9keExpc3RJdGVtUHJlZml4IC8+XG4gICAge3sgJ3NpZ25PdXRCdXR0b25UZXh0JyB8IG9keFRyYW5zbGF0ZSB8IGFzeW5jIH19XG4gIDwvYnV0dG9uPlxuPC9vZHgtbGlzdD5cbiJdfQ==
|