@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
package/dist/errors.js
ADDED
|
@@ -0,0 +1,260 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.GeneralFormulaError = exports.MaxExpressionLengthError = exports.MaxRecursionError = exports.MaxIterationsError = exports.SecurityError = exports.ConfigurationError = exports.InvalidDecimalError = exports.DecimalDivisionByZeroError = exports.DecimalUnderflowError = exports.DecimalOverflowError = exports.DecimalError = exports.IndexAccessError = exports.PropertyAccessError = exports.InvalidOperationError = exports.ArgumentCountError = exports.TypeMismatchError = exports.DivisionByZeroError = exports.DuplicateFormulaError = exports.UndefinedFunctionError = exports.UndefinedVariableError = exports.CircularDependencyError = exports.InvalidNumberError = exports.UnterminatedStringError = exports.UnexpectedTokenError = exports.SyntaxError = exports.FormulaEngineError = void 0;
|
|
4
|
+
class FormulaEngineError extends Error {
|
|
5
|
+
constructor(message) {
|
|
6
|
+
super(message);
|
|
7
|
+
this.name = this.constructor.name;
|
|
8
|
+
Object.setPrototypeOf(this, new.target.prototype);
|
|
9
|
+
}
|
|
10
|
+
}
|
|
11
|
+
exports.FormulaEngineError = FormulaEngineError;
|
|
12
|
+
// ============================================================================
|
|
13
|
+
// Parse Errors
|
|
14
|
+
// ============================================================================
|
|
15
|
+
class SyntaxError extends FormulaEngineError {
|
|
16
|
+
constructor(message, position, line, column, expression) {
|
|
17
|
+
super(`${message} at line ${line}, column ${column}`);
|
|
18
|
+
this.position = position;
|
|
19
|
+
this.line = line;
|
|
20
|
+
this.column = column;
|
|
21
|
+
this.expression = expression;
|
|
22
|
+
this.code = 'PARSE_SYNTAX_ERROR';
|
|
23
|
+
this.category = 'PARSE';
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
exports.SyntaxError = SyntaxError;
|
|
27
|
+
class UnexpectedTokenError extends FormulaEngineError {
|
|
28
|
+
constructor(token, expected, position) {
|
|
29
|
+
super(`Unexpected token '${token}', expected one of: ${expected.join(', ')}`);
|
|
30
|
+
this.token = token;
|
|
31
|
+
this.expected = expected;
|
|
32
|
+
this.position = position;
|
|
33
|
+
this.code = 'PARSE_UNEXPECTED_TOKEN';
|
|
34
|
+
this.category = 'PARSE';
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
exports.UnexpectedTokenError = UnexpectedTokenError;
|
|
38
|
+
class UnterminatedStringError extends FormulaEngineError {
|
|
39
|
+
constructor(position) {
|
|
40
|
+
super(`Unterminated string starting at position ${position}`);
|
|
41
|
+
this.code = 'PARSE_UNTERMINATED_STRING';
|
|
42
|
+
this.category = 'PARSE';
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
exports.UnterminatedStringError = UnterminatedStringError;
|
|
46
|
+
class InvalidNumberError extends FormulaEngineError {
|
|
47
|
+
constructor(value, position) {
|
|
48
|
+
super(`Invalid number '${value}' at position ${position}`);
|
|
49
|
+
this.code = 'PARSE_INVALID_NUMBER';
|
|
50
|
+
this.category = 'PARSE';
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
exports.InvalidNumberError = InvalidNumberError;
|
|
54
|
+
// ============================================================================
|
|
55
|
+
// Validation Errors
|
|
56
|
+
// ============================================================================
|
|
57
|
+
class CircularDependencyError extends FormulaEngineError {
|
|
58
|
+
constructor(cycle, involvedFormulas) {
|
|
59
|
+
super(`Circular dependency detected: ${cycle.join(' → ')}`);
|
|
60
|
+
this.cycle = cycle;
|
|
61
|
+
this.involvedFormulas = involvedFormulas;
|
|
62
|
+
this.code = 'VALIDATION_CIRCULAR_DEPENDENCY';
|
|
63
|
+
this.category = 'VALIDATION';
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
exports.CircularDependencyError = CircularDependencyError;
|
|
67
|
+
class UndefinedVariableError extends FormulaEngineError {
|
|
68
|
+
constructor(variableName, expression) {
|
|
69
|
+
super(`Undefined variable: ${variableName}`);
|
|
70
|
+
this.variableName = variableName;
|
|
71
|
+
this.expression = expression;
|
|
72
|
+
this.code = 'VALIDATION_UNDEFINED_VARIABLE';
|
|
73
|
+
this.category = 'VALIDATION';
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
exports.UndefinedVariableError = UndefinedVariableError;
|
|
77
|
+
class UndefinedFunctionError extends FormulaEngineError {
|
|
78
|
+
constructor(functionName) {
|
|
79
|
+
super(`Undefined function: ${functionName}`);
|
|
80
|
+
this.functionName = functionName;
|
|
81
|
+
this.code = 'VALIDATION_UNDEFINED_FUNCTION';
|
|
82
|
+
this.category = 'VALIDATION';
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
exports.UndefinedFunctionError = UndefinedFunctionError;
|
|
86
|
+
class DuplicateFormulaError extends FormulaEngineError {
|
|
87
|
+
constructor(formulaId) {
|
|
88
|
+
super(`Duplicate formula ID: ${formulaId}`);
|
|
89
|
+
this.formulaId = formulaId;
|
|
90
|
+
this.code = 'VALIDATION_DUPLICATE_FORMULA';
|
|
91
|
+
this.category = 'VALIDATION';
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
exports.DuplicateFormulaError = DuplicateFormulaError;
|
|
95
|
+
// ============================================================================
|
|
96
|
+
// Evaluation Errors
|
|
97
|
+
// ============================================================================
|
|
98
|
+
class DivisionByZeroError extends FormulaEngineError {
|
|
99
|
+
constructor() {
|
|
100
|
+
super('Division by zero');
|
|
101
|
+
this.code = 'EVAL_DIVISION_BY_ZERO';
|
|
102
|
+
this.category = 'EVALUATION';
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
exports.DivisionByZeroError = DivisionByZeroError;
|
|
106
|
+
class TypeMismatchError extends FormulaEngineError {
|
|
107
|
+
constructor(expected, actual, context) {
|
|
108
|
+
const expectedStr = Array.isArray(expected) ? expected.join(' or ') : expected;
|
|
109
|
+
super(`Type mismatch: expected ${expectedStr}, got ${actual} in ${context}`);
|
|
110
|
+
this.expected = expected;
|
|
111
|
+
this.actual = actual;
|
|
112
|
+
this.context = context;
|
|
113
|
+
this.code = 'EVAL_TYPE_MISMATCH';
|
|
114
|
+
this.category = 'EVALUATION';
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
exports.TypeMismatchError = TypeMismatchError;
|
|
118
|
+
class ArgumentCountError extends FormulaEngineError {
|
|
119
|
+
constructor(functionName, expected, actual) {
|
|
120
|
+
const expectedStr = expected.min === expected.max
|
|
121
|
+
? `${expected.min}`
|
|
122
|
+
: expected.max === -1
|
|
123
|
+
? `at least ${expected.min}`
|
|
124
|
+
: `${expected.min}-${expected.max}`;
|
|
125
|
+
super(`Function ${functionName} expects ${expectedStr} arguments, got ${actual}`);
|
|
126
|
+
this.functionName = functionName;
|
|
127
|
+
this.expected = expected;
|
|
128
|
+
this.actual = actual;
|
|
129
|
+
this.code = 'EVAL_ARGUMENT_COUNT';
|
|
130
|
+
this.category = 'EVALUATION';
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
exports.ArgumentCountError = ArgumentCountError;
|
|
134
|
+
class InvalidOperationError extends FormulaEngineError {
|
|
135
|
+
constructor(operator, operandTypes) {
|
|
136
|
+
super(`Cannot apply operator '${operator}' to types: ${operandTypes.join(', ')}`);
|
|
137
|
+
this.operator = operator;
|
|
138
|
+
this.operandTypes = operandTypes;
|
|
139
|
+
this.code = 'EVAL_INVALID_OPERATION';
|
|
140
|
+
this.category = 'EVALUATION';
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
exports.InvalidOperationError = InvalidOperationError;
|
|
144
|
+
class PropertyAccessError extends FormulaEngineError {
|
|
145
|
+
constructor(property, objectType) {
|
|
146
|
+
super(`Cannot access property '${property}' on ${objectType}`);
|
|
147
|
+
this.property = property;
|
|
148
|
+
this.objectType = objectType;
|
|
149
|
+
this.code = 'EVAL_PROPERTY_ACCESS';
|
|
150
|
+
this.category = 'EVALUATION';
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
exports.PropertyAccessError = PropertyAccessError;
|
|
154
|
+
class IndexAccessError extends FormulaEngineError {
|
|
155
|
+
constructor(index, objectType) {
|
|
156
|
+
super(`Cannot use index '${index}' on ${objectType}`);
|
|
157
|
+
this.index = index;
|
|
158
|
+
this.objectType = objectType;
|
|
159
|
+
this.code = 'EVAL_INDEX_ACCESS';
|
|
160
|
+
this.category = 'EVALUATION';
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
exports.IndexAccessError = IndexAccessError;
|
|
164
|
+
// ============================================================================
|
|
165
|
+
// Decimal Errors
|
|
166
|
+
// ============================================================================
|
|
167
|
+
class DecimalError extends FormulaEngineError {
|
|
168
|
+
constructor(message) {
|
|
169
|
+
super(message);
|
|
170
|
+
this.code = 'DECIMAL_ERROR';
|
|
171
|
+
this.category = 'EVALUATION';
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
exports.DecimalError = DecimalError;
|
|
175
|
+
class DecimalOverflowError extends DecimalError {
|
|
176
|
+
constructor(value, maxExponent) {
|
|
177
|
+
super(`Decimal overflow: exponent exceeds ${maxExponent}`);
|
|
178
|
+
this.value = value;
|
|
179
|
+
this.maxExponent = maxExponent;
|
|
180
|
+
this.code = 'DECIMAL_OVERFLOW';
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
exports.DecimalOverflowError = DecimalOverflowError;
|
|
184
|
+
class DecimalUnderflowError extends DecimalError {
|
|
185
|
+
constructor(value, minExponent) {
|
|
186
|
+
super(`Decimal underflow: exponent below ${minExponent}`);
|
|
187
|
+
this.value = value;
|
|
188
|
+
this.minExponent = minExponent;
|
|
189
|
+
this.code = 'DECIMAL_UNDERFLOW';
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
exports.DecimalUnderflowError = DecimalUnderflowError;
|
|
193
|
+
class DecimalDivisionByZeroError extends DecimalError {
|
|
194
|
+
constructor() {
|
|
195
|
+
super('Division by zero');
|
|
196
|
+
this.code = 'DECIMAL_DIVISION_BY_ZERO';
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
exports.DecimalDivisionByZeroError = DecimalDivisionByZeroError;
|
|
200
|
+
class InvalidDecimalError extends DecimalError {
|
|
201
|
+
constructor(input) {
|
|
202
|
+
super(`Invalid decimal value: "${input}"`);
|
|
203
|
+
this.input = input;
|
|
204
|
+
this.code = 'INVALID_DECIMAL';
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
exports.InvalidDecimalError = InvalidDecimalError;
|
|
208
|
+
// ============================================================================
|
|
209
|
+
// Configuration Errors
|
|
210
|
+
// ============================================================================
|
|
211
|
+
class ConfigurationError extends FormulaEngineError {
|
|
212
|
+
constructor(message) {
|
|
213
|
+
super(message);
|
|
214
|
+
this.code = 'CONFIGURATION_ERROR';
|
|
215
|
+
this.category = 'CONFIGURATION';
|
|
216
|
+
}
|
|
217
|
+
}
|
|
218
|
+
exports.ConfigurationError = ConfigurationError;
|
|
219
|
+
// ============================================================================
|
|
220
|
+
// Security Errors
|
|
221
|
+
// ============================================================================
|
|
222
|
+
class SecurityError extends FormulaEngineError {
|
|
223
|
+
constructor(message) {
|
|
224
|
+
super(message);
|
|
225
|
+
this.code = 'SECURITY_ERROR';
|
|
226
|
+
this.category = 'EVALUATION';
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
exports.SecurityError = SecurityError;
|
|
230
|
+
class MaxIterationsError extends SecurityError {
|
|
231
|
+
constructor(limit) {
|
|
232
|
+
super(`Maximum iterations exceeded: ${limit}`);
|
|
233
|
+
this.code = 'MAX_ITERATIONS_EXCEEDED';
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
exports.MaxIterationsError = MaxIterationsError;
|
|
237
|
+
class MaxRecursionError extends SecurityError {
|
|
238
|
+
constructor(limit) {
|
|
239
|
+
super(`Maximum recursion depth exceeded: ${limit}`);
|
|
240
|
+
this.code = 'MAX_RECURSION_EXCEEDED';
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
exports.MaxRecursionError = MaxRecursionError;
|
|
244
|
+
class MaxExpressionLengthError extends SecurityError {
|
|
245
|
+
constructor(length, limit) {
|
|
246
|
+
super(`Expression length ${length} exceeds maximum ${limit}`);
|
|
247
|
+
this.code = 'MAX_EXPRESSION_LENGTH_EXCEEDED';
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
exports.MaxExpressionLengthError = MaxExpressionLengthError;
|
|
251
|
+
// Concrete class for general errors
|
|
252
|
+
class GeneralFormulaError extends FormulaEngineError {
|
|
253
|
+
constructor(message) {
|
|
254
|
+
super(message);
|
|
255
|
+
this.code = 'GENERAL_ERROR';
|
|
256
|
+
this.category = 'EVALUATION';
|
|
257
|
+
}
|
|
258
|
+
}
|
|
259
|
+
exports.GeneralFormulaError = GeneralFormulaError;
|
|
260
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"errors.js","sourceRoot":"","sources":["../src/errors.ts"],"names":[],"mappings":"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","sourcesContent":["import { ValueType } from './types';\n\nexport type ErrorCategory = 'PARSE' | 'VALIDATION' | 'EVALUATION' | 'CONFIGURATION';\n\nexport abstract class FormulaEngineError extends Error {\n  abstract readonly code: string;\n  abstract readonly category: ErrorCategory;\n\n  constructor(message: string) {\n    super(message);\n    this.name = this.constructor.name;\n    Object.setPrototypeOf(this, new.target.prototype);\n  }\n}\n\n// ============================================================================\n// Parse Errors\n// ============================================================================\n\nexport class SyntaxError extends FormulaEngineError {\n  readonly code = 'PARSE_SYNTAX_ERROR';\n  readonly category: ErrorCategory = 'PARSE';\n\n  constructor(\n    message: string,\n    public position: number,\n    public line: number,\n    public column: number,\n    public expression: string\n  ) {\n    super(`${message} at line ${line}, column ${column}`);\n  }\n}\n\nexport class UnexpectedTokenError extends FormulaEngineError {\n  readonly code = 'PARSE_UNEXPECTED_TOKEN';\n  readonly category: ErrorCategory = 'PARSE';\n\n  constructor(\n    public token: string,\n    public expected: string[],\n    public position: number\n  ) {\n    super(`Unexpected token '${token}', expected one of: ${expected.join(', ')}`);\n  }\n}\n\nexport class UnterminatedStringError extends FormulaEngineError {\n  readonly code = 'PARSE_UNTERMINATED_STRING';\n  readonly category: ErrorCategory = 'PARSE';\n\n  constructor(position: number) {\n    super(`Unterminated string starting at position ${position}`);\n  }\n}\n\nexport class InvalidNumberError extends FormulaEngineError {\n  readonly code = 'PARSE_INVALID_NUMBER';\n  readonly category: ErrorCategory = 'PARSE';\n\n  constructor(value: string, position: number) {\n    super(`Invalid number '${value}' at position ${position}`);\n  }\n}\n\n// ============================================================================\n// Validation Errors\n// ============================================================================\n\nexport class CircularDependencyError extends FormulaEngineError {\n  readonly code = 'VALIDATION_CIRCULAR_DEPENDENCY';\n  readonly category: ErrorCategory = 'VALIDATION';\n\n  constructor(\n    public cycle: string[],\n    public involvedFormulas: string[]\n  ) {\n    super(`Circular dependency detected: ${cycle.join(' → ')}`);\n  }\n}\n\nexport class UndefinedVariableError extends FormulaEngineError {\n  readonly code = 'VALIDATION_UNDEFINED_VARIABLE';\n  readonly category: ErrorCategory = 'VALIDATION';\n\n  constructor(\n    public variableName: string,\n    public expression: string\n  ) {\n    super(`Undefined variable: ${variableName}`);\n  }\n}\n\nexport class UndefinedFunctionError extends FormulaEngineError {\n  readonly code = 'VALIDATION_UNDEFINED_FUNCTION';\n  readonly category: ErrorCategory = 'VALIDATION';\n\n  constructor(public functionName: string) {\n    super(`Undefined function: ${functionName}`);\n  }\n}\n\nexport class DuplicateFormulaError extends FormulaEngineError {\n  readonly code = 'VALIDATION_DUPLICATE_FORMULA';\n  readonly category: ErrorCategory = 'VALIDATION';\n\n  constructor(public formulaId: string) {\n    super(`Duplicate formula ID: ${formulaId}`);\n  }\n}\n\n// ============================================================================\n// Evaluation Errors\n// ============================================================================\n\nexport class DivisionByZeroError extends FormulaEngineError {\n  readonly code = 'EVAL_DIVISION_BY_ZERO';\n  readonly category: ErrorCategory = 'EVALUATION';\n\n  constructor() {\n    super('Division by zero');\n  }\n}\n\nexport class TypeMismatchError extends FormulaEngineError {\n  readonly code = 'EVAL_TYPE_MISMATCH';\n  readonly category: ErrorCategory = 'EVALUATION';\n\n  constructor(\n    public expected: ValueType | ValueType[] | string,\n    public actual: ValueType | string,\n    public context: string\n  ) {\n    const expectedStr = Array.isArray(expected) ? expected.join(' or ') : expected;\n    super(`Type mismatch: expected ${expectedStr}, got ${actual} in ${context}`);\n  }\n}\n\nexport class ArgumentCountError extends FormulaEngineError {\n  readonly code = 'EVAL_ARGUMENT_COUNT';\n  readonly category: ErrorCategory = 'EVALUATION';\n\n  constructor(\n    public functionName: string,\n    public expected: { min: number; max: number },\n    public actual: number\n  ) {\n    const expectedStr = expected.min === expected.max\n      ? `${expected.min}`\n      : expected.max === -1\n        ? `at least ${expected.min}`\n        : `${expected.min}-${expected.max}`;\n    super(`Function ${functionName} expects ${expectedStr} arguments, got ${actual}`);\n  }\n}\n\nexport class InvalidOperationError extends FormulaEngineError {\n  readonly code = 'EVAL_INVALID_OPERATION';\n  readonly category: ErrorCategory = 'EVALUATION';\n\n  constructor(\n    public operator: string,\n    public operandTypes: string[]\n  ) {\n    super(`Cannot apply operator '${operator}' to types: ${operandTypes.join(', ')}`);\n  }\n}\n\nexport class PropertyAccessError extends FormulaEngineError {\n  readonly code = 'EVAL_PROPERTY_ACCESS';\n  readonly category: ErrorCategory = 'EVALUATION';\n\n  constructor(\n    public property: string,\n    public objectType: string\n  ) {\n    super(`Cannot access property '${property}' on ${objectType}`);\n  }\n}\n\nexport class IndexAccessError extends FormulaEngineError {\n  readonly code = 'EVAL_INDEX_ACCESS';\n  readonly category: ErrorCategory = 'EVALUATION';\n\n  constructor(\n    public index: unknown,\n    public objectType: string\n  ) {\n    super(`Cannot use index '${index}' on ${objectType}`);\n  }\n}\n\n// ============================================================================\n// Decimal Errors\n// ============================================================================\n\nexport class DecimalError extends FormulaEngineError {\n  readonly code: string = 'DECIMAL_ERROR';\n  readonly category: ErrorCategory = 'EVALUATION';\n\n  constructor(message: string) {\n    super(message);\n  }\n}\n\nexport class DecimalOverflowError extends DecimalError {\n  override readonly code: string = 'DECIMAL_OVERFLOW';\n\n  constructor(public value: string, public maxExponent: number) {\n    super(`Decimal overflow: exponent exceeds ${maxExponent}`);\n  }\n}\n\nexport class DecimalUnderflowError extends DecimalError {\n  override readonly code: string = 'DECIMAL_UNDERFLOW';\n\n  constructor(public value: string, public minExponent: number) {\n    super(`Decimal underflow: exponent below ${minExponent}`);\n  }\n}\n\nexport class DecimalDivisionByZeroError extends DecimalError {\n  override readonly code: string = 'DECIMAL_DIVISION_BY_ZERO';\n\n  constructor() {\n    super('Division by zero');\n  }\n}\n\nexport class InvalidDecimalError extends DecimalError {\n  override readonly code: string = 'INVALID_DECIMAL';\n\n  constructor(public input: string) {\n    super(`Invalid decimal value: \"${input}\"`);\n  }\n}\n\n// ============================================================================\n// Configuration Errors\n// ============================================================================\n\nexport class ConfigurationError extends FormulaEngineError {\n  readonly code = 'CONFIGURATION_ERROR';\n  readonly category: ErrorCategory = 'CONFIGURATION';\n\n  constructor(message: string) {\n    super(message);\n  }\n}\n\n// ============================================================================\n// Security Errors\n// ============================================================================\n\nexport class SecurityError extends FormulaEngineError {\n  readonly code: string = 'SECURITY_ERROR';\n  readonly category: ErrorCategory = 'EVALUATION';\n\n  constructor(message: string) {\n    super(message);\n  }\n}\n\nexport class MaxIterationsError extends SecurityError {\n  override readonly code: string = 'MAX_ITERATIONS_EXCEEDED';\n\n  constructor(limit: number) {\n    super(`Maximum iterations exceeded: ${limit}`);\n  }\n}\n\nexport class MaxRecursionError extends SecurityError {\n  override readonly code: string = 'MAX_RECURSION_EXCEEDED';\n\n  constructor(limit: number) {\n    super(`Maximum recursion depth exceeded: ${limit}`);\n  }\n}\n\nexport class MaxExpressionLengthError extends SecurityError {\n  override readonly code: string = 'MAX_EXPRESSION_LENGTH_EXCEEDED';\n\n  constructor(length: number, limit: number) {\n    super(`Expression length ${length} exceeds maximum ${limit}`);\n  }\n}\n\n// Concrete class for general errors\nexport class GeneralFormulaError extends FormulaEngineError {\n  readonly code: string = 'GENERAL_ERROR';\n  readonly category: ErrorCategory = 'EVALUATION';\n\n  constructor(message: string) {\n    super(message);\n  }\n}\n"]}
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { ASTNode, EvaluationContext, EvaluationResult, FunctionDefinition, FormulaEngineConfig } from './types';
|
|
2
|
+
import { DecimalUtils } from './decimal-utils';
|
|
3
|
+
export declare class Evaluator {
|
|
4
|
+
private parser;
|
|
5
|
+
private decimalUtils;
|
|
6
|
+
private functions;
|
|
7
|
+
private strictMode;
|
|
8
|
+
private securityConfig;
|
|
9
|
+
private recursionDepth;
|
|
10
|
+
private iterationCount;
|
|
11
|
+
private accessedVariables;
|
|
12
|
+
constructor(decimalUtils: DecimalUtils, functions: Map<string, FunctionDefinition>, config?: FormulaEngineConfig);
|
|
13
|
+
/**
|
|
14
|
+
* Evaluate an expression string
|
|
15
|
+
*/
|
|
16
|
+
evaluate(expression: string, context: EvaluationContext): EvaluationResult;
|
|
17
|
+
/**
|
|
18
|
+
* Evaluate an AST node
|
|
19
|
+
*/
|
|
20
|
+
evaluateNode(node: ASTNode, context: EvaluationContext): unknown;
|
|
21
|
+
private checkRecursionLimit;
|
|
22
|
+
private checkIterationLimit;
|
|
23
|
+
private evaluateVariable;
|
|
24
|
+
private maybeConvertToDecimal;
|
|
25
|
+
private evaluateBinaryOperation;
|
|
26
|
+
private evaluateUnaryOperation;
|
|
27
|
+
private evaluateConditional;
|
|
28
|
+
private evaluateFunctionCall;
|
|
29
|
+
private evaluateSumWithExpression;
|
|
30
|
+
private evaluateFilter;
|
|
31
|
+
private evaluateMap;
|
|
32
|
+
private evaluateMemberAccess;
|
|
33
|
+
private evaluateIndexAccess;
|
|
34
|
+
private isNumeric;
|
|
35
|
+
private toDecimal;
|
|
36
|
+
private toNumber;
|
|
37
|
+
private toBool;
|
|
38
|
+
private typeOf;
|
|
39
|
+
private add;
|
|
40
|
+
private subtract;
|
|
41
|
+
private multiply;
|
|
42
|
+
private divide;
|
|
43
|
+
private modulo;
|
|
44
|
+
private power;
|
|
45
|
+
private negate;
|
|
46
|
+
private equals;
|
|
47
|
+
private lessThan;
|
|
48
|
+
private greaterThan;
|
|
49
|
+
private lessThanOrEqual;
|
|
50
|
+
private greaterThanOrEqual;
|
|
51
|
+
}
|