@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
package/dist/parser.js ADDED
@@ -0,0 +1,372 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.Parser = void 0;
4
+ const lexer_1 = require("./lexer");
5
+ const types_1 = require("./types");
6
+ const errors_1 = require("./errors");
7
+ // Operator precedence levels (higher = tighter binding)
8
+ const PRECEDENCE = {
9
+ LOWEST: 1,
10
+ TERNARY: 2, // ? :
11
+ OR: 3, // || OR
12
+ AND: 4, // && AND
13
+ EQUALITY: 5, // == !=
14
+ COMPARISON: 6, // < > <= >=
15
+ TERM: 7, // + -
16
+ FACTOR: 8, // * / %
17
+ POWER: 9, // ^
18
+ UNARY: 10, // - ! NOT
19
+ CALL: 11, // function calls
20
+ MEMBER: 12, // . []
21
+ };
22
+ class Parser {
23
+ constructor() {
24
+ this.tokens = [];
25
+ this.current = 0;
26
+ this.expression = '';
27
+ }
28
+ parse(expression) {
29
+ this.expression = expression;
30
+ const lexer = new lexer_1.Lexer(expression);
31
+ this.tokens = lexer.tokenize();
32
+ this.current = 0;
33
+ const ast = this.parseExpression(PRECEDENCE.LOWEST);
34
+ if (!this.isAtEnd()) {
35
+ const token = this.peek();
36
+ throw new errors_1.UnexpectedTokenError(String(token.value), ['end of expression'], token.position);
37
+ }
38
+ return ast;
39
+ }
40
+ parseExpression(precedence) {
41
+ let left = this.parsePrefixExpression();
42
+ while (!this.isAtEnd() && precedence < this.getPrecedence()) {
43
+ left = this.parseInfixExpression(left);
44
+ }
45
+ return left;
46
+ }
47
+ parsePrefixExpression() {
48
+ const token = this.peek();
49
+ switch (token.type) {
50
+ case types_1.TokenType.NUMBER:
51
+ return this.parseNumber();
52
+ case types_1.TokenType.STRING:
53
+ return this.parseString();
54
+ case types_1.TokenType.BOOLEAN:
55
+ return this.parseBoolean();
56
+ case types_1.TokenType.NULL:
57
+ return this.parseNull();
58
+ case types_1.TokenType.VARIABLE:
59
+ return this.parseVariable();
60
+ case types_1.TokenType.CONTEXT_VAR:
61
+ return this.parseContextVariable();
62
+ case types_1.TokenType.IDENTIFIER:
63
+ return this.parseIdentifierOrFunctionCall();
64
+ case types_1.TokenType.LPAREN:
65
+ return this.parseGroupedExpression();
66
+ case types_1.TokenType.LBRACKET:
67
+ return this.parseArrayLiteral();
68
+ case types_1.TokenType.MINUS:
69
+ case types_1.TokenType.NOT:
70
+ return this.parseUnaryExpression();
71
+ default:
72
+ throw new errors_1.UnexpectedTokenError(String(token.value), ['number', 'string', 'boolean', 'null', 'variable', 'identifier', '(', '[', '-', '!'], token.position);
73
+ }
74
+ }
75
+ parseInfixExpression(left) {
76
+ const token = this.peek();
77
+ switch (token.type) {
78
+ case types_1.TokenType.PLUS:
79
+ case types_1.TokenType.MINUS:
80
+ case types_1.TokenType.MULTIPLY:
81
+ case types_1.TokenType.DIVIDE:
82
+ case types_1.TokenType.MODULO:
83
+ case types_1.TokenType.POWER:
84
+ case types_1.TokenType.EQ:
85
+ case types_1.TokenType.NEQ:
86
+ case types_1.TokenType.LT:
87
+ case types_1.TokenType.GT:
88
+ case types_1.TokenType.LTE:
89
+ case types_1.TokenType.GTE:
90
+ case types_1.TokenType.AND:
91
+ case types_1.TokenType.OR:
92
+ return this.parseBinaryExpression(left);
93
+ case types_1.TokenType.QUESTION:
94
+ return this.parseTernaryExpression(left);
95
+ case types_1.TokenType.DOT:
96
+ return this.parseMemberAccess(left);
97
+ case types_1.TokenType.LBRACKET:
98
+ return this.parseIndexAccess(left);
99
+ case types_1.TokenType.LPAREN:
100
+ // This handles the case where we have an identifier followed by (
101
+ // But actually this should be handled in parseIdentifierOrFunctionCall
102
+ return left;
103
+ default:
104
+ return left;
105
+ }
106
+ }
107
+ parseNumber() {
108
+ const token = this.advance();
109
+ const value = token.value;
110
+ if (typeof value === 'number') {
111
+ // It's a float
112
+ return {
113
+ type: 'NumberLiteral',
114
+ value: value,
115
+ };
116
+ }
117
+ else {
118
+ // It's a decimal (stored as string)
119
+ return {
120
+ type: 'DecimalLiteral',
121
+ value: String(value),
122
+ raw: String(value),
123
+ };
124
+ }
125
+ }
126
+ parseString() {
127
+ const token = this.advance();
128
+ return {
129
+ type: 'StringLiteral',
130
+ value: String(token.value),
131
+ };
132
+ }
133
+ parseBoolean() {
134
+ const token = this.advance();
135
+ return {
136
+ type: 'BooleanLiteral',
137
+ value: token.value === true,
138
+ };
139
+ }
140
+ parseNull() {
141
+ this.advance();
142
+ return {
143
+ type: 'NullLiteral',
144
+ };
145
+ }
146
+ parseVariable() {
147
+ const token = this.advance();
148
+ return {
149
+ type: 'VariableReference',
150
+ prefix: '$',
151
+ name: String(token.value),
152
+ };
153
+ }
154
+ parseContextVariable() {
155
+ const token = this.advance();
156
+ return {
157
+ type: 'VariableReference',
158
+ prefix: '@',
159
+ name: String(token.value),
160
+ };
161
+ }
162
+ parseIdentifierOrFunctionCall() {
163
+ const token = this.advance();
164
+ const name = String(token.value);
165
+ // Check if it's a function call
166
+ if (this.peek().type === types_1.TokenType.LPAREN) {
167
+ return this.parseFunctionCall(name);
168
+ }
169
+ // Otherwise, check for keywords AND/OR/NOT used as standalone identifiers
170
+ const upperName = name.toUpperCase();
171
+ if (upperName === 'AND' || upperName === 'OR') {
172
+ // These should be handled as operators, but if we reach here,
173
+ // it means they were used in an invalid context
174
+ throw new errors_1.SyntaxError(`'${name}' cannot be used as an identifier`, token.position, token.line, token.column, this.expression);
175
+ }
176
+ // It's a bare identifier - could be a variable without prefix
177
+ // For now, treat it as an error - require explicit prefix
178
+ throw new errors_1.SyntaxError(`Unknown identifier '${name}'. Variables must be prefixed with $ or @`, token.position, token.line, token.column, this.expression);
179
+ }
180
+ parseFunctionCall(name) {
181
+ this.advance(); // consume '('
182
+ const args = [];
183
+ if (this.peek().type !== types_1.TokenType.RPAREN) {
184
+ do {
185
+ if (this.peek().type === types_1.TokenType.COMMA) {
186
+ this.advance();
187
+ }
188
+ args.push(this.parseExpression(PRECEDENCE.LOWEST));
189
+ } while (this.peek().type === types_1.TokenType.COMMA);
190
+ }
191
+ this.expect(types_1.TokenType.RPAREN, ')');
192
+ return {
193
+ type: 'FunctionCall',
194
+ name: name.toUpperCase(), // Functions are case-insensitive
195
+ arguments: args,
196
+ };
197
+ }
198
+ parseGroupedExpression() {
199
+ this.advance(); // consume '('
200
+ const expr = this.parseExpression(PRECEDENCE.LOWEST);
201
+ this.expect(types_1.TokenType.RPAREN, ')');
202
+ return expr;
203
+ }
204
+ parseArrayLiteral() {
205
+ this.advance(); // consume '['
206
+ const elements = [];
207
+ if (this.peek().type !== types_1.TokenType.RBRACKET) {
208
+ do {
209
+ if (this.peek().type === types_1.TokenType.COMMA) {
210
+ this.advance();
211
+ }
212
+ elements.push(this.parseExpression(PRECEDENCE.LOWEST));
213
+ } while (this.peek().type === types_1.TokenType.COMMA);
214
+ }
215
+ this.expect(types_1.TokenType.RBRACKET, ']');
216
+ return {
217
+ type: 'ArrayLiteral',
218
+ elements,
219
+ };
220
+ }
221
+ parseUnaryExpression() {
222
+ const token = this.advance();
223
+ const operator = this.getOperatorSymbol(token.type);
224
+ const operand = this.parseExpression(PRECEDENCE.UNARY);
225
+ return {
226
+ type: 'UnaryOperation',
227
+ operator,
228
+ operand,
229
+ };
230
+ }
231
+ parseBinaryExpression(left) {
232
+ const token = this.advance();
233
+ const operator = this.getOperatorSymbol(token.type);
234
+ const precedence = this.getTokenPrecedence(token.type);
235
+ // Right associativity for power operator
236
+ const nextPrecedence = token.type === types_1.TokenType.POWER ? precedence - 1 : precedence;
237
+ const right = this.parseExpression(nextPrecedence);
238
+ return {
239
+ type: 'BinaryOperation',
240
+ operator,
241
+ left,
242
+ right,
243
+ };
244
+ }
245
+ parseTernaryExpression(condition) {
246
+ this.advance(); // consume '?'
247
+ const consequent = this.parseExpression(PRECEDENCE.LOWEST);
248
+ this.expect(types_1.TokenType.COLON, ':');
249
+ const alternate = this.parseExpression(PRECEDENCE.TERNARY - 1);
250
+ return {
251
+ type: 'ConditionalExpression',
252
+ condition,
253
+ consequent,
254
+ alternate,
255
+ };
256
+ }
257
+ parseMemberAccess(object) {
258
+ this.advance(); // consume '.'
259
+ const token = this.peek();
260
+ if (token.type !== types_1.TokenType.IDENTIFIER && token.type !== types_1.TokenType.VARIABLE) {
261
+ throw new errors_1.UnexpectedTokenError(String(token.value), ['identifier'], token.position);
262
+ }
263
+ this.advance();
264
+ const property = String(token.value);
265
+ const node = {
266
+ type: 'MemberAccess',
267
+ object,
268
+ property,
269
+ };
270
+ // Check for chained access
271
+ if (this.peek().type === types_1.TokenType.DOT) {
272
+ return this.parseMemberAccess(node);
273
+ }
274
+ if (this.peek().type === types_1.TokenType.LBRACKET) {
275
+ return this.parseIndexAccess(node);
276
+ }
277
+ return node;
278
+ }
279
+ parseIndexAccess(object) {
280
+ this.advance(); // consume '['
281
+ const index = this.parseExpression(PRECEDENCE.LOWEST);
282
+ this.expect(types_1.TokenType.RBRACKET, ']');
283
+ const node = {
284
+ type: 'IndexAccess',
285
+ object,
286
+ index,
287
+ };
288
+ // Check for chained access
289
+ if (this.peek().type === types_1.TokenType.DOT) {
290
+ return this.parseMemberAccess(node);
291
+ }
292
+ if (this.peek().type === types_1.TokenType.LBRACKET) {
293
+ return this.parseIndexAccess(node);
294
+ }
295
+ return node;
296
+ }
297
+ getPrecedence() {
298
+ return this.getTokenPrecedence(this.peek().type);
299
+ }
300
+ getTokenPrecedence(type) {
301
+ switch (type) {
302
+ case types_1.TokenType.OR:
303
+ return PRECEDENCE.OR;
304
+ case types_1.TokenType.AND:
305
+ return PRECEDENCE.AND;
306
+ case types_1.TokenType.EQ:
307
+ case types_1.TokenType.NEQ:
308
+ return PRECEDENCE.EQUALITY;
309
+ case types_1.TokenType.LT:
310
+ case types_1.TokenType.GT:
311
+ case types_1.TokenType.LTE:
312
+ case types_1.TokenType.GTE:
313
+ return PRECEDENCE.COMPARISON;
314
+ case types_1.TokenType.PLUS:
315
+ case types_1.TokenType.MINUS:
316
+ return PRECEDENCE.TERM;
317
+ case types_1.TokenType.MULTIPLY:
318
+ case types_1.TokenType.DIVIDE:
319
+ case types_1.TokenType.MODULO:
320
+ return PRECEDENCE.FACTOR;
321
+ case types_1.TokenType.POWER:
322
+ return PRECEDENCE.POWER;
323
+ case types_1.TokenType.DOT:
324
+ case types_1.TokenType.LBRACKET:
325
+ return PRECEDENCE.MEMBER;
326
+ case types_1.TokenType.QUESTION:
327
+ return PRECEDENCE.TERNARY;
328
+ default:
329
+ return PRECEDENCE.LOWEST;
330
+ }
331
+ }
332
+ getOperatorSymbol(type) {
333
+ switch (type) {
334
+ case types_1.TokenType.PLUS: return '+';
335
+ case types_1.TokenType.MINUS: return '-';
336
+ case types_1.TokenType.MULTIPLY: return '*';
337
+ case types_1.TokenType.DIVIDE: return '/';
338
+ case types_1.TokenType.MODULO: return '%';
339
+ case types_1.TokenType.POWER: return '^';
340
+ case types_1.TokenType.EQ: return '==';
341
+ case types_1.TokenType.NEQ: return '!=';
342
+ case types_1.TokenType.LT: return '<';
343
+ case types_1.TokenType.GT: return '>';
344
+ case types_1.TokenType.LTE: return '<=';
345
+ case types_1.TokenType.GTE: return '>=';
346
+ case types_1.TokenType.AND: return '&&';
347
+ case types_1.TokenType.OR: return '||';
348
+ case types_1.TokenType.NOT: return '!';
349
+ default: return '';
350
+ }
351
+ }
352
+ peek() {
353
+ return this.tokens[this.current];
354
+ }
355
+ advance() {
356
+ if (!this.isAtEnd()) {
357
+ this.current++;
358
+ }
359
+ return this.tokens[this.current - 1];
360
+ }
361
+ isAtEnd() {
362
+ return this.peek().type === types_1.TokenType.EOF;
363
+ }
364
+ expect(type, expected) {
365
+ if (this.peek().type === type) {
366
+ return this.advance();
367
+ }
368
+ throw new errors_1.UnexpectedTokenError(String(this.peek().value), [expected], this.peek().position);
369
+ }
370
+ }
371
+ exports.Parser = Parser;
372
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parser.js","sourceRoot":"","sources":["../src/parser.ts"],"names":[],"mappings":"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","sourcesContent":["import { Lexer } from './lexer';\nimport { Token, TokenType, ASTNode } from './types';\nimport { SyntaxError, UnexpectedTokenError } from './errors';\n\n// Operator precedence levels (higher = tighter binding)\nconst PRECEDENCE = {\n  LOWEST: 1,\n  TERNARY: 2,     // ? :\n  OR: 3,          // || OR\n  AND: 4,         // && AND\n  EQUALITY: 5,    // == !=\n  COMPARISON: 6,  // < > <= >=\n  TERM: 7,        // + -\n  FACTOR: 8,      // * / %\n  POWER: 9,       // ^\n  UNARY: 10,      // - ! NOT\n  CALL: 11,       // function calls\n  MEMBER: 12,     // . []\n};\n\nexport class Parser {\n  private tokens: Token[] = [];\n  private current: number = 0;\n  private expression: string = '';\n\n  parse(expression: string): ASTNode {\n    this.expression = expression;\n    const lexer = new Lexer(expression);\n    this.tokens = lexer.tokenize();\n    this.current = 0;\n\n    const ast = this.parseExpression(PRECEDENCE.LOWEST);\n\n    if (!this.isAtEnd()) {\n      const token = this.peek();\n      throw new UnexpectedTokenError(\n        String(token.value),\n        ['end of expression'],\n        token.position\n      );\n    }\n\n    return ast;\n  }\n\n  private parseExpression(precedence: number): ASTNode {\n    let left = this.parsePrefixExpression();\n\n    while (!this.isAtEnd() && precedence < this.getPrecedence()) {\n      left = this.parseInfixExpression(left);\n    }\n\n    return left;\n  }\n\n  private parsePrefixExpression(): ASTNode {\n    const token = this.peek();\n\n    switch (token.type) {\n      case TokenType.NUMBER:\n        return this.parseNumber();\n      case TokenType.STRING:\n        return this.parseString();\n      case TokenType.BOOLEAN:\n        return this.parseBoolean();\n      case TokenType.NULL:\n        return this.parseNull();\n      case TokenType.VARIABLE:\n        return this.parseVariable();\n      case TokenType.CONTEXT_VAR:\n        return this.parseContextVariable();\n      case TokenType.IDENTIFIER:\n        return this.parseIdentifierOrFunctionCall();\n      case TokenType.LPAREN:\n        return this.parseGroupedExpression();\n      case TokenType.LBRACKET:\n        return this.parseArrayLiteral();\n      case TokenType.MINUS:\n      case TokenType.NOT:\n        return this.parseUnaryExpression();\n      default:\n        throw new UnexpectedTokenError(\n          String(token.value),\n          ['number', 'string', 'boolean', 'null', 'variable', 'identifier', '(', '[', '-', '!'],\n          token.position\n        );\n    }\n  }\n\n  private parseInfixExpression(left: ASTNode): ASTNode {\n    const token = this.peek();\n\n    switch (token.type) {\n      case TokenType.PLUS:\n      case TokenType.MINUS:\n      case TokenType.MULTIPLY:\n      case TokenType.DIVIDE:\n      case TokenType.MODULO:\n      case TokenType.POWER:\n      case TokenType.EQ:\n      case TokenType.NEQ:\n      case TokenType.LT:\n      case TokenType.GT:\n      case TokenType.LTE:\n      case TokenType.GTE:\n      case TokenType.AND:\n      case TokenType.OR:\n        return this.parseBinaryExpression(left);\n      case TokenType.QUESTION:\n        return this.parseTernaryExpression(left);\n      case TokenType.DOT:\n        return this.parseMemberAccess(left);\n      case TokenType.LBRACKET:\n        return this.parseIndexAccess(left);\n      case TokenType.LPAREN:\n        // This handles the case where we have an identifier followed by (\n        // But actually this should be handled in parseIdentifierOrFunctionCall\n        return left;\n      default:\n        return left;\n    }\n  }\n\n  private parseNumber(): ASTNode {\n    const token = this.advance();\n    const value = token.value;\n\n    if (typeof value === 'number') {\n      // It's a float\n      return {\n        type: 'NumberLiteral',\n        value: value,\n      };\n    } else {\n      // It's a decimal (stored as string)\n      return {\n        type: 'DecimalLiteral',\n        value: String(value),\n        raw: String(value),\n      };\n    }\n  }\n\n  private parseString(): ASTNode {\n    const token = this.advance();\n    return {\n      type: 'StringLiteral',\n      value: String(token.value),\n    };\n  }\n\n  private parseBoolean(): ASTNode {\n    const token = this.advance();\n    return {\n      type: 'BooleanLiteral',\n      value: token.value === true,\n    };\n  }\n\n  private parseNull(): ASTNode {\n    this.advance();\n    return {\n      type: 'NullLiteral',\n    };\n  }\n\n  private parseVariable(): ASTNode {\n    const token = this.advance();\n    return {\n      type: 'VariableReference',\n      prefix: '$',\n      name: String(token.value),\n    };\n  }\n\n  private parseContextVariable(): ASTNode {\n    const token = this.advance();\n    return {\n      type: 'VariableReference',\n      prefix: '@',\n      name: String(token.value),\n    };\n  }\n\n  private parseIdentifierOrFunctionCall(): ASTNode {\n    const token = this.advance();\n    const name = String(token.value);\n\n    // Check if it's a function call\n    if (this.peek().type === TokenType.LPAREN) {\n      return this.parseFunctionCall(name);\n    }\n\n    // Otherwise, check for keywords AND/OR/NOT used as standalone identifiers\n    const upperName = name.toUpperCase();\n    if (upperName === 'AND' || upperName === 'OR') {\n      // These should be handled as operators, but if we reach here,\n      // it means they were used in an invalid context\n      throw new SyntaxError(\n        `'${name}' cannot be used as an identifier`,\n        token.position,\n        token.line,\n        token.column,\n        this.expression\n      );\n    }\n\n    // It's a bare identifier - could be a variable without prefix\n    // For now, treat it as an error - require explicit prefix\n    throw new SyntaxError(\n      `Unknown identifier '${name}'. Variables must be prefixed with $ or @`,\n      token.position,\n      token.line,\n      token.column,\n      this.expression\n    );\n  }\n\n  private parseFunctionCall(name: string): ASTNode {\n    this.advance(); // consume '('\n    const args: ASTNode[] = [];\n\n    if (this.peek().type !== TokenType.RPAREN) {\n      do {\n        if (this.peek().type === TokenType.COMMA) {\n          this.advance();\n        }\n        args.push(this.parseExpression(PRECEDENCE.LOWEST));\n      } while (this.peek().type === TokenType.COMMA);\n    }\n\n    this.expect(TokenType.RPAREN, ')');\n\n    return {\n      type: 'FunctionCall',\n      name: name.toUpperCase(), // Functions are case-insensitive\n      arguments: args,\n    };\n  }\n\n  private parseGroupedExpression(): ASTNode {\n    this.advance(); // consume '('\n    const expr = this.parseExpression(PRECEDENCE.LOWEST);\n    this.expect(TokenType.RPAREN, ')');\n    return expr;\n  }\n\n  private parseArrayLiteral(): ASTNode {\n    this.advance(); // consume '['\n    const elements: ASTNode[] = [];\n\n    if (this.peek().type !== TokenType.RBRACKET) {\n      do {\n        if (this.peek().type === TokenType.COMMA) {\n          this.advance();\n        }\n        elements.push(this.parseExpression(PRECEDENCE.LOWEST));\n      } while (this.peek().type === TokenType.COMMA);\n    }\n\n    this.expect(TokenType.RBRACKET, ']');\n\n    return {\n      type: 'ArrayLiteral',\n      elements,\n    };\n  }\n\n  private parseUnaryExpression(): ASTNode {\n    const token = this.advance();\n    const operator = this.getOperatorSymbol(token.type);\n    const operand = this.parseExpression(PRECEDENCE.UNARY);\n\n    return {\n      type: 'UnaryOperation',\n      operator,\n      operand,\n    };\n  }\n\n  private parseBinaryExpression(left: ASTNode): ASTNode {\n    const token = this.advance();\n    const operator = this.getOperatorSymbol(token.type);\n    const precedence = this.getTokenPrecedence(token.type);\n\n    // Right associativity for power operator\n    const nextPrecedence = token.type === TokenType.POWER ? precedence - 1 : precedence;\n    const right = this.parseExpression(nextPrecedence);\n\n    return {\n      type: 'BinaryOperation',\n      operator,\n      left,\n      right,\n    };\n  }\n\n  private parseTernaryExpression(condition: ASTNode): ASTNode {\n    this.advance(); // consume '?'\n    const consequent = this.parseExpression(PRECEDENCE.LOWEST);\n    this.expect(TokenType.COLON, ':');\n    const alternate = this.parseExpression(PRECEDENCE.TERNARY - 1);\n\n    return {\n      type: 'ConditionalExpression',\n      condition,\n      consequent,\n      alternate,\n    };\n  }\n\n  private parseMemberAccess(object: ASTNode): ASTNode {\n    this.advance(); // consume '.'\n    const token = this.peek();\n\n    if (token.type !== TokenType.IDENTIFIER && token.type !== TokenType.VARIABLE) {\n      throw new UnexpectedTokenError(\n        String(token.value),\n        ['identifier'],\n        token.position\n      );\n    }\n\n    this.advance();\n    const property = String(token.value);\n\n    const node: ASTNode = {\n      type: 'MemberAccess',\n      object,\n      property,\n    };\n\n    // Check for chained access\n    if (this.peek().type === TokenType.DOT) {\n      return this.parseMemberAccess(node);\n    }\n    if (this.peek().type === TokenType.LBRACKET) {\n      return this.parseIndexAccess(node);\n    }\n\n    return node;\n  }\n\n  private parseIndexAccess(object: ASTNode): ASTNode {\n    this.advance(); // consume '['\n    const index = this.parseExpression(PRECEDENCE.LOWEST);\n    this.expect(TokenType.RBRACKET, ']');\n\n    const node: ASTNode = {\n      type: 'IndexAccess',\n      object,\n      index,\n    };\n\n    // Check for chained access\n    if (this.peek().type === TokenType.DOT) {\n      return this.parseMemberAccess(node);\n    }\n    if (this.peek().type === TokenType.LBRACKET) {\n      return this.parseIndexAccess(node);\n    }\n\n    return node;\n  }\n\n  private getPrecedence(): number {\n    return this.getTokenPrecedence(this.peek().type);\n  }\n\n  private getTokenPrecedence(type: TokenType): number {\n    switch (type) {\n      case TokenType.OR:\n        return PRECEDENCE.OR;\n      case TokenType.AND:\n        return PRECEDENCE.AND;\n      case TokenType.EQ:\n      case TokenType.NEQ:\n        return PRECEDENCE.EQUALITY;\n      case TokenType.LT:\n      case TokenType.GT:\n      case TokenType.LTE:\n      case TokenType.GTE:\n        return PRECEDENCE.COMPARISON;\n      case TokenType.PLUS:\n      case TokenType.MINUS:\n        return PRECEDENCE.TERM;\n      case TokenType.MULTIPLY:\n      case TokenType.DIVIDE:\n      case TokenType.MODULO:\n        return PRECEDENCE.FACTOR;\n      case TokenType.POWER:\n        return PRECEDENCE.POWER;\n      case TokenType.DOT:\n      case TokenType.LBRACKET:\n        return PRECEDENCE.MEMBER;\n      case TokenType.QUESTION:\n        return PRECEDENCE.TERNARY;\n      default:\n        return PRECEDENCE.LOWEST;\n    }\n  }\n\n  private getOperatorSymbol(type: TokenType): string {\n    switch (type) {\n      case TokenType.PLUS: return '+';\n      case TokenType.MINUS: return '-';\n      case TokenType.MULTIPLY: return '*';\n      case TokenType.DIVIDE: return '/';\n      case TokenType.MODULO: return '%';\n      case TokenType.POWER: return '^';\n      case TokenType.EQ: return '==';\n      case TokenType.NEQ: return '!=';\n      case TokenType.LT: return '<';\n      case TokenType.GT: return '>';\n      case TokenType.LTE: return '<=';\n      case TokenType.GTE: return '>=';\n      case TokenType.AND: return '&&';\n      case TokenType.OR: return '||';\n      case TokenType.NOT: return '!';\n      default: return '';\n    }\n  }\n\n  private peek(): Token {\n    return this.tokens[this.current];\n  }\n\n  private advance(): Token {\n    if (!this.isAtEnd()) {\n      this.current++;\n    }\n    return this.tokens[this.current - 1];\n  }\n\n  private isAtEnd(): boolean {\n    return this.peek().type === TokenType.EOF;\n  }\n\n  private expect(type: TokenType, expected: string): Token {\n    if (this.peek().type === type) {\n      return this.advance();\n    }\n    throw new UnexpectedTokenError(\n      String(this.peek().value),\n      [expected],\n      this.peek().position\n    );\n  }\n}\n"]}
@@ -0,0 +1,228 @@
1
+ import { Decimal } from 'decimal.js';
2
+ export type ValueType = 'number' | 'decimal' | 'string' | 'boolean' | 'array' | 'object' | 'null' | 'any';
3
+ export type FormulaValue = Decimal | number | string | boolean | null | FormulaValue[] | {
4
+ [key: string]: FormulaValue;
5
+ };
6
+ export declare enum DecimalRoundingMode {
7
+ CEIL = "CEIL",
8
+ FLOOR = "FLOOR",
9
+ DOWN = "DOWN",
10
+ UP = "UP",
11
+ HALF_UP = "HALF_UP",
12
+ HALF_DOWN = "HALF_DOWN",
13
+ HALF_EVEN = "HALF_EVEN",
14
+ HALF_ODD = "HALF_ODD"
15
+ }
16
+ export interface DecimalConfig {
17
+ precision?: number;
18
+ roundingMode?: DecimalRoundingMode;
19
+ divisionScale?: number;
20
+ preserveTrailingZeros?: boolean;
21
+ autoConvertFloats?: boolean;
22
+ maxExponent?: number;
23
+ minExponent?: number;
24
+ }
25
+ export interface RoundingConfig {
26
+ mode: 'HALF_UP' | 'HALF_DOWN' | 'FLOOR' | 'CEIL' | 'NONE';
27
+ precision: number;
28
+ }
29
+ export interface ErrorBehavior {
30
+ type: 'THROW' | 'NULL' | 'ZERO' | 'DEFAULT' | 'SKIP';
31
+ defaultValue?: unknown;
32
+ }
33
+ export interface FormulaDefinition {
34
+ id: string;
35
+ expression: string;
36
+ dependencies?: string[];
37
+ onError?: ErrorBehavior;
38
+ defaultValue?: unknown;
39
+ rounding?: RoundingConfig;
40
+ metadata?: Record<string, unknown>;
41
+ }
42
+ export interface EvaluationContext {
43
+ variables: Record<string, unknown>;
44
+ collections?: Record<string, unknown[]>;
45
+ extra?: Record<string, unknown>;
46
+ }
47
+ export interface OperatorDefinition {
48
+ symbol: string;
49
+ precedence: number;
50
+ associativity: 'left' | 'right';
51
+ handler: (left: unknown, right: unknown) => unknown;
52
+ }
53
+ export interface SecurityConfig {
54
+ maxExpressionLength?: number;
55
+ maxRecursionDepth?: number;
56
+ maxIterations?: number;
57
+ maxExecutionTime?: number;
58
+ allowedFunctions?: string[];
59
+ blockedFunctions?: string[];
60
+ }
61
+ export interface FormulaEngineConfig {
62
+ enableCache?: boolean;
63
+ maxCacheSize?: number;
64
+ defaultErrorBehavior?: ErrorBehavior;
65
+ defaultRounding?: RoundingConfig;
66
+ variablePrefix?: string;
67
+ contextPrefix?: string;
68
+ strictMode?: boolean;
69
+ operators?: OperatorDefinition[];
70
+ functions?: FunctionDefinition[];
71
+ decimal?: DecimalConfig;
72
+ security?: SecurityConfig;
73
+ }
74
+ export interface DecimalLiteral {
75
+ type: 'DecimalLiteral';
76
+ value: string;
77
+ raw: string;
78
+ }
79
+ export interface NumberLiteral {
80
+ type: 'NumberLiteral';
81
+ value: number;
82
+ }
83
+ export interface StringLiteral {
84
+ type: 'StringLiteral';
85
+ value: string;
86
+ }
87
+ export interface BooleanLiteral {
88
+ type: 'BooleanLiteral';
89
+ value: boolean;
90
+ }
91
+ export interface NullLiteral {
92
+ type: 'NullLiteral';
93
+ }
94
+ export interface ArrayLiteral {
95
+ type: 'ArrayLiteral';
96
+ elements: ASTNode[];
97
+ }
98
+ export interface VariableReference {
99
+ type: 'VariableReference';
100
+ prefix: '$' | '@';
101
+ name: string;
102
+ }
103
+ export interface BinaryOperation {
104
+ type: 'BinaryOperation';
105
+ operator: string;
106
+ left: ASTNode;
107
+ right: ASTNode;
108
+ }
109
+ export interface UnaryOperation {
110
+ type: 'UnaryOperation';
111
+ operator: string;
112
+ operand: ASTNode;
113
+ }
114
+ export interface ConditionalExpression {
115
+ type: 'ConditionalExpression';
116
+ condition: ASTNode;
117
+ consequent: ASTNode;
118
+ alternate: ASTNode;
119
+ }
120
+ export interface FunctionCall {
121
+ type: 'FunctionCall';
122
+ name: string;
123
+ arguments: ASTNode[];
124
+ }
125
+ export interface MemberAccess {
126
+ type: 'MemberAccess';
127
+ object: ASTNode;
128
+ property: string;
129
+ }
130
+ export interface IndexAccess {
131
+ type: 'IndexAccess';
132
+ object: ASTNode;
133
+ index: ASTNode;
134
+ }
135
+ export type ASTNode = DecimalLiteral | NumberLiteral | StringLiteral | BooleanLiteral | NullLiteral | ArrayLiteral | VariableReference | BinaryOperation | UnaryOperation | ConditionalExpression | FunctionCall | MemberAccess | IndexAccess;
136
+ export interface ArgumentType {
137
+ name: string;
138
+ type: ValueType;
139
+ required: boolean;
140
+ default?: unknown;
141
+ }
142
+ export type FunctionImplementation = (args: unknown[], context: EvaluationContext, engine: unknown) => unknown;
143
+ export interface FunctionDefinition {
144
+ name: string;
145
+ minArgs: number;
146
+ maxArgs: number;
147
+ argTypes?: ArgumentType[];
148
+ returnType: ValueType;
149
+ implementation: FunctionImplementation;
150
+ description?: string;
151
+ }
152
+ export interface DependencyGraph {
153
+ nodes: Set<string>;
154
+ edges: Map<string, Set<string>>;
155
+ hasCycles(): boolean;
156
+ getRoots(): Set<string>;
157
+ getDependents(nodeId: string): Set<string>;
158
+ getDependencies(nodeId: string): Set<string>;
159
+ getTransitiveDependencies(nodeId: string): Set<string>;
160
+ topologicalSort(): string[];
161
+ }
162
+ export interface EvaluationResult {
163
+ value: unknown;
164
+ success: boolean;
165
+ error?: Error;
166
+ executionTimeMs: number;
167
+ accessedVariables: Set<string>;
168
+ }
169
+ export interface EvaluationResultSet {
170
+ results: Map<string, EvaluationResult>;
171
+ success: boolean;
172
+ errors: Error[];
173
+ totalExecutionTimeMs: number;
174
+ evaluationOrder: string[];
175
+ }
176
+ export interface ValidationResult {
177
+ valid: boolean;
178
+ errors: Error[];
179
+ warnings: string[];
180
+ dependencyGraph: DependencyGraph;
181
+ evaluationOrder: string[];
182
+ }
183
+ export interface CacheStats {
184
+ size: number;
185
+ hits: number;
186
+ misses: number;
187
+ hitRate: number;
188
+ }
189
+ export declare enum TokenType {
190
+ NUMBER = "NUMBER",
191
+ STRING = "STRING",
192
+ BOOLEAN = "BOOLEAN",
193
+ NULL = "NULL",
194
+ IDENTIFIER = "IDENTIFIER",
195
+ VARIABLE = "VARIABLE",
196
+ CONTEXT_VAR = "CONTEXT_VAR",
197
+ PLUS = "PLUS",
198
+ MINUS = "MINUS",
199
+ MULTIPLY = "MULTIPLY",
200
+ DIVIDE = "DIVIDE",
201
+ MODULO = "MODULO",
202
+ POWER = "POWER",
203
+ EQ = "EQ",
204
+ NEQ = "NEQ",
205
+ LT = "LT",
206
+ GT = "GT",
207
+ LTE = "LTE",
208
+ GTE = "GTE",
209
+ AND = "AND",
210
+ OR = "OR",
211
+ NOT = "NOT",
212
+ LPAREN = "LPAREN",
213
+ RPAREN = "RPAREN",
214
+ LBRACKET = "LBRACKET",
215
+ RBRACKET = "RBRACKET",
216
+ COMMA = "COMMA",
217
+ DOT = "DOT",
218
+ QUESTION = "QUESTION",
219
+ COLON = "COLON",
220
+ EOF = "EOF"
221
+ }
222
+ export interface Token {
223
+ type: TokenType;
224
+ value: string | number | boolean | null;
225
+ position: number;
226
+ line: number;
227
+ column: number;
228
+ }