@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,355 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.Decimal = exports.decimalUtils = exports.DecimalUtils = void 0;
7
+ const decimal_js_1 = __importDefault(require("decimal.js"));
8
+ exports.Decimal = decimal_js_1.default;
9
+ const types_1 = require("./types");
10
+ const errors_1 = require("./errors");
11
+ // Map our rounding modes to decimal.js rounding modes
12
+ const ROUNDING_MODE_MAP = {
13
+ [types_1.DecimalRoundingMode.CEIL]: decimal_js_1.default.ROUND_CEIL,
14
+ [types_1.DecimalRoundingMode.FLOOR]: decimal_js_1.default.ROUND_FLOOR,
15
+ [types_1.DecimalRoundingMode.DOWN]: decimal_js_1.default.ROUND_DOWN,
16
+ [types_1.DecimalRoundingMode.UP]: decimal_js_1.default.ROUND_UP,
17
+ [types_1.DecimalRoundingMode.HALF_UP]: decimal_js_1.default.ROUND_HALF_UP,
18
+ [types_1.DecimalRoundingMode.HALF_DOWN]: decimal_js_1.default.ROUND_HALF_DOWN,
19
+ [types_1.DecimalRoundingMode.HALF_EVEN]: decimal_js_1.default.ROUND_HALF_EVEN,
20
+ [types_1.DecimalRoundingMode.HALF_ODD]: decimal_js_1.default.ROUND_HALF_CEIL, // decimal.js doesn't have HALF_ODD, use HALF_CEIL as approximation
21
+ };
22
+ const DEFAULT_CONFIG = {
23
+ precision: 20,
24
+ roundingMode: types_1.DecimalRoundingMode.HALF_UP,
25
+ divisionScale: 10,
26
+ maxExponent: 1000,
27
+ minExponent: -1000,
28
+ };
29
+ class DecimalUtils {
30
+ constructor(config) {
31
+ this.config = {
32
+ ...DEFAULT_CONFIG,
33
+ ...config,
34
+ };
35
+ // Configure decimal.js globally
36
+ decimal_js_1.default.set({
37
+ precision: this.config.precision,
38
+ rounding: ROUNDING_MODE_MAP[this.config.roundingMode],
39
+ toExpNeg: this.config.minExponent,
40
+ toExpPos: this.config.maxExponent,
41
+ });
42
+ }
43
+ /**
44
+ * Create a Decimal from various input types
45
+ */
46
+ from(value) {
47
+ if (value instanceof decimal_js_1.default) {
48
+ return value;
49
+ }
50
+ try {
51
+ if (typeof value === 'bigint') {
52
+ return new decimal_js_1.default(value.toString());
53
+ }
54
+ return new decimal_js_1.default(value);
55
+ }
56
+ catch {
57
+ throw new errors_1.InvalidDecimalError(String(value));
58
+ }
59
+ }
60
+ /**
61
+ * Check if a value is a Decimal
62
+ */
63
+ isDecimal(value) {
64
+ return value instanceof decimal_js_1.default;
65
+ }
66
+ /**
67
+ * Convert a value to Decimal if it's numeric
68
+ */
69
+ toDecimal(value) {
70
+ if (value instanceof decimal_js_1.default) {
71
+ return value;
72
+ }
73
+ if (typeof value === 'number' && !isNaN(value) && isFinite(value)) {
74
+ return new decimal_js_1.default(value);
75
+ }
76
+ if (typeof value === 'string') {
77
+ const trimmed = value.trim();
78
+ if (/^-?\d+\.?\d*$/.test(trimmed) || /^-?\d*\.?\d+$/.test(trimmed)) {
79
+ try {
80
+ return new decimal_js_1.default(trimmed);
81
+ }
82
+ catch {
83
+ return value;
84
+ }
85
+ }
86
+ }
87
+ if (typeof value === 'bigint') {
88
+ return new decimal_js_1.default(value.toString());
89
+ }
90
+ return value;
91
+ }
92
+ /**
93
+ * Addition
94
+ */
95
+ add(a, b) {
96
+ return this.from(a).plus(this.from(b));
97
+ }
98
+ /**
99
+ * Subtraction
100
+ */
101
+ subtract(a, b) {
102
+ return this.from(a).minus(this.from(b));
103
+ }
104
+ /**
105
+ * Multiplication
106
+ */
107
+ multiply(a, b) {
108
+ return this.from(a).times(this.from(b));
109
+ }
110
+ /**
111
+ * Division with scale
112
+ */
113
+ divide(a, b, scale, roundingMode) {
114
+ const divisor = this.from(b);
115
+ if (divisor.isZero()) {
116
+ throw new errors_1.DecimalDivisionByZeroError();
117
+ }
118
+ const result = this.from(a).dividedBy(divisor);
119
+ if (scale !== undefined) {
120
+ const rounding = roundingMode
121
+ ? ROUNDING_MODE_MAP[roundingMode]
122
+ : ROUNDING_MODE_MAP[this.config.roundingMode];
123
+ return result.toDecimalPlaces(scale, rounding);
124
+ }
125
+ return result.toDecimalPlaces(this.config.divisionScale, ROUNDING_MODE_MAP[this.config.roundingMode]);
126
+ }
127
+ /**
128
+ * Modulo
129
+ */
130
+ modulo(a, b) {
131
+ const divisor = this.from(b);
132
+ if (divisor.isZero()) {
133
+ throw new errors_1.DecimalDivisionByZeroError();
134
+ }
135
+ return this.from(a).modulo(divisor);
136
+ }
137
+ /**
138
+ * Power
139
+ */
140
+ power(base, exponent) {
141
+ return this.from(base).pow(exponent);
142
+ }
143
+ /**
144
+ * Negation
145
+ */
146
+ negate(value) {
147
+ return this.from(value).negated();
148
+ }
149
+ /**
150
+ * Absolute value
151
+ */
152
+ abs(value) {
153
+ return this.from(value).absoluteValue();
154
+ }
155
+ /**
156
+ * Round to specified decimal places
157
+ */
158
+ round(value, scale, roundingMode) {
159
+ const rounding = roundingMode
160
+ ? ROUNDING_MODE_MAP[roundingMode]
161
+ : ROUNDING_MODE_MAP[this.config.roundingMode];
162
+ return this.from(value).toDecimalPlaces(scale, rounding);
163
+ }
164
+ /**
165
+ * Floor to specified decimal places
166
+ */
167
+ floor(value, scale = 0) {
168
+ return this.from(value).toDecimalPlaces(scale, decimal_js_1.default.ROUND_FLOOR);
169
+ }
170
+ /**
171
+ * Ceiling to specified decimal places
172
+ */
173
+ ceil(value, scale = 0) {
174
+ return this.from(value).toDecimalPlaces(scale, decimal_js_1.default.ROUND_CEIL);
175
+ }
176
+ /**
177
+ * Truncate to specified decimal places
178
+ */
179
+ truncate(value, scale = 0) {
180
+ return this.from(value).toDecimalPlaces(scale, decimal_js_1.default.ROUND_DOWN);
181
+ }
182
+ /**
183
+ * Square root
184
+ */
185
+ sqrt(value) {
186
+ return this.from(value).sqrt();
187
+ }
188
+ /**
189
+ * Natural logarithm
190
+ */
191
+ ln(value) {
192
+ return this.from(value).ln();
193
+ }
194
+ /**
195
+ * Base-10 logarithm
196
+ */
197
+ log10(value) {
198
+ return this.from(value).log(10);
199
+ }
200
+ /**
201
+ * Comparison: returns -1, 0, or 1
202
+ */
203
+ compare(a, b) {
204
+ const result = this.from(a).comparedTo(this.from(b));
205
+ return result;
206
+ }
207
+ /**
208
+ * Equality check
209
+ */
210
+ equals(a, b) {
211
+ return this.from(a).equals(this.from(b));
212
+ }
213
+ /**
214
+ * Greater than
215
+ */
216
+ greaterThan(a, b) {
217
+ return this.from(a).greaterThan(this.from(b));
218
+ }
219
+ /**
220
+ * Greater than or equal
221
+ */
222
+ greaterThanOrEqual(a, b) {
223
+ return this.from(a).greaterThanOrEqualTo(this.from(b));
224
+ }
225
+ /**
226
+ * Less than
227
+ */
228
+ lessThan(a, b) {
229
+ return this.from(a).lessThan(this.from(b));
230
+ }
231
+ /**
232
+ * Less than or equal
233
+ */
234
+ lessThanOrEqual(a, b) {
235
+ return this.from(a).lessThanOrEqualTo(this.from(b));
236
+ }
237
+ /**
238
+ * Check if zero
239
+ */
240
+ isZero(value) {
241
+ return this.from(value).isZero();
242
+ }
243
+ /**
244
+ * Check if positive
245
+ */
246
+ isPositive(value) {
247
+ return this.from(value).isPositive();
248
+ }
249
+ /**
250
+ * Check if negative
251
+ */
252
+ isNegative(value) {
253
+ return this.from(value).isNegative();
254
+ }
255
+ /**
256
+ * Check if integer
257
+ */
258
+ isInteger(value) {
259
+ return this.from(value).isInteger();
260
+ }
261
+ /**
262
+ * Get sign: -1, 0, or 1
263
+ */
264
+ sign(value) {
265
+ const d = this.from(value);
266
+ if (d.isZero())
267
+ return 0;
268
+ return d.isNegative() ? -1 : 1;
269
+ }
270
+ /**
271
+ * Get precision (total significant digits)
272
+ */
273
+ precision(value) {
274
+ return this.from(value).precision();
275
+ }
276
+ /**
277
+ * Get scale (decimal places)
278
+ */
279
+ scale(value) {
280
+ return this.from(value).decimalPlaces();
281
+ }
282
+ /**
283
+ * Convert to JavaScript number (may lose precision)
284
+ */
285
+ toNumber(value) {
286
+ return this.from(value).toNumber();
287
+ }
288
+ /**
289
+ * Convert to string
290
+ */
291
+ toString(value) {
292
+ return this.from(value).toString();
293
+ }
294
+ /**
295
+ * Convert to fixed decimal places string
296
+ */
297
+ toFixed(value, scale) {
298
+ return this.from(value).toFixed(scale);
299
+ }
300
+ /**
301
+ * Minimum of values
302
+ */
303
+ min(...values) {
304
+ if (values.length === 0) {
305
+ throw new Error('min requires at least one argument');
306
+ }
307
+ return decimal_js_1.default.min(...values.map(v => this.from(v)));
308
+ }
309
+ /**
310
+ * Maximum of values
311
+ */
312
+ max(...values) {
313
+ if (values.length === 0) {
314
+ throw new Error('max requires at least one argument');
315
+ }
316
+ return decimal_js_1.default.max(...values.map(v => this.from(v)));
317
+ }
318
+ /**
319
+ * Sum of values
320
+ */
321
+ sum(values) {
322
+ return values.reduce((acc, v) => acc.plus(this.from(v)), new decimal_js_1.default(0));
323
+ }
324
+ /**
325
+ * Average of values
326
+ */
327
+ avg(values) {
328
+ if (values.length === 0) {
329
+ throw new Error('avg requires at least one value');
330
+ }
331
+ return this.sum(values).dividedBy(values.length);
332
+ }
333
+ /**
334
+ * Product of values
335
+ */
336
+ product(values) {
337
+ return values.reduce((acc, v) => acc.times(this.from(v)), new decimal_js_1.default(1));
338
+ }
339
+ /**
340
+ * Create zero
341
+ */
342
+ zero() {
343
+ return new decimal_js_1.default(0);
344
+ }
345
+ /**
346
+ * Create one
347
+ */
348
+ one() {
349
+ return new decimal_js_1.default(1);
350
+ }
351
+ }
352
+ exports.DecimalUtils = DecimalUtils;
353
+ // Export a default instance
354
+ exports.decimalUtils = new DecimalUtils();
355
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decimal-utils.js","sourceRoot":"","sources":["../src/decimal-utils.ts"],"names":[],"mappings":";;;;;;AAAA,4DAAiC;AAuZxB,kBAvZF,oBAAO,CAuZE;AAtZhB,mCAA6D;AAC7D,qCAA2E;AAE3E,sDAAsD;AACtD,MAAM,iBAAiB,GAAkD;IACvE,CAAC,2BAAmB,CAAC,IAAI,CAAC,EAAE,oBAAO,CAAC,UAAU;IAC9C,CAAC,2BAAmB,CAAC,KAAK,CAAC,EAAE,oBAAO,CAAC,WAAW;IAChD,CAAC,2BAAmB,CAAC,IAAI,CAAC,EAAE,oBAAO,CAAC,UAAU;IAC9C,CAAC,2BAAmB,CAAC,EAAE,CAAC,EAAE,oBAAO,CAAC,QAAQ;IAC1C,CAAC,2BAAmB,CAAC,OAAO,CAAC,EAAE,oBAAO,CAAC,aAAa;IACpD,CAAC,2BAAmB,CAAC,SAAS,CAAC,EAAE,oBAAO,CAAC,eAAe;IACxD,CAAC,2BAAmB,CAAC,SAAS,CAAC,EAAE,oBAAO,CAAC,eAAe;IACxD,CAAC,2BAAmB,CAAC,QAAQ,CAAC,EAAE,oBAAO,CAAC,eAAe,EAAE,mEAAmE;CAC7H,CAAC;AAYF,MAAM,cAAc,GAAuB;IACzC,SAAS,EAAE,EAAE;IACb,YAAY,EAAE,2BAAmB,CAAC,OAAO;IACzC,aAAa,EAAE,EAAE;IACjB,WAAW,EAAE,IAAI;IACjB,WAAW,EAAE,CAAC,IAAI;CACnB,CAAC;AAEF,MAAa,YAAY;IAGvB,YAAY,MAA+B;QACzC,IAAI,CAAC,MAAM,GAAG;YACZ,GAAG,cAAc;YACjB,GAAG,MAAM;SACV,CAAC;QAEF,gCAAgC;QAChC,oBAAO,CAAC,GAAG,CAAC;YACV,SAAS,EAAE,IAAI,CAAC,MAAM,CAAC,SAAS;YAChC,QAAQ,EAAE,iBAAiB,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC;YACrD,QAAQ,EAAE,IAAI,CAAC,MAAM,CAAC,WAAW;YACjC,QAAQ,EAAE,IAAI,CAAC,MAAM,CAAC,WAAW;SAClC,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACH,IAAI,CAAC,KAAkB;QACrB,IAAI,KAAK,YAAY,oBAAO,EAAE,CAAC;YAC7B,OAAO,KAAK,CAAC;QACf,CAAC;QAED,IAAI,CAAC;YACH,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;gBAC9B,OAAO,IAAI,oBAAO,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC;YACvC,CAAC;YACD,OAAO,IAAI,oBAAO,CAAC,KAAK,CAAC,CAAC;QAC5B,CAAC;QAAC,MAAM,CAAC;YACP,MAAM,IAAI,4BAAmB,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;QAC/C,CAAC;IACH,CAAC;IAED;;OAEG;IACH,SAAS,CAAC,KAAc;QACtB,OAAO,KAAK,YAAY,oBAAO,CAAC;IAClC,CAAC;IAED;;OAEG;IACH,SAAS,CAAC,KAAc;QACtB,IAAI,KAAK,YAAY,oBAAO,EAAE,CAAC;YAC7B,OAAO,KAAK,CAAC;QACf,CAAC;QACD,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC;YAClE,OAAO,IAAI,oBAAO,CAAC,KAAK,CAAC,CAAC;QAC5B,CAAC;QACD,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;YAC9B,MAAM,OAAO,GAAG,KAAK,CAAC,IAAI,EAAE,CAAC;YAC7B,IAAI,eAAe,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,eAAe,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;gBACnE,IAAI,CAAC;oBACH,OAAO,IAAI,oBAAO,CAAC,OAAO,CAAC,CAAC;gBAC9B,CAAC;gBAAC,MAAM,CAAC;oBACP,OAAO,KAAK,CAAC;gBACf,CAAC;YACH,CAAC;QACH,CAAC;QACD,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;YAC9B,OAAO,IAAI,oBAAO,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC;QACvC,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;OAEG;IACH,GAAG,CAAC,CAAc,EAAE,CAAc;QAChC,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;IACzC,CAAC;IAED;;OAEG;IACH,QAAQ,CAAC,CAAc,EAAE,CAAc;QACrC,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;IAC1C,CAAC;IAED;;OAEG;IACH,QAAQ,CAAC,CAAc,EAAE,CAAc;QACrC,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;IAC1C,CAAC;IAED;;OAEG;IACH,MAAM,CAAC,CAAc,EAAE,CAAc,EAAE,KAAc,EAAE,YAAkC;QACvF,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAC7B,IAAI,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC;YACrB,MAAM,IAAI,mCAA0B,EAAE,CAAC;QACzC,CAAC;QAED,MAAM,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;QAE/C,IAAI,KAAK,KAAK,SAAS,EAAE,CAAC;YACxB,MAAM,QAAQ,GAAG,YAAY;gBAC3B,CAAC,CAAC,iBAAiB,CAAC,YAAY,CAAC;gBACjC,CAAC,CAAC,iBAAiB,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;YAChD,OAAO,MAAM,CAAC,eAAe,CAAC,KAAK,EAAE,QAAQ,CAAC,CAAC;QACjD,CAAC;QAED,OAAO,MAAM,CAAC,eAAe,CAAC,IAAI,CAAC,MAAM,CAAC,aAAa,EAAE,iBAAiB,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC;IACxG,CAAC;IAED;;OAEG;IACH,MAAM,CAAC,CAAc,EAAE,CAAc;QACnC,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAC7B,IAAI,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC;YACrB,MAAM,IAAI,mCAA0B,EAAE,CAAC;QACzC,CAAC;QACD,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IACtC,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,IAAiB,EAAE,QAAgB;QACvC,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IACvC,CAAC;IAED;;OAEG;IACH,MAAM,CAAC,KAAkB;QACvB,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,OAAO,EAAE,CAAC;IACpC,CAAC;IAED;;OAEG;IACH,GAAG,CAAC,KAAkB;QACpB,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,aAAa,EAAE,CAAC;IAC1C,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,KAAkB,EAAE,KAAa,EAAE,YAAkC;QACzE,MAAM,QAAQ,GAAG,YAAY;YAC3B,CAAC,CAAC,iBAAiB,CAAC,YAAY,CAAC;YACjC,CAAC,CAAC,iBAAiB,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;QAChD,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,eAAe,CAAC,KAAK,EAAE,QAAQ,CAAC,CAAC;IAC3D,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,KAAkB,EAAE,QAAgB,CAAC;QACzC,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,eAAe,CAAC,KAAK,EAAE,oBAAO,CAAC,WAAW,CAAC,CAAC;IACtE,CAAC;IAED;;OAEG;IACH,IAAI,CAAC,KAAkB,EAAE,QAAgB,CAAC;QACxC,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,eAAe,CAAC,KAAK,EAAE,oBAAO,CAAC,UAAU,CAAC,CAAC;IACrE,CAAC;IAED;;OAEG;IACH,QAAQ,CAAC,KAAkB,EAAE,QAAgB,CAAC;QAC5C,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,eAAe,CAAC,KAAK,EAAE,oBAAO,CAAC,UAAU,CAAC,CAAC;IACrE,CAAC;IAED;;OAEG;IACH,IAAI,CAAC,KAAkB;QACrB,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,IAAI,EAAE,CAAC;IACjC,CAAC;IAED;;OAEG;IACH,EAAE,CAAC,KAAkB;QACnB,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC;IAC/B,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,KAAkB;QACtB,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;IAClC,CAAC;IAED;;OAEG;IACH,OAAO,CAAC,CAAc,EAAE,CAAc;QACpC,MAAM,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;QACrD,OAAO,MAAoB,CAAC;IAC9B,CAAC;IAED;;OAEG;IACH,MAAM,CAAC,CAAc,EAAE,CAAc;QACnC,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;IAC3C,CAAC;IAED;;OAEG;IACH,WAAW,CAAC,CAAc,EAAE,CAAc;QACxC,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;IAChD,CAAC;IAED;;OAEG;IACH,kBAAkB,CAAC,CAAc,EAAE,CAAc;QAC/C,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,oBAAoB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;IACzD,CAAC;IAED;;OAEG;IACH,QAAQ,CAAC,CAAc,EAAE,CAAc;QACrC,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;IAC7C,CAAC;IAED;;OAEG;IACH,eAAe,CAAC,CAAc,EAAE,CAAc;QAC5C,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;IACtD,CAAC;IAED;;OAEG;IACH,MAAM,CAAC,KAAkB;QACvB,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,MAAM,EAAE,CAAC;IACnC,CAAC;IAED;;OAEG;IACH,UAAU,CAAC,KAAkB;QAC3B,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,UAAU,EAAE,CAAC;IACvC,CAAC;IAED;;OAEG;IACH,UAAU,CAAC,KAAkB;QAC3B,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,UAAU,EAAE,CAAC;IACvC,CAAC;IAED;;OAEG;IACH,SAAS,CAAC,KAAkB;QAC1B,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,SAAS,EAAE,CAAC;IACtC,CAAC;IAED;;OAEG;IACH,IAAI,CAAC,KAAkB;QACrB,MAAM,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC3B,IAAI,CAAC,CAAC,MAAM,EAAE;YAAE,OAAO,CAAC,CAAC;QACzB,OAAO,CAAC,CAAC,UAAU,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACjC,CAAC;IAED;;OAEG;IACH,SAAS,CAAC,KAAkB;QAC1B,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,SAAS,EAAE,CAAC;IACtC,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,KAAkB;QACtB,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,aAAa,EAAE,CAAC;IAC1C,CAAC;IAED;;OAEG;IACH,QAAQ,CAAC,KAAkB;QACzB,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,QAAQ,EAAE,CAAC;IACrC,CAAC;IAED;;OAEG;IACH,QAAQ,CAAC,KAAkB;QACzB,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,QAAQ,EAAE,CAAC;IACrC,CAAC;IAED;;OAEG;IACH,OAAO,CAAC,KAAkB,EAAE,KAAa;QACvC,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;IACzC,CAAC;IAED;;OAEG;IACH,GAAG,CAAC,GAAG,MAAqB;QAC1B,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACxB,MAAM,IAAI,KAAK,CAAC,oCAAoC,CAAC,CAAC;QACxD,CAAC;QACD,OAAO,oBAAO,CAAC,GAAG,CAAC,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACvD,CAAC;IAED;;OAEG;IACH,GAAG,CAAC,GAAG,MAAqB;QAC1B,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACxB,MAAM,IAAI,KAAK,CAAC,oCAAoC,CAAC,CAAC;QACxD,CAAC;QACD,OAAO,oBAAO,CAAC,GAAG,CAAC,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACvD,CAAC;IAED;;OAEG;IACH,GAAG,CAAC,MAAqB;QACvB,OAAO,MAAM,CAAC,MAAM,CAAU,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,IAAI,oBAAO,CAAC,CAAC,CAAC,CAAC,CAAC;IACpF,CAAC;IAED;;OAEG;IACH,GAAG,CAAC,MAAqB;QACvB,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACxB,MAAM,IAAI,KAAK,CAAC,iCAAiC,CAAC,CAAC;QACrD,CAAC;QACD,OAAO,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,SAAS,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;IACnD,CAAC;IAED;;OAEG;IACH,OAAO,CAAC,MAAqB;QAC3B,OAAO,MAAM,CAAC,MAAM,CAAU,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,IAAI,oBAAO,CAAC,CAAC,CAAC,CAAC,CAAC;IACrF,CAAC;IAED;;OAEG;IACH,IAAI;QACF,OAAO,IAAI,oBAAO,CAAC,CAAC,CAAC,CAAC;IACxB,CAAC;IAED;;OAEG;IACH,GAAG;QACD,OAAO,IAAI,oBAAO,CAAC,CAAC,CAAC,CAAC;IACxB,CAAC;CACF;AA/WD,oCA+WC;AAED,4BAA4B;AACf,QAAA,YAAY,GAAG,IAAI,YAAY,EAAE,CAAC","sourcesContent":["import Decimal from 'decimal.js';\nimport { DecimalConfig, DecimalRoundingMode } from './types';\nimport { InvalidDecimalError, DecimalDivisionByZeroError } from './errors';\n\n// Map our rounding modes to decimal.js rounding modes\nconst ROUNDING_MODE_MAP: Record<DecimalRoundingMode, Decimal.Rounding> = {\n  [DecimalRoundingMode.CEIL]: Decimal.ROUND_CEIL,\n  [DecimalRoundingMode.FLOOR]: Decimal.ROUND_FLOOR,\n  [DecimalRoundingMode.DOWN]: Decimal.ROUND_DOWN,\n  [DecimalRoundingMode.UP]: Decimal.ROUND_UP,\n  [DecimalRoundingMode.HALF_UP]: Decimal.ROUND_HALF_UP,\n  [DecimalRoundingMode.HALF_DOWN]: Decimal.ROUND_HALF_DOWN,\n  [DecimalRoundingMode.HALF_EVEN]: Decimal.ROUND_HALF_EVEN,\n  [DecimalRoundingMode.HALF_ODD]: Decimal.ROUND_HALF_CEIL, // decimal.js doesn't have HALF_ODD, use HALF_CEIL as approximation\n};\n\nexport type DecimalLike = Decimal | string | number | bigint;\n\nexport interface DecimalUtilsConfig {\n  precision: number;\n  roundingMode: DecimalRoundingMode;\n  divisionScale: number;\n  maxExponent: number;\n  minExponent: number;\n}\n\nconst DEFAULT_CONFIG: DecimalUtilsConfig = {\n  precision: 20,\n  roundingMode: DecimalRoundingMode.HALF_UP,\n  divisionScale: 10,\n  maxExponent: 1000,\n  minExponent: -1000,\n};\n\nexport class DecimalUtils {\n  private config: DecimalUtilsConfig;\n\n  constructor(config?: Partial<DecimalConfig>) {\n    this.config = {\n      ...DEFAULT_CONFIG,\n      ...config,\n    };\n\n    // Configure decimal.js globally\n    Decimal.set({\n      precision: this.config.precision,\n      rounding: ROUNDING_MODE_MAP[this.config.roundingMode],\n      toExpNeg: this.config.minExponent,\n      toExpPos: this.config.maxExponent,\n    });\n  }\n\n  /**\n   * Create a Decimal from various input types\n   */\n  from(value: DecimalLike): Decimal {\n    if (value instanceof Decimal) {\n      return value;\n    }\n\n    try {\n      if (typeof value === 'bigint') {\n        return new Decimal(value.toString());\n      }\n      return new Decimal(value);\n    } catch {\n      throw new InvalidDecimalError(String(value));\n    }\n  }\n\n  /**\n   * Check if a value is a Decimal\n   */\n  isDecimal(value: unknown): value is Decimal {\n    return value instanceof Decimal;\n  }\n\n  /**\n   * Convert a value to Decimal if it's numeric\n   */\n  toDecimal(value: unknown): Decimal | unknown {\n    if (value instanceof Decimal) {\n      return value;\n    }\n    if (typeof value === 'number' && !isNaN(value) && isFinite(value)) {\n      return new Decimal(value);\n    }\n    if (typeof value === 'string') {\n      const trimmed = value.trim();\n      if (/^-?\\d+\\.?\\d*$/.test(trimmed) || /^-?\\d*\\.?\\d+$/.test(trimmed)) {\n        try {\n          return new Decimal(trimmed);\n        } catch {\n          return value;\n        }\n      }\n    }\n    if (typeof value === 'bigint') {\n      return new Decimal(value.toString());\n    }\n    return value;\n  }\n\n  /**\n   * Addition\n   */\n  add(a: DecimalLike, b: DecimalLike): Decimal {\n    return this.from(a).plus(this.from(b));\n  }\n\n  /**\n   * Subtraction\n   */\n  subtract(a: DecimalLike, b: DecimalLike): Decimal {\n    return this.from(a).minus(this.from(b));\n  }\n\n  /**\n   * Multiplication\n   */\n  multiply(a: DecimalLike, b: DecimalLike): Decimal {\n    return this.from(a).times(this.from(b));\n  }\n\n  /**\n   * Division with scale\n   */\n  divide(a: DecimalLike, b: DecimalLike, scale?: number, roundingMode?: DecimalRoundingMode): Decimal {\n    const divisor = this.from(b);\n    if (divisor.isZero()) {\n      throw new DecimalDivisionByZeroError();\n    }\n\n    const result = this.from(a).dividedBy(divisor);\n\n    if (scale !== undefined) {\n      const rounding = roundingMode\n        ? ROUNDING_MODE_MAP[roundingMode]\n        : ROUNDING_MODE_MAP[this.config.roundingMode];\n      return result.toDecimalPlaces(scale, rounding);\n    }\n\n    return result.toDecimalPlaces(this.config.divisionScale, ROUNDING_MODE_MAP[this.config.roundingMode]);\n  }\n\n  /**\n   * Modulo\n   */\n  modulo(a: DecimalLike, b: DecimalLike): Decimal {\n    const divisor = this.from(b);\n    if (divisor.isZero()) {\n      throw new DecimalDivisionByZeroError();\n    }\n    return this.from(a).modulo(divisor);\n  }\n\n  /**\n   * Power\n   */\n  power(base: DecimalLike, exponent: number): Decimal {\n    return this.from(base).pow(exponent);\n  }\n\n  /**\n   * Negation\n   */\n  negate(value: DecimalLike): Decimal {\n    return this.from(value).negated();\n  }\n\n  /**\n   * Absolute value\n   */\n  abs(value: DecimalLike): Decimal {\n    return this.from(value).absoluteValue();\n  }\n\n  /**\n   * Round to specified decimal places\n   */\n  round(value: DecimalLike, scale: number, roundingMode?: DecimalRoundingMode): Decimal {\n    const rounding = roundingMode\n      ? ROUNDING_MODE_MAP[roundingMode]\n      : ROUNDING_MODE_MAP[this.config.roundingMode];\n    return this.from(value).toDecimalPlaces(scale, rounding);\n  }\n\n  /**\n   * Floor to specified decimal places\n   */\n  floor(value: DecimalLike, scale: number = 0): Decimal {\n    return this.from(value).toDecimalPlaces(scale, Decimal.ROUND_FLOOR);\n  }\n\n  /**\n   * Ceiling to specified decimal places\n   */\n  ceil(value: DecimalLike, scale: number = 0): Decimal {\n    return this.from(value).toDecimalPlaces(scale, Decimal.ROUND_CEIL);\n  }\n\n  /**\n   * Truncate to specified decimal places\n   */\n  truncate(value: DecimalLike, scale: number = 0): Decimal {\n    return this.from(value).toDecimalPlaces(scale, Decimal.ROUND_DOWN);\n  }\n\n  /**\n   * Square root\n   */\n  sqrt(value: DecimalLike): Decimal {\n    return this.from(value).sqrt();\n  }\n\n  /**\n   * Natural logarithm\n   */\n  ln(value: DecimalLike): Decimal {\n    return this.from(value).ln();\n  }\n\n  /**\n   * Base-10 logarithm\n   */\n  log10(value: DecimalLike): Decimal {\n    return this.from(value).log(10);\n  }\n\n  /**\n   * Comparison: returns -1, 0, or 1\n   */\n  compare(a: DecimalLike, b: DecimalLike): -1 | 0 | 1 {\n    const result = this.from(a).comparedTo(this.from(b));\n    return result as -1 | 0 | 1;\n  }\n\n  /**\n   * Equality check\n   */\n  equals(a: DecimalLike, b: DecimalLike): boolean {\n    return this.from(a).equals(this.from(b));\n  }\n\n  /**\n   * Greater than\n   */\n  greaterThan(a: DecimalLike, b: DecimalLike): boolean {\n    return this.from(a).greaterThan(this.from(b));\n  }\n\n  /**\n   * Greater than or equal\n   */\n  greaterThanOrEqual(a: DecimalLike, b: DecimalLike): boolean {\n    return this.from(a).greaterThanOrEqualTo(this.from(b));\n  }\n\n  /**\n   * Less than\n   */\n  lessThan(a: DecimalLike, b: DecimalLike): boolean {\n    return this.from(a).lessThan(this.from(b));\n  }\n\n  /**\n   * Less than or equal\n   */\n  lessThanOrEqual(a: DecimalLike, b: DecimalLike): boolean {\n    return this.from(a).lessThanOrEqualTo(this.from(b));\n  }\n\n  /**\n   * Check if zero\n   */\n  isZero(value: DecimalLike): boolean {\n    return this.from(value).isZero();\n  }\n\n  /**\n   * Check if positive\n   */\n  isPositive(value: DecimalLike): boolean {\n    return this.from(value).isPositive();\n  }\n\n  /**\n   * Check if negative\n   */\n  isNegative(value: DecimalLike): boolean {\n    return this.from(value).isNegative();\n  }\n\n  /**\n   * Check if integer\n   */\n  isInteger(value: DecimalLike): boolean {\n    return this.from(value).isInteger();\n  }\n\n  /**\n   * Get sign: -1, 0, or 1\n   */\n  sign(value: DecimalLike): -1 | 0 | 1 {\n    const d = this.from(value);\n    if (d.isZero()) return 0;\n    return d.isNegative() ? -1 : 1;\n  }\n\n  /**\n   * Get precision (total significant digits)\n   */\n  precision(value: DecimalLike): number {\n    return this.from(value).precision();\n  }\n\n  /**\n   * Get scale (decimal places)\n   */\n  scale(value: DecimalLike): number {\n    return this.from(value).decimalPlaces();\n  }\n\n  /**\n   * Convert to JavaScript number (may lose precision)\n   */\n  toNumber(value: DecimalLike): number {\n    return this.from(value).toNumber();\n  }\n\n  /**\n   * Convert to string\n   */\n  toString(value: DecimalLike): string {\n    return this.from(value).toString();\n  }\n\n  /**\n   * Convert to fixed decimal places string\n   */\n  toFixed(value: DecimalLike, scale: number): string {\n    return this.from(value).toFixed(scale);\n  }\n\n  /**\n   * Minimum of values\n   */\n  min(...values: DecimalLike[]): Decimal {\n    if (values.length === 0) {\n      throw new Error('min requires at least one argument');\n    }\n    return Decimal.min(...values.map(v => this.from(v)));\n  }\n\n  /**\n   * Maximum of values\n   */\n  max(...values: DecimalLike[]): Decimal {\n    if (values.length === 0) {\n      throw new Error('max requires at least one argument');\n    }\n    return Decimal.max(...values.map(v => this.from(v)));\n  }\n\n  /**\n   * Sum of values\n   */\n  sum(values: DecimalLike[]): Decimal {\n    return values.reduce<Decimal>((acc, v) => acc.plus(this.from(v)), new Decimal(0));\n  }\n\n  /**\n   * Average of values\n   */\n  avg(values: DecimalLike[]): Decimal {\n    if (values.length === 0) {\n      throw new Error('avg requires at least one value');\n    }\n    return this.sum(values).dividedBy(values.length);\n  }\n\n  /**\n   * Product of values\n   */\n  product(values: DecimalLike[]): Decimal {\n    return values.reduce<Decimal>((acc, v) => acc.times(this.from(v)), new Decimal(1));\n  }\n\n  /**\n   * Create zero\n   */\n  zero(): Decimal {\n    return new Decimal(0);\n  }\n\n  /**\n   * Create one\n   */\n  one(): Decimal {\n    return new Decimal(1);\n  }\n}\n\n// Export a default instance\nexport const decimalUtils = new DecimalUtils();\n\n// Re-export Decimal type for convenience\nexport { Decimal };\n"]}
@@ -0,0 +1,20 @@
1
+ import { ASTNode } from './types';
2
+ export declare class DependencyExtractor {
3
+ private parser;
4
+ constructor();
5
+ /**
6
+ * Extract all variable dependencies from an expression string
7
+ * Returns a set of variable names (without the $ or @ prefix)
8
+ */
9
+ extract(expression: string): Set<string>;
10
+ /**
11
+ * Extract dependencies from an AST node
12
+ */
13
+ extractFromNode(node: ASTNode): Set<string>;
14
+ private visit;
15
+ /**
16
+ * Visit the root of a member/index access chain
17
+ * For $product.price.value, we want to extract 'product'
18
+ */
19
+ private visitMemberRoot;
20
+ }
@@ -0,0 +1,103 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.DependencyExtractor = void 0;
4
+ const parser_1 = require("./parser");
5
+ class DependencyExtractor {
6
+ constructor() {
7
+ this.parser = new parser_1.Parser();
8
+ }
9
+ /**
10
+ * Extract all variable dependencies from an expression string
11
+ * Returns a set of variable names (without the $ or @ prefix)
12
+ */
13
+ extract(expression) {
14
+ const ast = this.parser.parse(expression);
15
+ return this.extractFromNode(ast);
16
+ }
17
+ /**
18
+ * Extract dependencies from an AST node
19
+ */
20
+ extractFromNode(node) {
21
+ const dependencies = new Set();
22
+ this.visit(node, dependencies);
23
+ return dependencies;
24
+ }
25
+ visit(node, dependencies) {
26
+ switch (node.type) {
27
+ case 'VariableReference':
28
+ // Only extract $ prefixed variables (not @ context variables)
29
+ // Context variables are considered external and don't form part of the dependency graph
30
+ if (node.prefix === '$') {
31
+ dependencies.add(node.name);
32
+ }
33
+ break;
34
+ case 'BinaryOperation':
35
+ this.visit(node.left, dependencies);
36
+ this.visit(node.right, dependencies);
37
+ break;
38
+ case 'UnaryOperation':
39
+ this.visit(node.operand, dependencies);
40
+ break;
41
+ case 'ConditionalExpression':
42
+ this.visit(node.condition, dependencies);
43
+ this.visit(node.consequent, dependencies);
44
+ this.visit(node.alternate, dependencies);
45
+ break;
46
+ case 'FunctionCall':
47
+ // Visit all function arguments
48
+ for (const arg of node.arguments) {
49
+ this.visit(arg, dependencies);
50
+ }
51
+ break;
52
+ case 'MemberAccess':
53
+ // For member access like $product.price, we want to extract 'product'
54
+ // The root variable is what matters for dependency tracking
55
+ this.visitMemberRoot(node.object, dependencies);
56
+ break;
57
+ case 'IndexAccess':
58
+ // Similar to MemberAccess - extract the root variable
59
+ this.visitMemberRoot(node.object, dependencies);
60
+ // Also visit the index expression as it might contain variables
61
+ this.visit(node.index, dependencies);
62
+ break;
63
+ case 'ArrayLiteral':
64
+ for (const element of node.elements) {
65
+ this.visit(element, dependencies);
66
+ }
67
+ break;
68
+ // Literals don't have dependencies
69
+ case 'DecimalLiteral':
70
+ case 'NumberLiteral':
71
+ case 'StringLiteral':
72
+ case 'BooleanLiteral':
73
+ case 'NullLiteral':
74
+ break;
75
+ }
76
+ }
77
+ /**
78
+ * Visit the root of a member/index access chain
79
+ * For $product.price.value, we want to extract 'product'
80
+ */
81
+ visitMemberRoot(node, dependencies) {
82
+ switch (node.type) {
83
+ case 'VariableReference':
84
+ if (node.prefix === '$') {
85
+ dependencies.add(node.name);
86
+ }
87
+ break;
88
+ case 'MemberAccess':
89
+ this.visitMemberRoot(node.object, dependencies);
90
+ break;
91
+ case 'IndexAccess':
92
+ this.visitMemberRoot(node.object, dependencies);
93
+ this.visit(node.index, dependencies);
94
+ break;
95
+ default:
96
+ // For other node types (like function calls), visit normally
97
+ this.visit(node, dependencies);
98
+ break;
99
+ }
100
+ }
101
+ }
102
+ exports.DependencyExtractor = DependencyExtractor;
103
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dependency-extractor.js","sourceRoot":"","sources":["../src/dependency-extractor.ts"],"names":[],"mappings":"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","sourcesContent":["import { ASTNode } from './types';\nimport { Parser } from './parser';\n\nexport class DependencyExtractor {\n  private parser: Parser;\n\n  constructor() {\n    this.parser = new Parser();\n  }\n\n  /**\n   * Extract all variable dependencies from an expression string\n   * Returns a set of variable names (without the $ or @ prefix)\n   */\n  extract(expression: string): Set<string> {\n    const ast = this.parser.parse(expression);\n    return this.extractFromNode(ast);\n  }\n\n  /**\n   * Extract dependencies from an AST node\n   */\n  extractFromNode(node: ASTNode): Set<string> {\n    const dependencies = new Set<string>();\n    this.visit(node, dependencies);\n    return dependencies;\n  }\n\n  private visit(node: ASTNode, dependencies: Set<string>): void {\n    switch (node.type) {\n      case 'VariableReference':\n        // Only extract $ prefixed variables (not @ context variables)\n        // Context variables are considered external and don't form part of the dependency graph\n        if (node.prefix === '$') {\n          dependencies.add(node.name);\n        }\n        break;\n\n      case 'BinaryOperation':\n        this.visit(node.left, dependencies);\n        this.visit(node.right, dependencies);\n        break;\n\n      case 'UnaryOperation':\n        this.visit(node.operand, dependencies);\n        break;\n\n      case 'ConditionalExpression':\n        this.visit(node.condition, dependencies);\n        this.visit(node.consequent, dependencies);\n        this.visit(node.alternate, dependencies);\n        break;\n\n      case 'FunctionCall':\n        // Visit all function arguments\n        for (const arg of node.arguments) {\n          this.visit(arg, dependencies);\n        }\n        break;\n\n      case 'MemberAccess':\n        // For member access like $product.price, we want to extract 'product'\n        // The root variable is what matters for dependency tracking\n        this.visitMemberRoot(node.object, dependencies);\n        break;\n\n      case 'IndexAccess':\n        // Similar to MemberAccess - extract the root variable\n        this.visitMemberRoot(node.object, dependencies);\n        // Also visit the index expression as it might contain variables\n        this.visit(node.index, dependencies);\n        break;\n\n      case 'ArrayLiteral':\n        for (const element of node.elements) {\n          this.visit(element, dependencies);\n        }\n        break;\n\n      // Literals don't have dependencies\n      case 'DecimalLiteral':\n      case 'NumberLiteral':\n      case 'StringLiteral':\n      case 'BooleanLiteral':\n      case 'NullLiteral':\n        break;\n    }\n  }\n\n  /**\n   * Visit the root of a member/index access chain\n   * For $product.price.value, we want to extract 'product'\n   */\n  private visitMemberRoot(node: ASTNode, dependencies: Set<string>): void {\n    switch (node.type) {\n      case 'VariableReference':\n        if (node.prefix === '$') {\n          dependencies.add(node.name);\n        }\n        break;\n\n      case 'MemberAccess':\n        this.visitMemberRoot(node.object, dependencies);\n        break;\n\n      case 'IndexAccess':\n        this.visitMemberRoot(node.object, dependencies);\n        this.visit(node.index, dependencies);\n        break;\n\n      default:\n        // For other node types (like function calls), visit normally\n        this.visit(node, dependencies);\n        break;\n    }\n  }\n}\n"]}
@@ -0,0 +1,60 @@
1
+ import { DependencyGraph as IDependencyGraph, FormulaDefinition } from './types';
2
+ export declare class DependencyGraph implements IDependencyGraph {
3
+ nodes: Set<string>;
4
+ edges: Map<string, Set<string>>;
5
+ constructor();
6
+ /**
7
+ * Add a node to the graph
8
+ */
9
+ addNode(id: string): void;
10
+ /**
11
+ * Add an edge from source to target (source depends on target)
12
+ */
13
+ addEdge(source: string, target: string): void;
14
+ /**
15
+ * Check if the graph has any cycles
16
+ */
17
+ hasCycles(): boolean;
18
+ /**
19
+ * Get all root nodes (nodes with no dependencies)
20
+ */
21
+ getRoots(): Set<string>;
22
+ /**
23
+ * Get all nodes that depend on the given node
24
+ */
25
+ getDependents(nodeId: string): Set<string>;
26
+ /**
27
+ * Get direct dependencies of a node
28
+ */
29
+ getDependencies(nodeId: string): Set<string>;
30
+ /**
31
+ * Get all transitive dependencies of a node
32
+ */
33
+ getTransitiveDependencies(nodeId: string): Set<string>;
34
+ private collectTransitiveDependencies;
35
+ /**
36
+ * Perform topological sort using Kahn's algorithm
37
+ * Returns nodes in evaluation order (dependencies first)
38
+ * Throws CircularDependencyError if a cycle is detected
39
+ */
40
+ topologicalSort(): string[];
41
+ /**
42
+ * Find a cycle in the graph using DFS
43
+ */
44
+ private findCycle;
45
+ }
46
+ /**
47
+ * Build a dependency graph from formula definitions
48
+ */
49
+ export declare class DependencyGraphBuilder {
50
+ private extractor;
51
+ constructor();
52
+ /**
53
+ * Build a dependency graph from formula definitions
54
+ */
55
+ build(formulas: FormulaDefinition[]): DependencyGraph;
56
+ /**
57
+ * Get the evaluation order for a set of formulas
58
+ */
59
+ getEvaluationOrder(formulas: FormulaDefinition[]): string[];
60
+ }