@the-trybe/formula-engine 1.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/.claude/settings.local.json +6 -0
- package/PRD_FORMULA_ENGINE.md +1863 -0
- package/README.md +382 -0
- package/dist/decimal-utils.d.ts +180 -0
- package/dist/decimal-utils.js +355 -0
- package/dist/dependency-extractor.d.ts +20 -0
- package/dist/dependency-extractor.js +103 -0
- package/dist/dependency-graph.d.ts +60 -0
- package/dist/dependency-graph.js +252 -0
- package/dist/errors.d.ts +161 -0
- package/dist/errors.js +260 -0
- package/dist/evaluator.d.ts +51 -0
- package/dist/evaluator.js +494 -0
- package/dist/formula-engine.d.ts +79 -0
- package/dist/formula-engine.js +355 -0
- package/dist/functions.d.ts +3 -0
- package/dist/functions.js +720 -0
- package/dist/index.d.ts +10 -0
- package/dist/index.js +61 -0
- package/dist/lexer.d.ts +25 -0
- package/dist/lexer.js +357 -0
- package/dist/parser.d.ts +32 -0
- package/dist/parser.js +372 -0
- package/dist/types.d.ts +228 -0
- package/dist/types.js +62 -0
- package/jest.config.js +23 -0
- package/package.json +35 -0
- package/src/decimal-utils.ts +408 -0
- package/src/dependency-extractor.ts +117 -0
- package/src/dependency-graph.test.ts +238 -0
- package/src/dependency-graph.ts +288 -0
- package/src/errors.ts +296 -0
- package/src/evaluator.ts +604 -0
- package/src/formula-engine.test.ts +660 -0
- package/src/formula-engine.ts +430 -0
- package/src/functions.ts +770 -0
- package/src/index.ts +103 -0
- package/src/lexer.test.ts +288 -0
- package/src/lexer.ts +394 -0
- package/src/parser.test.ts +349 -0
- package/src/parser.ts +449 -0
- package/src/types.ts +347 -0
- package/tsconfig.json +29 -0
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
import { FormulaEngineConfig, FormulaDefinition, EvaluationContext, EvaluationResult, EvaluationResultSet, ValidationResult, FunctionDefinition, ASTNode, CacheStats, DependencyGraph as IDependencyGraph } from './types';
|
|
2
|
+
import { DecimalUtils, Decimal } from './decimal-utils';
|
|
3
|
+
export declare class FormulaEngine {
|
|
4
|
+
private config;
|
|
5
|
+
private parser;
|
|
6
|
+
private evaluator;
|
|
7
|
+
private dependencyExtractor;
|
|
8
|
+
private graphBuilder;
|
|
9
|
+
private decimalUtils;
|
|
10
|
+
private functions;
|
|
11
|
+
private astCache;
|
|
12
|
+
private dependencyCache;
|
|
13
|
+
private cacheHits;
|
|
14
|
+
private cacheMisses;
|
|
15
|
+
constructor(config?: FormulaEngineConfig);
|
|
16
|
+
/**
|
|
17
|
+
* Parse an expression into an AST
|
|
18
|
+
*/
|
|
19
|
+
parse(expression: string): ASTNode;
|
|
20
|
+
/**
|
|
21
|
+
* Extract variable dependencies from an expression
|
|
22
|
+
*/
|
|
23
|
+
extractDependencies(expression: string): Set<string>;
|
|
24
|
+
/**
|
|
25
|
+
* Build a dependency graph from formula definitions
|
|
26
|
+
*/
|
|
27
|
+
buildDependencyGraph(formulas: FormulaDefinition[]): IDependencyGraph;
|
|
28
|
+
/**
|
|
29
|
+
* Get the evaluation order for formulas
|
|
30
|
+
*/
|
|
31
|
+
getEvaluationOrder(formulas: FormulaDefinition[]): string[];
|
|
32
|
+
/**
|
|
33
|
+
* Validate formulas without evaluating
|
|
34
|
+
*/
|
|
35
|
+
validate(formulas: FormulaDefinition[]): ValidationResult;
|
|
36
|
+
/**
|
|
37
|
+
* Evaluate a single expression
|
|
38
|
+
*/
|
|
39
|
+
evaluate(expression: string, context: EvaluationContext): EvaluationResult;
|
|
40
|
+
/**
|
|
41
|
+
* Evaluate all formulas in dependency order
|
|
42
|
+
*/
|
|
43
|
+
evaluateAll(formulas: FormulaDefinition[], context: EvaluationContext): EvaluationResultSet;
|
|
44
|
+
/**
|
|
45
|
+
* Register a custom function
|
|
46
|
+
*/
|
|
47
|
+
registerFunction(definition: FunctionDefinition): void;
|
|
48
|
+
/**
|
|
49
|
+
* Register multiple custom functions
|
|
50
|
+
*/
|
|
51
|
+
registerFunctions(definitions: FunctionDefinition[]): void;
|
|
52
|
+
/**
|
|
53
|
+
* Get registered function names
|
|
54
|
+
*/
|
|
55
|
+
getRegisteredFunctions(): string[];
|
|
56
|
+
/**
|
|
57
|
+
* Clear the AST cache
|
|
58
|
+
*/
|
|
59
|
+
clearCache(): void;
|
|
60
|
+
/**
|
|
61
|
+
* Get cache statistics
|
|
62
|
+
*/
|
|
63
|
+
getCacheStats(): CacheStats;
|
|
64
|
+
/**
|
|
65
|
+
* Get the decimal utilities instance
|
|
66
|
+
*/
|
|
67
|
+
getDecimalUtils(): DecimalUtils;
|
|
68
|
+
/**
|
|
69
|
+
* Create a Decimal from a value
|
|
70
|
+
*/
|
|
71
|
+
createDecimal(value: string | number): Decimal;
|
|
72
|
+
private checkExpressionLength;
|
|
73
|
+
private normalizeContext;
|
|
74
|
+
private convertValue;
|
|
75
|
+
private isDecimal;
|
|
76
|
+
private applyRounding;
|
|
77
|
+
private handleError;
|
|
78
|
+
private maybeEvictCache;
|
|
79
|
+
}
|
|
@@ -0,0 +1,355 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.FormulaEngine = void 0;
|
|
4
|
+
const parser_1 = require("./parser");
|
|
5
|
+
const evaluator_1 = require("./evaluator");
|
|
6
|
+
const dependency_extractor_1 = require("./dependency-extractor");
|
|
7
|
+
const dependency_graph_1 = require("./dependency-graph");
|
|
8
|
+
const decimal_utils_1 = require("./decimal-utils");
|
|
9
|
+
const functions_1 = require("./functions");
|
|
10
|
+
const errors_1 = require("./errors");
|
|
11
|
+
class FormulaEngine {
|
|
12
|
+
constructor(config) {
|
|
13
|
+
// Caches
|
|
14
|
+
this.astCache = new Map();
|
|
15
|
+
this.dependencyCache = new Map();
|
|
16
|
+
this.cacheHits = 0;
|
|
17
|
+
this.cacheMisses = 0;
|
|
18
|
+
this.config = {
|
|
19
|
+
enableCache: true,
|
|
20
|
+
maxCacheSize: 1000,
|
|
21
|
+
strictMode: true,
|
|
22
|
+
...config,
|
|
23
|
+
};
|
|
24
|
+
this.decimalUtils = new decimal_utils_1.DecimalUtils(this.config.decimal);
|
|
25
|
+
this.functions = (0, functions_1.createBuiltInFunctions)(this.decimalUtils);
|
|
26
|
+
this.parser = new parser_1.Parser();
|
|
27
|
+
this.dependencyExtractor = new dependency_extractor_1.DependencyExtractor();
|
|
28
|
+
this.graphBuilder = new dependency_graph_1.DependencyGraphBuilder();
|
|
29
|
+
this.evaluator = new evaluator_1.Evaluator(this.decimalUtils, this.functions, this.config);
|
|
30
|
+
// Register any custom functions from config
|
|
31
|
+
if (this.config.functions) {
|
|
32
|
+
for (const fn of this.config.functions) {
|
|
33
|
+
this.registerFunction(fn);
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* Parse an expression into an AST
|
|
39
|
+
*/
|
|
40
|
+
parse(expression) {
|
|
41
|
+
this.checkExpressionLength(expression);
|
|
42
|
+
if (this.config.enableCache) {
|
|
43
|
+
const cached = this.astCache.get(expression);
|
|
44
|
+
if (cached) {
|
|
45
|
+
this.cacheHits++;
|
|
46
|
+
return cached;
|
|
47
|
+
}
|
|
48
|
+
this.cacheMisses++;
|
|
49
|
+
}
|
|
50
|
+
const ast = this.parser.parse(expression);
|
|
51
|
+
if (this.config.enableCache) {
|
|
52
|
+
this.maybeEvictCache();
|
|
53
|
+
this.astCache.set(expression, ast);
|
|
54
|
+
}
|
|
55
|
+
return ast;
|
|
56
|
+
}
|
|
57
|
+
/**
|
|
58
|
+
* Extract variable dependencies from an expression
|
|
59
|
+
*/
|
|
60
|
+
extractDependencies(expression) {
|
|
61
|
+
if (this.config.enableCache) {
|
|
62
|
+
const cached = this.dependencyCache.get(expression);
|
|
63
|
+
if (cached) {
|
|
64
|
+
return new Set(cached);
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
const deps = this.dependencyExtractor.extract(expression);
|
|
68
|
+
if (this.config.enableCache) {
|
|
69
|
+
this.dependencyCache.set(expression, new Set(deps));
|
|
70
|
+
}
|
|
71
|
+
return deps;
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* Build a dependency graph from formula definitions
|
|
75
|
+
*/
|
|
76
|
+
buildDependencyGraph(formulas) {
|
|
77
|
+
return this.graphBuilder.build(formulas);
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* Get the evaluation order for formulas
|
|
81
|
+
*/
|
|
82
|
+
getEvaluationOrder(formulas) {
|
|
83
|
+
return this.graphBuilder.getEvaluationOrder(formulas);
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Validate formulas without evaluating
|
|
87
|
+
*/
|
|
88
|
+
validate(formulas) {
|
|
89
|
+
const errors = [];
|
|
90
|
+
const warnings = [];
|
|
91
|
+
// Check for duplicate IDs
|
|
92
|
+
const ids = new Set();
|
|
93
|
+
for (const formula of formulas) {
|
|
94
|
+
if (ids.has(formula.id)) {
|
|
95
|
+
errors.push(new errors_1.DuplicateFormulaError(formula.id));
|
|
96
|
+
}
|
|
97
|
+
ids.add(formula.id);
|
|
98
|
+
}
|
|
99
|
+
// Try to parse all expressions
|
|
100
|
+
for (const formula of formulas) {
|
|
101
|
+
try {
|
|
102
|
+
this.parse(formula.expression);
|
|
103
|
+
}
|
|
104
|
+
catch (error) {
|
|
105
|
+
if (error instanceof errors_1.FormulaEngineError) {
|
|
106
|
+
errors.push(error);
|
|
107
|
+
}
|
|
108
|
+
else {
|
|
109
|
+
errors.push(new errors_1.GeneralFormulaError(String(error)));
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
// Build dependency graph and check for cycles
|
|
114
|
+
let dependencyGraph = new dependency_graph_1.DependencyGraph();
|
|
115
|
+
let evaluationOrder = [];
|
|
116
|
+
try {
|
|
117
|
+
dependencyGraph = this.buildDependencyGraph(formulas);
|
|
118
|
+
evaluationOrder = dependencyGraph.topologicalSort();
|
|
119
|
+
}
|
|
120
|
+
catch (error) {
|
|
121
|
+
if (error instanceof errors_1.FormulaEngineError) {
|
|
122
|
+
errors.push(error);
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
return {
|
|
126
|
+
valid: errors.length === 0,
|
|
127
|
+
errors,
|
|
128
|
+
warnings,
|
|
129
|
+
dependencyGraph,
|
|
130
|
+
evaluationOrder,
|
|
131
|
+
};
|
|
132
|
+
}
|
|
133
|
+
/**
|
|
134
|
+
* Evaluate a single expression
|
|
135
|
+
*/
|
|
136
|
+
evaluate(expression, context) {
|
|
137
|
+
this.checkExpressionLength(expression);
|
|
138
|
+
const normalizedContext = this.normalizeContext(context);
|
|
139
|
+
return this.evaluator.evaluate(expression, normalizedContext);
|
|
140
|
+
}
|
|
141
|
+
/**
|
|
142
|
+
* Evaluate all formulas in dependency order
|
|
143
|
+
*/
|
|
144
|
+
evaluateAll(formulas, context) {
|
|
145
|
+
const startTime = Date.now();
|
|
146
|
+
const results = new Map();
|
|
147
|
+
const errors = [];
|
|
148
|
+
// Get evaluation order
|
|
149
|
+
let evaluationOrder;
|
|
150
|
+
try {
|
|
151
|
+
evaluationOrder = this.getEvaluationOrder(formulas);
|
|
152
|
+
}
|
|
153
|
+
catch (error) {
|
|
154
|
+
return {
|
|
155
|
+
results,
|
|
156
|
+
success: false,
|
|
157
|
+
errors: [error],
|
|
158
|
+
totalExecutionTimeMs: Date.now() - startTime,
|
|
159
|
+
evaluationOrder: [],
|
|
160
|
+
};
|
|
161
|
+
}
|
|
162
|
+
// Create a map of formulas by ID
|
|
163
|
+
const formulaMap = new Map();
|
|
164
|
+
for (const formula of formulas) {
|
|
165
|
+
formulaMap.set(formula.id, formula);
|
|
166
|
+
}
|
|
167
|
+
// Evaluate in order, merging results into context
|
|
168
|
+
const workingContext = this.normalizeContext(context);
|
|
169
|
+
for (const formulaId of evaluationOrder) {
|
|
170
|
+
const formula = formulaMap.get(formulaId);
|
|
171
|
+
if (!formula)
|
|
172
|
+
continue;
|
|
173
|
+
try {
|
|
174
|
+
const result = this.evaluator.evaluate(formula.expression, workingContext);
|
|
175
|
+
// Apply rounding if configured
|
|
176
|
+
let value = result.value;
|
|
177
|
+
if (formula.rounding && this.isDecimal(value)) {
|
|
178
|
+
value = this.applyRounding(value, formula.rounding);
|
|
179
|
+
}
|
|
180
|
+
// Handle errors based on formula config
|
|
181
|
+
if (!result.success && formula.onError) {
|
|
182
|
+
value = this.handleError(formula, result.error);
|
|
183
|
+
}
|
|
184
|
+
results.set(formulaId, {
|
|
185
|
+
...result,
|
|
186
|
+
value,
|
|
187
|
+
});
|
|
188
|
+
// Merge result into context for subsequent formulas
|
|
189
|
+
workingContext.variables[formulaId] = value;
|
|
190
|
+
if (!result.success) {
|
|
191
|
+
errors.push(result.error);
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
catch (error) {
|
|
195
|
+
const evalResult = {
|
|
196
|
+
value: null,
|
|
197
|
+
success: false,
|
|
198
|
+
error: error,
|
|
199
|
+
executionTimeMs: 0,
|
|
200
|
+
accessedVariables: new Set(),
|
|
201
|
+
};
|
|
202
|
+
// Handle error based on formula config
|
|
203
|
+
if (formula.onError) {
|
|
204
|
+
evalResult.value = this.handleError(formula, error);
|
|
205
|
+
}
|
|
206
|
+
results.set(formulaId, evalResult);
|
|
207
|
+
errors.push(error);
|
|
208
|
+
// Still add to context (as null or default value) so dependent formulas can proceed
|
|
209
|
+
workingContext.variables[formulaId] = evalResult.value;
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
return {
|
|
213
|
+
results,
|
|
214
|
+
success: errors.length === 0,
|
|
215
|
+
errors,
|
|
216
|
+
totalExecutionTimeMs: Date.now() - startTime,
|
|
217
|
+
evaluationOrder,
|
|
218
|
+
};
|
|
219
|
+
}
|
|
220
|
+
/**
|
|
221
|
+
* Register a custom function
|
|
222
|
+
*/
|
|
223
|
+
registerFunction(definition) {
|
|
224
|
+
const name = definition.name.toUpperCase();
|
|
225
|
+
this.functions.set(name, definition);
|
|
226
|
+
}
|
|
227
|
+
/**
|
|
228
|
+
* Register multiple custom functions
|
|
229
|
+
*/
|
|
230
|
+
registerFunctions(definitions) {
|
|
231
|
+
for (const definition of definitions) {
|
|
232
|
+
this.registerFunction(definition);
|
|
233
|
+
}
|
|
234
|
+
}
|
|
235
|
+
/**
|
|
236
|
+
* Get registered function names
|
|
237
|
+
*/
|
|
238
|
+
getRegisteredFunctions() {
|
|
239
|
+
return Array.from(this.functions.keys());
|
|
240
|
+
}
|
|
241
|
+
/**
|
|
242
|
+
* Clear the AST cache
|
|
243
|
+
*/
|
|
244
|
+
clearCache() {
|
|
245
|
+
this.astCache.clear();
|
|
246
|
+
this.dependencyCache.clear();
|
|
247
|
+
this.cacheHits = 0;
|
|
248
|
+
this.cacheMisses = 0;
|
|
249
|
+
}
|
|
250
|
+
/**
|
|
251
|
+
* Get cache statistics
|
|
252
|
+
*/
|
|
253
|
+
getCacheStats() {
|
|
254
|
+
const total = this.cacheHits + this.cacheMisses;
|
|
255
|
+
return {
|
|
256
|
+
size: this.astCache.size,
|
|
257
|
+
hits: this.cacheHits,
|
|
258
|
+
misses: this.cacheMisses,
|
|
259
|
+
hitRate: total > 0 ? this.cacheHits / total : 0,
|
|
260
|
+
};
|
|
261
|
+
}
|
|
262
|
+
/**
|
|
263
|
+
* Get the decimal utilities instance
|
|
264
|
+
*/
|
|
265
|
+
getDecimalUtils() {
|
|
266
|
+
return this.decimalUtils;
|
|
267
|
+
}
|
|
268
|
+
/**
|
|
269
|
+
* Create a Decimal from a value
|
|
270
|
+
*/
|
|
271
|
+
createDecimal(value) {
|
|
272
|
+
return this.decimalUtils.from(value);
|
|
273
|
+
}
|
|
274
|
+
// ============================================================================
|
|
275
|
+
// Private methods
|
|
276
|
+
// ============================================================================
|
|
277
|
+
checkExpressionLength(expression) {
|
|
278
|
+
const maxLength = this.config.security?.maxExpressionLength ?? 10000;
|
|
279
|
+
if (expression.length > maxLength) {
|
|
280
|
+
throw new errors_1.MaxExpressionLengthError(expression.length, maxLength);
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
normalizeContext(context) {
|
|
284
|
+
// Deep copy and convert numeric values to Decimal if autoConvertFloats is enabled
|
|
285
|
+
const autoConvert = this.config.decimal?.autoConvertFloats ?? true;
|
|
286
|
+
if (!autoConvert) {
|
|
287
|
+
return { ...context };
|
|
288
|
+
}
|
|
289
|
+
const normalizedVariables = {};
|
|
290
|
+
for (const [key, value] of Object.entries(context.variables)) {
|
|
291
|
+
normalizedVariables[key] = this.convertValue(value);
|
|
292
|
+
}
|
|
293
|
+
return {
|
|
294
|
+
...context,
|
|
295
|
+
variables: normalizedVariables,
|
|
296
|
+
};
|
|
297
|
+
}
|
|
298
|
+
convertValue(value) {
|
|
299
|
+
if (typeof value === 'number' && !isNaN(value) && isFinite(value)) {
|
|
300
|
+
return this.decimalUtils.from(value);
|
|
301
|
+
}
|
|
302
|
+
if (Array.isArray(value)) {
|
|
303
|
+
return value.map(v => this.convertValue(v));
|
|
304
|
+
}
|
|
305
|
+
if (value !== null && typeof value === 'object') {
|
|
306
|
+
const converted = {};
|
|
307
|
+
for (const [k, v] of Object.entries(value)) {
|
|
308
|
+
converted[k] = this.convertValue(v);
|
|
309
|
+
}
|
|
310
|
+
return converted;
|
|
311
|
+
}
|
|
312
|
+
return value;
|
|
313
|
+
}
|
|
314
|
+
isDecimal(value) {
|
|
315
|
+
return value instanceof decimal_utils_1.Decimal;
|
|
316
|
+
}
|
|
317
|
+
applyRounding(value, config) {
|
|
318
|
+
if (config.mode === 'NONE') {
|
|
319
|
+
return value;
|
|
320
|
+
}
|
|
321
|
+
return this.decimalUtils.round(value, config.precision, config.mode);
|
|
322
|
+
}
|
|
323
|
+
handleError(formula, _error) {
|
|
324
|
+
const behavior = formula.onError;
|
|
325
|
+
if (!behavior)
|
|
326
|
+
return null;
|
|
327
|
+
switch (behavior.type) {
|
|
328
|
+
case 'NULL':
|
|
329
|
+
return null;
|
|
330
|
+
case 'ZERO':
|
|
331
|
+
return this.decimalUtils.zero();
|
|
332
|
+
case 'DEFAULT':
|
|
333
|
+
return behavior.defaultValue ?? formula.defaultValue ?? null;
|
|
334
|
+
case 'SKIP':
|
|
335
|
+
return undefined;
|
|
336
|
+
case 'THROW':
|
|
337
|
+
default:
|
|
338
|
+
throw _error;
|
|
339
|
+
}
|
|
340
|
+
}
|
|
341
|
+
maybeEvictCache() {
|
|
342
|
+
const maxSize = this.config.maxCacheSize ?? 1000;
|
|
343
|
+
if (this.astCache.size >= maxSize) {
|
|
344
|
+
// Simple FIFO eviction - remove first 10% of entries
|
|
345
|
+
const toRemove = Math.ceil(maxSize * 0.1);
|
|
346
|
+
const keys = Array.from(this.astCache.keys()).slice(0, toRemove);
|
|
347
|
+
for (const key of keys) {
|
|
348
|
+
this.astCache.delete(key);
|
|
349
|
+
this.dependencyCache.delete(key);
|
|
350
|
+
}
|
|
351
|
+
}
|
|
352
|
+
}
|
|
353
|
+
}
|
|
354
|
+
exports.FormulaEngine = FormulaEngine;
|
|
355
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"formula-engine.js","sourceRoot":"","sources":["../src/formula-engine.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  FormulaEngineConfig,\n  FormulaDefinition,\n  EvaluationContext,\n  EvaluationResult,\n  EvaluationResultSet,\n  ValidationResult,\n  FunctionDefinition,\n  ASTNode,\n  CacheStats,\n  DependencyGraph as IDependencyGraph,\n} from './types';\nimport { Parser } from './parser';\nimport { Evaluator } from './evaluator';\nimport { DependencyExtractor } from './dependency-extractor';\nimport { DependencyGraph, DependencyGraphBuilder } from './dependency-graph';\nimport { DecimalUtils, Decimal } from './decimal-utils';\nimport { createBuiltInFunctions } from './functions';\nimport {\n  FormulaEngineError,\n  GeneralFormulaError,\n  DuplicateFormulaError,\n  MaxExpressionLengthError,\n} from './errors';\n\nexport class FormulaEngine {\n  private config: FormulaEngineConfig;\n  private parser: Parser;\n  private evaluator: Evaluator;\n  private dependencyExtractor: DependencyExtractor;\n  private graphBuilder: DependencyGraphBuilder;\n  private decimalUtils: DecimalUtils;\n  private functions: Map<string, FunctionDefinition>;\n\n  // Caches\n  private astCache: Map<string, ASTNode> = new Map();\n  private dependencyCache: Map<string, Set<string>> = new Map();\n  private cacheHits: number = 0;\n  private cacheMisses: number = 0;\n\n  constructor(config?: FormulaEngineConfig) {\n    this.config = {\n      enableCache: true,\n      maxCacheSize: 1000,\n      strictMode: true,\n      ...config,\n    };\n\n    this.decimalUtils = new DecimalUtils(this.config.decimal);\n    this.functions = createBuiltInFunctions(this.decimalUtils);\n    this.parser = new Parser();\n    this.dependencyExtractor = new DependencyExtractor();\n    this.graphBuilder = new DependencyGraphBuilder();\n    this.evaluator = new Evaluator(this.decimalUtils, this.functions, this.config);\n\n    // Register any custom functions from config\n    if (this.config.functions) {\n      for (const fn of this.config.functions) {\n        this.registerFunction(fn);\n      }\n    }\n  }\n\n  /**\n   * Parse an expression into an AST\n   */\n  parse(expression: string): ASTNode {\n    this.checkExpressionLength(expression);\n\n    if (this.config.enableCache) {\n      const cached = this.astCache.get(expression);\n      if (cached) {\n        this.cacheHits++;\n        return cached;\n      }\n      this.cacheMisses++;\n    }\n\n    const ast = this.parser.parse(expression);\n\n    if (this.config.enableCache) {\n      this.maybeEvictCache();\n      this.astCache.set(expression, ast);\n    }\n\n    return ast;\n  }\n\n  /**\n   * Extract variable dependencies from an expression\n   */\n  extractDependencies(expression: string): Set<string> {\n    if (this.config.enableCache) {\n      const cached = this.dependencyCache.get(expression);\n      if (cached) {\n        return new Set(cached);\n      }\n    }\n\n    const deps = this.dependencyExtractor.extract(expression);\n\n    if (this.config.enableCache) {\n      this.dependencyCache.set(expression, new Set(deps));\n    }\n\n    return deps;\n  }\n\n  /**\n   * Build a dependency graph from formula definitions\n   */\n  buildDependencyGraph(formulas: FormulaDefinition[]): IDependencyGraph {\n    return this.graphBuilder.build(formulas);\n  }\n\n  /**\n   * Get the evaluation order for formulas\n   */\n  getEvaluationOrder(formulas: FormulaDefinition[]): string[] {\n    return this.graphBuilder.getEvaluationOrder(formulas);\n  }\n\n  /**\n   * Validate formulas without evaluating\n   */\n  validate(formulas: FormulaDefinition[]): ValidationResult {\n    const errors: FormulaEngineError[] = [];\n    const warnings: string[] = [];\n\n    // Check for duplicate IDs\n    const ids = new Set<string>();\n    for (const formula of formulas) {\n      if (ids.has(formula.id)) {\n        errors.push(new DuplicateFormulaError(formula.id));\n      }\n      ids.add(formula.id);\n    }\n\n    // Try to parse all expressions\n    for (const formula of formulas) {\n      try {\n        this.parse(formula.expression);\n      } catch (error) {\n        if (error instanceof FormulaEngineError) {\n          errors.push(error);\n        } else {\n          errors.push(new GeneralFormulaError(String(error)));\n        }\n      }\n    }\n\n    // Build dependency graph and check for cycles\n    let dependencyGraph: IDependencyGraph = new DependencyGraph();\n    let evaluationOrder: string[] = [];\n\n    try {\n      dependencyGraph = this.buildDependencyGraph(formulas);\n      evaluationOrder = dependencyGraph.topologicalSort();\n    } catch (error) {\n      if (error instanceof FormulaEngineError) {\n        errors.push(error);\n      }\n    }\n\n    return {\n      valid: errors.length === 0,\n      errors,\n      warnings,\n      dependencyGraph,\n      evaluationOrder,\n    };\n  }\n\n  /**\n   * Evaluate a single expression\n   */\n  evaluate(expression: string, context: EvaluationContext): EvaluationResult {\n    this.checkExpressionLength(expression);\n    const normalizedContext = this.normalizeContext(context);\n    return this.evaluator.evaluate(expression, normalizedContext);\n  }\n\n  /**\n   * Evaluate all formulas in dependency order\n   */\n  evaluateAll(\n    formulas: FormulaDefinition[],\n    context: EvaluationContext\n  ): EvaluationResultSet {\n    const startTime = Date.now();\n    const results = new Map<string, EvaluationResult>();\n    const errors: Error[] = [];\n\n    // Get evaluation order\n    let evaluationOrder: string[];\n    try {\n      evaluationOrder = this.getEvaluationOrder(formulas);\n    } catch (error) {\n      return {\n        results,\n        success: false,\n        errors: [error as Error],\n        totalExecutionTimeMs: Date.now() - startTime,\n        evaluationOrder: [],\n      };\n    }\n\n    // Create a map of formulas by ID\n    const formulaMap = new Map<string, FormulaDefinition>();\n    for (const formula of formulas) {\n      formulaMap.set(formula.id, formula);\n    }\n\n    // Evaluate in order, merging results into context\n    const workingContext: EvaluationContext = this.normalizeContext(context);\n\n    for (const formulaId of evaluationOrder) {\n      const formula = formulaMap.get(formulaId);\n      if (!formula) continue;\n\n      try {\n        const result = this.evaluator.evaluate(formula.expression, workingContext);\n\n        // Apply rounding if configured\n        let value = result.value;\n        if (formula.rounding && this.isDecimal(value)) {\n          value = this.applyRounding(value as Decimal, formula.rounding);\n        }\n\n        // Handle errors based on formula config\n        if (!result.success && formula.onError) {\n          value = this.handleError(formula, result.error);\n        }\n\n        results.set(formulaId, {\n          ...result,\n          value,\n        });\n\n        // Merge result into context for subsequent formulas\n        workingContext.variables[formulaId] = value;\n\n        if (!result.success) {\n          errors.push(result.error!);\n        }\n      } catch (error) {\n        const evalResult: EvaluationResult = {\n          value: null,\n          success: false,\n          error: error as Error,\n          executionTimeMs: 0,\n          accessedVariables: new Set(),\n        };\n\n        // Handle error based on formula config\n        if (formula.onError) {\n          evalResult.value = this.handleError(formula, error as Error);\n        }\n\n        results.set(formulaId, evalResult);\n        errors.push(error as Error);\n\n        // Still add to context (as null or default value) so dependent formulas can proceed\n        workingContext.variables[formulaId] = evalResult.value;\n      }\n    }\n\n    return {\n      results,\n      success: errors.length === 0,\n      errors,\n      totalExecutionTimeMs: Date.now() - startTime,\n      evaluationOrder,\n    };\n  }\n\n  /**\n   * Register a custom function\n   */\n  registerFunction(definition: FunctionDefinition): void {\n    const name = definition.name.toUpperCase();\n    this.functions.set(name, definition);\n  }\n\n  /**\n   * Register multiple custom functions\n   */\n  registerFunctions(definitions: FunctionDefinition[]): void {\n    for (const definition of definitions) {\n      this.registerFunction(definition);\n    }\n  }\n\n  /**\n   * Get registered function names\n   */\n  getRegisteredFunctions(): string[] {\n    return Array.from(this.functions.keys());\n  }\n\n  /**\n   * Clear the AST cache\n   */\n  clearCache(): void {\n    this.astCache.clear();\n    this.dependencyCache.clear();\n    this.cacheHits = 0;\n    this.cacheMisses = 0;\n  }\n\n  /**\n   * Get cache statistics\n   */\n  getCacheStats(): CacheStats {\n    const total = this.cacheHits + this.cacheMisses;\n    return {\n      size: this.astCache.size,\n      hits: this.cacheHits,\n      misses: this.cacheMisses,\n      hitRate: total > 0 ? this.cacheHits / total : 0,\n    };\n  }\n\n  /**\n   * Get the decimal utilities instance\n   */\n  getDecimalUtils(): DecimalUtils {\n    return this.decimalUtils;\n  }\n\n  /**\n   * Create a Decimal from a value\n   */\n  createDecimal(value: string | number): Decimal {\n    return this.decimalUtils.from(value);\n  }\n\n  // ============================================================================\n  // Private methods\n  // ============================================================================\n\n  private checkExpressionLength(expression: string): void {\n    const maxLength = this.config.security?.maxExpressionLength ?? 10000;\n    if (expression.length > maxLength) {\n      throw new MaxExpressionLengthError(expression.length, maxLength);\n    }\n  }\n\n  private normalizeContext(context: EvaluationContext): EvaluationContext {\n    // Deep copy and convert numeric values to Decimal if autoConvertFloats is enabled\n    const autoConvert = this.config.decimal?.autoConvertFloats ?? true;\n\n    if (!autoConvert) {\n      return { ...context };\n    }\n\n    const normalizedVariables: Record<string, unknown> = {};\n    for (const [key, value] of Object.entries(context.variables)) {\n      normalizedVariables[key] = this.convertValue(value);\n    }\n\n    return {\n      ...context,\n      variables: normalizedVariables,\n    };\n  }\n\n  private convertValue(value: unknown): unknown {\n    if (typeof value === 'number' && !isNaN(value) && isFinite(value)) {\n      return this.decimalUtils.from(value);\n    }\n    if (Array.isArray(value)) {\n      return value.map(v => this.convertValue(v));\n    }\n    if (value !== null && typeof value === 'object') {\n      const converted: Record<string, unknown> = {};\n      for (const [k, v] of Object.entries(value)) {\n        converted[k] = this.convertValue(v);\n      }\n      return converted;\n    }\n    return value;\n  }\n\n  private isDecimal(value: unknown): boolean {\n    return value instanceof Decimal;\n  }\n\n  private applyRounding(\n    value: Decimal,\n    config: { mode: string; precision: number }\n  ): Decimal {\n    if (config.mode === 'NONE') {\n      return value;\n    }\n    return this.decimalUtils.round(value, config.precision, config.mode as any);\n  }\n\n  private handleError(formula: FormulaDefinition, _error?: Error): unknown {\n    const behavior = formula.onError;\n    if (!behavior) return null;\n\n    switch (behavior.type) {\n      case 'NULL':\n        return null;\n      case 'ZERO':\n        return this.decimalUtils.zero();\n      case 'DEFAULT':\n        return behavior.defaultValue ?? formula.defaultValue ?? null;\n      case 'SKIP':\n        return undefined;\n      case 'THROW':\n      default:\n        throw _error;\n    }\n  }\n\n  private maybeEvictCache(): void {\n    const maxSize = this.config.maxCacheSize ?? 1000;\n    if (this.astCache.size >= maxSize) {\n      // Simple FIFO eviction - remove first 10% of entries\n      const toRemove = Math.ceil(maxSize * 0.1);\n      const keys = Array.from(this.astCache.keys()).slice(0, toRemove);\n      for (const key of keys) {\n        this.astCache.delete(key);\n        this.dependencyCache.delete(key);\n      }\n    }\n  }\n}\n"]}
|