@the-trybe/formula-engine 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.claude/settings.local.json +6 -0
- package/PRD_FORMULA_ENGINE.md +1863 -0
- package/README.md +382 -0
- package/dist/decimal-utils.d.ts +180 -0
- package/dist/decimal-utils.js +355 -0
- package/dist/dependency-extractor.d.ts +20 -0
- package/dist/dependency-extractor.js +103 -0
- package/dist/dependency-graph.d.ts +60 -0
- package/dist/dependency-graph.js +252 -0
- package/dist/errors.d.ts +161 -0
- package/dist/errors.js +260 -0
- package/dist/evaluator.d.ts +51 -0
- package/dist/evaluator.js +494 -0
- package/dist/formula-engine.d.ts +79 -0
- package/dist/formula-engine.js +355 -0
- package/dist/functions.d.ts +3 -0
- package/dist/functions.js +720 -0
- package/dist/index.d.ts +10 -0
- package/dist/index.js +61 -0
- package/dist/lexer.d.ts +25 -0
- package/dist/lexer.js +357 -0
- package/dist/parser.d.ts +32 -0
- package/dist/parser.js +372 -0
- package/dist/types.d.ts +228 -0
- package/dist/types.js +62 -0
- package/jest.config.js +23 -0
- package/package.json +35 -0
- package/src/decimal-utils.ts +408 -0
- package/src/dependency-extractor.ts +117 -0
- package/src/dependency-graph.test.ts +238 -0
- package/src/dependency-graph.ts +288 -0
- package/src/errors.ts +296 -0
- package/src/evaluator.ts +604 -0
- package/src/formula-engine.test.ts +660 -0
- package/src/formula-engine.ts +430 -0
- package/src/functions.ts +770 -0
- package/src/index.ts +103 -0
- package/src/lexer.test.ts +288 -0
- package/src/lexer.ts +394 -0
- package/src/parser.test.ts +349 -0
- package/src/parser.ts +449 -0
- package/src/types.ts +347 -0
- package/tsconfig.json +29 -0
|
@@ -0,0 +1,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":"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","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":";;;AACA,qCAAkC;AAElC,MAAa,mBAAmB;IAG9B;QACE,IAAI,CAAC,MAAM,GAAG,IAAI,eAAM,EAAE,CAAC;IAC7B,CAAC;IAED;;;OAGG;IACH,OAAO,CAAC,UAAkB;QACxB,MAAM,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QAC1C,OAAO,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC;IACnC,CAAC;IAED;;OAEG;IACH,eAAe,CAAC,IAAa;QAC3B,MAAM,YAAY,GAAG,IAAI,GAAG,EAAU,CAAC;QACvC,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,YAAY,CAAC,CAAC;QAC/B,OAAO,YAAY,CAAC;IACtB,CAAC;IAEO,KAAK,CAAC,IAAa,EAAE,YAAyB;QACpD,QAAQ,IAAI,CAAC,IAAI,EAAE,CAAC;YAClB,KAAK,mBAAmB;gBACtB,8DAA8D;gBAC9D,wFAAwF;gBACxF,IAAI,IAAI,CAAC,MAAM,KAAK,GAAG,EAAE,CAAC;oBACxB,YAAY,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBAC9B,CAAC;gBACD,MAAM;YAER,KAAK,iBAAiB;gBACpB,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,EAAE,YAAY,CAAC,CAAC;gBACpC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,EAAE,YAAY,CAAC,CAAC;gBACrC,MAAM;YAER,KAAK,gBAAgB;gBACnB,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;gBACvC,MAAM;YAER,KAAK,uBAAuB;gBAC1B,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;gBACzC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,EAAE,YAAY,CAAC,CAAC;gBAC1C,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;gBACzC,MAAM;YAER,KAAK,cAAc;gBACjB,+BAA+B;gBAC/B,KAAK,MAAM,GAAG,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;oBACjC,IAAI,CAAC,KAAK,CAAC,GAAG,EAAE,YAAY,CAAC,CAAC;gBAChC,CAAC;gBACD,MAAM;YAER,KAAK,cAAc;gBACjB,sEAAsE;gBACtE,4DAA4D;gBAC5D,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC;gBAChD,MAAM;YAER,KAAK,aAAa;gBAChB,sDAAsD;gBACtD,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC;gBAChD,gEAAgE;gBAChE,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,EAAE,YAAY,CAAC,CAAC;gBACrC,MAAM;YAER,KAAK,cAAc;gBACjB,KAAK,MAAM,OAAO,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;oBACpC,IAAI,CAAC,KAAK,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;gBACpC,CAAC;gBACD,MAAM;YAER,mCAAmC;YACnC,KAAK,gBAAgB,CAAC;YACtB,KAAK,eAAe,CAAC;YACrB,KAAK,eAAe,CAAC;YACrB,KAAK,gBAAgB,CAAC;YACtB,KAAK,aAAa;gBAChB,MAAM;QACV,CAAC;IACH,CAAC;IAED;;;OAGG;IACK,eAAe,CAAC,IAAa,EAAE,YAAyB;QAC9D,QAAQ,IAAI,CAAC,IAAI,EAAE,CAAC;YAClB,KAAK,mBAAmB;gBACtB,IAAI,IAAI,CAAC,MAAM,KAAK,GAAG,EAAE,CAAC;oBACxB,YAAY,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBAC9B,CAAC;gBACD,MAAM;YAER,KAAK,cAAc;gBACjB,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC;gBAChD,MAAM;YAER,KAAK,aAAa;gBAChB,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC;gBAChD,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,EAAE,YAAY,CAAC,CAAC;gBACrC,MAAM;YAER;gBACE,6DAA6D;gBAC7D,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,YAAY,CAAC,CAAC;gBAC/B,MAAM;QACV,CAAC;IACH,CAAC;CACF;AAjHD,kDAiHC","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
|
+
}
|