From cb54013e4922cf3a2c02267d02b1bb2e3bcca3ee Mon Sep 17 00:00:00 2001 From: Tim Deubler Date: Fri, 12 Jul 2024 18:23:27 +0200 Subject: [PATCH] updated expression cache handling Signed-off-by: Tim Deubler --- packages/common/src/Expressions/Expression.ts | 18 ++++-- .../src/Expressions/ExpressionParser.ts | 63 +++++++++++-------- 2 files changed, 51 insertions(+), 30 deletions(-) diff --git a/packages/common/src/Expressions/Expression.ts b/packages/common/src/Expressions/Expression.ts index a0f16d9d9..cb98c3cdd 100644 --- a/packages/common/src/Expressions/Expression.ts +++ b/packages/common/src/Expressions/Expression.ts @@ -25,16 +25,19 @@ export enum ExpressionMode { export interface IExpression { json: any[]; + eval(context: any); } export type JSONExpression = [string, ...any[]]; -let expId =0; +let expId = 0; + export abstract class Expression implements IExpression { static operator: string; id?: number; + static isExpression(exp) { return exp instanceof Expression; } @@ -50,6 +53,7 @@ export abstract class Expression implements IExpression { json: JSONExpression; supportsPartialEval: boolean; + constructor(json: JSONExpression, env: ExpressionParser) { // this.id = expId++; this.json = json; @@ -78,11 +82,17 @@ export abstract class Expression implements IExpression { } toJSON() { - return this.json.map((v)=>Expression.isExpression(v) ? v.toJSON(): v); + return this.json.map((v) => Expression.isExpression(v) ? v.toJSON() : v); } - resolve(context=this.env.context, mode?: ExpressionMode) { + resolve(context?, mode?: ExpressionMode) { const {env} = this; - return env.evaluate(this, context, mode); + let cache; + if (context === undefined) { + // resolve dynamic expression + context = this.env.context; + cache = env.dynamicResultCache; + } + return env.evaluate(this, context, mode, cache); } } diff --git a/packages/common/src/Expressions/ExpressionParser.ts b/packages/common/src/Expressions/ExpressionParser.ts index ad57ecca2..be6b0c495 100644 --- a/packages/common/src/Expressions/ExpressionParser.ts +++ b/packages/common/src/Expressions/ExpressionParser.ts @@ -34,6 +34,12 @@ class DynamicExpressionInterrupt extends Error { } } +type Context = { [name: string]: any }; +type Def = JSONExpression | Expression | boolean | number | string | null; +type Value = {value:Def}; +type Definitions = { [name: string]: Def | Value }; + + export class ExpressionParser { static DYNAMIC_EXPRESSION_INTERRUPT: DynamicExpressionInterrupt = new DynamicExpressionInterrupt(); static Mode = ExpressionMode; @@ -41,14 +47,14 @@ export class ExpressionParser { [op: string]: new (e: JSONExpression, p: ExpressionParser) => Expression & { [K in keyof typeof Expression]: typeof Expression[K] } }; - private definitions: {}; + private definitions: Definitions; private cache = new Map(); - context: { [name: string]: any }; + context: Context; private _cacheHits: number = 0; private defaultResultCache: ResultCache = new Map(); private resultCache: ResultCache; private _mode: ExpressionMode; - private dynamicResultCache: ResultCache = new Map(); + dynamicResultCache: ResultCache = new Map(); static { let expressions = {}; @@ -58,26 +64,29 @@ export class ExpressionParser { this.Expressions = expressions; } - constructor(definitions = {}, context = {}) { + constructor(definitions: Definitions = {}, context: Context = {}) { this.definitions = definitions; this.context = context; this.setMode(ExpressionMode.static); + + this.defaultResultCache.hits = 0; + this.dynamicResultCache.hits = 0; + // console.time('clone definitions'); // this.definitions = JSUtils.clone(definitions); // console.timeEnd('clone definitions'); - // this.cache.get = this.cache.set =()=>undefined; // this.defaultResultCache.get = this.defaultResultCache.set =()=>undefined; // this.dynamicResultCache.get = this.dynamicResultCache.set =()=>undefined; } - init(def, mapContext) { + init(def: Definitions, mapContext: Context) { this.clearCache(); this.setDefinitions(def); this.context = mapContext; } - setDefinitions(def) { + setDefinitions(def: Definitions) { this.definitions = def; } @@ -86,12 +95,12 @@ export class ExpressionParser { this.cache.clear(); } - evaluate(exp, context, mode: ExpressionMode = ExpressionMode.static) { + evaluate(exp: Expression | JSONExpression, context: Context, mode: ExpressionMode = ExpressionMode.static, cache?) { let result; exp = this.parseJSON(exp); try { - this.setMode(mode); - result = this.evaluateParsed(exp, context); + this.setMode(mode, cache); + result = this.evaluateParsed(exp as Expression, context); } catch (e) { if (e.message === 'DynamicExpressionInterrupt') { return e.exp; @@ -102,17 +111,18 @@ export class ExpressionParser { return result; } - evaluateParsed(exp: Expression, context) { + evaluateParsed(exp: Expression, context: Context) { if (exp instanceof Expression) { if (this.getMode() == ExpressionParser.Mode.dynamic && exp.dynamic() && !exp.supportsPartialEval) { const DYNAMIC_EXPRESSION_INTERRUPT = ExpressionParser.DYNAMIC_EXPRESSION_INTERRUPT; DYNAMIC_EXPRESSION_INTERRUPT.exp = exp; throw DYNAMIC_EXPRESSION_INTERRUPT; + // return exp; } // return exp.eval(context); let result = this.resultCache.get(exp); if (result !== undefined) { - this.resultCache.hits = (this.resultCache.hits || 0) + 1; + this.resultCache.hits++; return result; } result = exp.eval(context); @@ -124,17 +134,17 @@ export class ExpressionParser { } - resolveReference(exp: JSONExpression, definitions = this.definitions) { + resolveReference(exp: JSONExpression, definitions: Definitions = this.definitions) { let key = exp?.[1]; let value = definitions[key]; if (value != null) { - if (value.value != null) { - value = value.value; + if ((value as Value).value != null) { + value = (value as Value).value; } while (ExpressionParser.isJSONExp(value) && value[0] == 'ref') { value = definitions[value[1]]; - if (value.value !== undefined) { - value = value.value; + if ((value as Value).value !== undefined) { + value = (value as Value).value; } } } @@ -183,7 +193,7 @@ export class ExpressionParser { return Expression && new Expression(jsonExp, this); } - static isJSONExp(exp) { + static isJSONExp(exp: any) { return Array.isArray(exp) && typeof exp[0] == 'string'; } @@ -195,19 +205,20 @@ export class ExpressionParser { this.dynamicResultCache.clear(); } - isSupported(exp: JSONExpression) { + isSupported(exp: JSONExpression): boolean { return Boolean(ExpressionParser.Expressions[exp[0]]); } - setMode(mode: ExpressionMode) { - if (mode != this._mode) { - this._mode = mode; - this.context.mode = mode; - this.resultCache = mode === ExpressionMode.static ? this.defaultResultCache : this.dynamicResultCache; - } + setMode(mode: ExpressionMode, cache?: ResultCache) { + // if (mode != this._mode) { + this._mode = mode; + this.context.mode = mode; + this.resultCache = cache || this.defaultResultCache; + // this.resultCache = cache || (mode === ExpressionMode.static ? this.defaultResultCache : this.dynamicResultCache); + // } } - getMode() { + getMode(): ExpressionMode { return this._mode; } }