mirror of
https://github.com/Alfresco/alfresco-ng2-components.git
synced 2025-05-12 17:04:57 +00:00
* NG-16 - migration step 1 * NG16 - fixed the builds * [MIGRATION] - Switching off no empty functions rule * [MIGRATION] - Somehow the fix for the cli types got lost * [MIGRATION] - Something happened on the way to heaven * [MIGRATION] - Ok this is weird * [MIGRATION] - Ok this is really weird * [MIGRATION] - change selector for expanded expansion panel content * [MIGRATION] - storybook chips related e2e fails fix, unit alignments * [MIGRATION] - Ok this is really weird * [MIGRATION] - Updating outdated packages * [MIGRATION] - updated deps for process cloud * [MIGRATION] - Rebased to latest * [MIGRATION] - Fixed lint * [MIGRATION] - Fixed package version * Rebased with lastest changes * [MIGRATION] - check to update the script * [MIGRATION] - rebased after the move of AlfrescoApiService * [MIGRATION] - fixed schematics build * [MIGRATION] - the lesson is : do not change what is already working * Rebased to latestp * Fixes after CR (#10202) * Fixed white background issue in sidenav after ng16 upgrade (#10207) * [MIGRATION] fix header background color not being read from input prop * Fixed build * Rebased * Fix package version after release * Thanks apollo deps * custom theme missing sidenav color * fix filter subscript overlapping dropdown items * [ACS-8749] View More tags button no longer shows when all tags are removed from a node (#10285) * [ACS-8741] Sidenav text for Process and Task filters is now grey (#10284) * AAE-26037 Moved permissions role selector styles to a dedicated file (#10297) * Removed fallback and starting using simple token * Removed fallback and starting using simple token * Fixed related unit test * Fixed related unit test * Writing should use the same calss as reading * Fixed unit tests for insights * Fixed unit tests for insights * Added fallback for JWT token * Added fallback for JWT token * Auto import fix * Added fallback for injection * Added fallback for injection * Trying to fix JWT token issue * Fixed unit tests * Fixed unit tests * Fixed style for collaborators * fixed permission row height * AAE-26163 Fix infinite loop when authentication error event occured (#10272) * AAE-26163 Logout user after 3 login attempts failed, avoiding infinite loop when an authentication error occured, like when a user machine clock is significantly out of sync * AAE-26163 Wait to discovery document to be loaded and user not authenticated to perform a ssoLogin, logout user if login fails after 3 attempts * AAE-26163 Fix missed id_token_hint invoking logout when a login error occured due to a clock significantly out of sync * AAE-26163 Add fake observable to unit test * AAE-26163 Show oauth event logs if showDebugInformation is enabled, remove auth items if access token is not valid * AAE-26163 Improve tryLogin error message * AAE-26163 Check if token has expired to fix case when user access the application after the token is expired and with a clock significantly out of sync * AAE-26163 Test logout when clock is out of sync * AAE-26163 Create a service to check if local machine time is out of sync * AAE-26163 Update oauthErrorEvent$ and combinedOAuthErrorsStream$ to return errors * AAE-26163 Output error within combined oauth error event subscription * AAE-26163 Fix lint problems * AAE-26163 Logout user when token refresh error happens for the second time, if the token is not refreshed properly after first refresh error * AAE-26163 Logout user once an oauth error event occur due to clock out of sync * AAE-26163 Fix retry login error message if the OAuthErrorEvent doesn t return reason * AAE-26163 Fix the issue where the logout API call is canceled by the authorize call when login fails due to clock synchronization problems, causing an infinite loop. * remove console.log * AAE-26163 Fix retry login error message if the OAuthErrorEvent reason is an empty object * Updating dependencies * Fixed subject complete to avoid calling a reset event when complete --------- Co-authored-by: Wojciech Duda <69160975+wojd0@users.noreply.github.com> Co-authored-by: dominikiwanekhyland <141320833+dominikiwanekhyland@users.noreply.github.com> Co-authored-by: swapnil-verma-gl <92505353+swapnil-verma-gl@users.noreply.github.com> Co-authored-by: Ehsan Rezaei <ehsan.rezaei@hyland.com> Co-authored-by: Amedeo Lepore <amedeo.lepore@hyland.com>
292 lines
9.0 KiB
TypeScript
292 lines
9.0 KiB
TypeScript
/*!
|
|
* @license
|
|
* Copyright © 2005-2024 Hyland Software, Inc. and its affiliates. All rights reserved.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with 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, Type, InjectionToken, Inject } from '@angular/core';
|
|
import { RuleEvaluator, RuleRef, RuleContext } from '../config/rule.extensions';
|
|
import { ExtensionConfig } from '../config/extension.config';
|
|
import { ExtensionLoaderService } from './extension-loader.service';
|
|
import { RouteRef } from '../config/routing.extensions';
|
|
import { ActionRef } from '../config/action.extensions';
|
|
import * as core from '../evaluators/core.evaluators';
|
|
import { ComponentRegisterService } from './component-register.service';
|
|
import { RuleService } from './rule.service';
|
|
import { ExtensionElement } from '../config/extension-element';
|
|
import { BehaviorSubject, Observable } from 'rxjs';
|
|
|
|
/**
|
|
* The default extensions factory
|
|
*
|
|
* @returns the list of extension json files
|
|
*/
|
|
export function extensionJsonsFactory() {
|
|
return [];
|
|
}
|
|
|
|
export const EXTENSION_JSONS = new InjectionToken<string[][]>('extension-jsons', {
|
|
providedIn: 'root',
|
|
factory: extensionJsonsFactory
|
|
});
|
|
|
|
export const EXTENSION_JSON_VALUES = new InjectionToken<string[][]>('extension-jsons-values', {
|
|
providedIn: 'root',
|
|
factory: extensionJsonsFactory
|
|
});
|
|
|
|
/**
|
|
* Provides the extension json values for the angular modules
|
|
*
|
|
* @param jsons files to provide
|
|
* @returns a provider section
|
|
*/
|
|
export function provideExtensionConfig(jsons: string[]) {
|
|
return {
|
|
provide: EXTENSION_JSONS,
|
|
useValue: jsons,
|
|
multi: true
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Provides the extension json raw values for the angular modules
|
|
*
|
|
* @param extensionConfigValue config value
|
|
* @returns a provider section
|
|
*/
|
|
export function provideExtensionConfigValues(extensionConfigValue: ExtensionConfig[]) {
|
|
return {
|
|
provide: EXTENSION_JSON_VALUES,
|
|
useValue: extensionConfigValue,
|
|
multi: true
|
|
};
|
|
}
|
|
|
|
@Injectable({
|
|
providedIn: 'root'
|
|
})
|
|
export class ExtensionService {
|
|
configPath = 'assets/app.extensions.json';
|
|
pluginsPath = 'assets/plugins';
|
|
|
|
routes: Array<RouteRef> = [];
|
|
actions: Array<ActionRef> = [];
|
|
features: Array<any> = [];
|
|
authGuards: Record<string, unknown> = {};
|
|
|
|
setup$: Observable<ExtensionConfig>;
|
|
|
|
protected config: ExtensionConfig = null;
|
|
protected onSetup$ = new BehaviorSubject<ExtensionConfig>(this.config);
|
|
|
|
constructor(
|
|
protected loader: ExtensionLoaderService,
|
|
protected componentRegister: ComponentRegisterService,
|
|
protected ruleService: RuleService,
|
|
@Inject(EXTENSION_JSONS) protected extensionJsons: string[],
|
|
@Inject(EXTENSION_JSON_VALUES) protected extensionJsonValues: ExtensionConfig[]
|
|
) {
|
|
this.setup$ = this.onSetup$.asObservable();
|
|
}
|
|
|
|
/**
|
|
* Loads and registers an extension config file and plugins (specified by path properties).
|
|
*
|
|
* @returns The loaded config data
|
|
*/
|
|
async load(): Promise<ExtensionConfig> {
|
|
const config = await this.loader.load(this.configPath, this.pluginsPath, this.extensionJsons.flat(), this.extensionJsonValues.flat());
|
|
|
|
this.setup(config);
|
|
return config;
|
|
}
|
|
|
|
/**
|
|
* Registers extensions from a config object.
|
|
*
|
|
* @param config Object with config data
|
|
*/
|
|
setup(config: ExtensionConfig) {
|
|
if (!config) {
|
|
console.warn('Extension configuration not found');
|
|
return;
|
|
}
|
|
|
|
this.config = config;
|
|
|
|
this.setEvaluators({
|
|
'core.every': core.every,
|
|
'core.some': core.some,
|
|
'core.not': core.not
|
|
});
|
|
|
|
this.actions = this.loader.getActions(config);
|
|
this.routes = this.loader.getRoutes(config);
|
|
this.features = this.loader.getFeatures(config);
|
|
|
|
this.ruleService.setup(config);
|
|
this.onSetup$.next(config);
|
|
}
|
|
|
|
/**
|
|
* Gets features by key.
|
|
*
|
|
* @param key Key string using dot notation or array of strings
|
|
* @param defaultValue Default value returned if feature is not found, default is empty array
|
|
* @returns Feature found by key
|
|
*/
|
|
getFeature<T = any[]>(key: string | string[], defaultValue: any = []): T {
|
|
const properties: string[] = Array.isArray(key) ? key : key.split('.');
|
|
return properties.reduce((prev, curr) => prev?.[curr], this.features) || defaultValue;
|
|
}
|
|
|
|
getElements<T extends ExtensionElement>(key: string, fallback: Array<T> = []): Array<T> {
|
|
return this.loader.getElements(this.config, key, fallback);
|
|
}
|
|
|
|
/**
|
|
* Adds one or more new rule evaluators to the existing set.
|
|
*
|
|
* @param values The new evaluators to add
|
|
*/
|
|
setEvaluators(values: { [key: string]: RuleEvaluator }) {
|
|
this.ruleService.setEvaluators(values);
|
|
}
|
|
|
|
/**
|
|
* Adds one or more new auth guards to the existing set.
|
|
*
|
|
* @param values The new auth guards to add
|
|
*/
|
|
setAuthGuards(values: Record<string, unknown>) {
|
|
if (values) {
|
|
this.authGuards = Object.assign({}, this.authGuards, values);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds one or more new components to the existing set.
|
|
*
|
|
* @param values The new components to add
|
|
*/
|
|
setComponents(values: { [key: string]: Type<any> }) {
|
|
this.componentRegister.setComponents(values);
|
|
}
|
|
|
|
/**
|
|
* Retrieves a route using its ID value.
|
|
*
|
|
* @param id The ID value to look for
|
|
* @returns The route or null if not found
|
|
*/
|
|
getRouteById(id: string): RouteRef {
|
|
return this.routes.find((route) => route.id === id);
|
|
}
|
|
|
|
/**
|
|
* Retrieves one or more auth guards using an array of ID values.
|
|
*
|
|
* @param ids Array of ID value to look for
|
|
* @returns Array of auth guards or empty array if none were found
|
|
*/
|
|
getAuthGuards(ids: string[]): Array<unknown> {
|
|
return (ids || []).map((id) => this.authGuards[id]).filter((guard) => guard);
|
|
}
|
|
|
|
/**
|
|
* Retrieves an action using its ID value.
|
|
*
|
|
* @param id The ID value to look for
|
|
* @returns Action or null if not found
|
|
*/
|
|
getActionById(id: string): ActionRef {
|
|
return this.actions.find((action) => action.id === id);
|
|
}
|
|
|
|
/**
|
|
* Retrieves a RuleEvaluator function using its key name.
|
|
*
|
|
* @param key Key name to look for
|
|
* @returns RuleEvaluator or null if not found
|
|
*/
|
|
getEvaluator(key: string): RuleEvaluator {
|
|
return this.ruleService.getEvaluator(key);
|
|
}
|
|
|
|
/**
|
|
* Evaluates a rule.
|
|
*
|
|
* @param ruleId ID of the rule to evaluate
|
|
* @param context Custom rule execution context.
|
|
* @returns True if the rule passed, false otherwise
|
|
*/
|
|
evaluateRule(ruleId: string, context?: RuleContext): boolean {
|
|
return this.ruleService.evaluateRule(ruleId, context);
|
|
}
|
|
|
|
/**
|
|
* Retrieves a registered extension component using its ID value.
|
|
*
|
|
* @param id The ID value to look for
|
|
* @returns The component or null if not found
|
|
*/
|
|
getComponentById<T>(id: string) {
|
|
return this.componentRegister.getComponentById<T>(id);
|
|
}
|
|
|
|
/**
|
|
* Retrieves a rule using its ID value.
|
|
*
|
|
* @param id The ID value to look for
|
|
* @returns The rule or null if not found
|
|
*/
|
|
getRuleById(id: string): RuleRef {
|
|
return this.ruleService.getRuleById(id);
|
|
}
|
|
|
|
/**
|
|
* Runs a lightweight expression stored in a string.
|
|
*
|
|
* @param value String containing the expression or literal value
|
|
* @param context Parameter object for the expression with details of app state
|
|
* @returns Result of evaluated expression, if found, or the literal value otherwise
|
|
*/
|
|
runExpression(value: string | any, context?: any) {
|
|
if (typeof value === 'string') {
|
|
return this.evaluateExpression(value, context);
|
|
} else {
|
|
const duplicate = Object.assign({}, value);
|
|
Object.keys(duplicate).forEach((key) => {
|
|
duplicate[key] = this.evaluateExpression(duplicate[key], context);
|
|
});
|
|
return duplicate;
|
|
}
|
|
}
|
|
|
|
private evaluateExpression(value: string, context?: any): string {
|
|
const pattern = new RegExp(/\$\((.*\)?)\)/g);
|
|
const matches = pattern.exec(value);
|
|
|
|
if (matches && matches.length > 1) {
|
|
const expression = matches[1];
|
|
const fn = new Function('context', `return ${expression}`);
|
|
const result = fn(context);
|
|
|
|
return result;
|
|
}
|
|
return value;
|
|
}
|
|
}
|