@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.
Files changed (43) hide show
  1. package/.claude/settings.local.json +6 -0
  2. package/PRD_FORMULA_ENGINE.md +1863 -0
  3. package/README.md +382 -0
  4. package/dist/decimal-utils.d.ts +180 -0
  5. package/dist/decimal-utils.js +355 -0
  6. package/dist/dependency-extractor.d.ts +20 -0
  7. package/dist/dependency-extractor.js +103 -0
  8. package/dist/dependency-graph.d.ts +60 -0
  9. package/dist/dependency-graph.js +252 -0
  10. package/dist/errors.d.ts +161 -0
  11. package/dist/errors.js +260 -0
  12. package/dist/evaluator.d.ts +51 -0
  13. package/dist/evaluator.js +494 -0
  14. package/dist/formula-engine.d.ts +79 -0
  15. package/dist/formula-engine.js +355 -0
  16. package/dist/functions.d.ts +3 -0
  17. package/dist/functions.js +720 -0
  18. package/dist/index.d.ts +10 -0
  19. package/dist/index.js +61 -0
  20. package/dist/lexer.d.ts +25 -0
  21. package/dist/lexer.js +357 -0
  22. package/dist/parser.d.ts +32 -0
  23. package/dist/parser.js +372 -0
  24. package/dist/types.d.ts +228 -0
  25. package/dist/types.js +62 -0
  26. package/jest.config.js +23 -0
  27. package/package.json +35 -0
  28. package/src/decimal-utils.ts +408 -0
  29. package/src/dependency-extractor.ts +117 -0
  30. package/src/dependency-graph.test.ts +238 -0
  31. package/src/dependency-graph.ts +288 -0
  32. package/src/errors.ts +296 -0
  33. package/src/evaluator.ts +604 -0
  34. package/src/formula-engine.test.ts +660 -0
  35. package/src/formula-engine.ts +430 -0
  36. package/src/functions.ts +770 -0
  37. package/src/index.ts +103 -0
  38. package/src/lexer.test.ts +288 -0
  39. package/src/lexer.ts +394 -0
  40. package/src/parser.test.ts +349 -0
  41. package/src/parser.ts +449 -0
  42. package/src/types.ts +347 -0
  43. 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"]}
@@ -0,0 +1,3 @@
1
+ import { FunctionDefinition } from './types';
2
+ import { DecimalUtils } from './decimal-utils';
3
+ export declare function createBuiltInFunctions(decimalUtils: DecimalUtils): Map<string, FunctionDefinition>;