@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/types.js ADDED
@@ -0,0 +1,62 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.TokenType = exports.DecimalRoundingMode = void 0;
4
+ // ============================================================================
5
+ // Decimal Configuration
6
+ // ============================================================================
7
+ var DecimalRoundingMode;
8
+ (function (DecimalRoundingMode) {
9
+ DecimalRoundingMode["CEIL"] = "CEIL";
10
+ DecimalRoundingMode["FLOOR"] = "FLOOR";
11
+ DecimalRoundingMode["DOWN"] = "DOWN";
12
+ DecimalRoundingMode["UP"] = "UP";
13
+ DecimalRoundingMode["HALF_UP"] = "HALF_UP";
14
+ DecimalRoundingMode["HALF_DOWN"] = "HALF_DOWN";
15
+ DecimalRoundingMode["HALF_EVEN"] = "HALF_EVEN";
16
+ DecimalRoundingMode["HALF_ODD"] = "HALF_ODD";
17
+ })(DecimalRoundingMode || (exports.DecimalRoundingMode = DecimalRoundingMode = {}));
18
+ // ============================================================================
19
+ // Token Types
20
+ // ============================================================================
21
+ var TokenType;
22
+ (function (TokenType) {
23
+ // Literals
24
+ TokenType["NUMBER"] = "NUMBER";
25
+ TokenType["STRING"] = "STRING";
26
+ TokenType["BOOLEAN"] = "BOOLEAN";
27
+ TokenType["NULL"] = "NULL";
28
+ // Identifiers and Variables
29
+ TokenType["IDENTIFIER"] = "IDENTIFIER";
30
+ TokenType["VARIABLE"] = "VARIABLE";
31
+ TokenType["CONTEXT_VAR"] = "CONTEXT_VAR";
32
+ // Operators
33
+ TokenType["PLUS"] = "PLUS";
34
+ TokenType["MINUS"] = "MINUS";
35
+ TokenType["MULTIPLY"] = "MULTIPLY";
36
+ TokenType["DIVIDE"] = "DIVIDE";
37
+ TokenType["MODULO"] = "MODULO";
38
+ TokenType["POWER"] = "POWER";
39
+ // Comparison
40
+ TokenType["EQ"] = "EQ";
41
+ TokenType["NEQ"] = "NEQ";
42
+ TokenType["LT"] = "LT";
43
+ TokenType["GT"] = "GT";
44
+ TokenType["LTE"] = "LTE";
45
+ TokenType["GTE"] = "GTE";
46
+ // Logical
47
+ TokenType["AND"] = "AND";
48
+ TokenType["OR"] = "OR";
49
+ TokenType["NOT"] = "NOT";
50
+ // Punctuation
51
+ TokenType["LPAREN"] = "LPAREN";
52
+ TokenType["RPAREN"] = "RPAREN";
53
+ TokenType["LBRACKET"] = "LBRACKET";
54
+ TokenType["RBRACKET"] = "RBRACKET";
55
+ TokenType["COMMA"] = "COMMA";
56
+ TokenType["DOT"] = "DOT";
57
+ TokenType["QUESTION"] = "QUESTION";
58
+ TokenType["COLON"] = "COLON";
59
+ // Special
60
+ TokenType["EOF"] = "EOF";
61
+ })(TokenType || (exports.TokenType = TokenType = {}));
62
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":";;;AAUA,+EAA+E;AAC/E,wBAAwB;AACxB,+EAA+E;AAE/E,IAAY,mBASX;AATD,WAAY,mBAAmB;IAC7B,oCAAa,CAAA;IACb,sCAAe,CAAA;IACf,oCAAa,CAAA;IACb,gCAAS,CAAA;IACT,0CAAmB,CAAA;IACnB,8CAAuB,CAAA;IACvB,8CAAuB,CAAA;IACvB,4CAAqB,CAAA;AACvB,CAAC,EATW,mBAAmB,mCAAnB,mBAAmB,QAS9B;AA0QD,+EAA+E;AAC/E,cAAc;AACd,+EAA+E;AAE/E,IAAY,SA6CX;AA7CD,WAAY,SAAS;IACnB,WAAW;IACX,8BAAiB,CAAA;IACjB,8BAAiB,CAAA;IACjB,gCAAmB,CAAA;IACnB,0BAAa,CAAA;IAEb,4BAA4B;IAC5B,sCAAyB,CAAA;IACzB,kCAAqB,CAAA;IACrB,wCAA2B,CAAA;IAE3B,YAAY;IACZ,0BAAa,CAAA;IACb,4BAAe,CAAA;IACf,kCAAqB,CAAA;IACrB,8BAAiB,CAAA;IACjB,8BAAiB,CAAA;IACjB,4BAAe,CAAA;IAEf,aAAa;IACb,sBAAS,CAAA;IACT,wBAAW,CAAA;IACX,sBAAS,CAAA;IACT,sBAAS,CAAA;IACT,wBAAW,CAAA;IACX,wBAAW,CAAA;IAEX,UAAU;IACV,wBAAW,CAAA;IACX,sBAAS,CAAA;IACT,wBAAW,CAAA;IAEX,cAAc;IACd,8BAAiB,CAAA;IACjB,8BAAiB,CAAA;IACjB,kCAAqB,CAAA;IACrB,kCAAqB,CAAA;IACrB,4BAAe,CAAA;IACf,wBAAW,CAAA;IACX,kCAAqB,CAAA;IACrB,4BAAe,CAAA;IAEf,UAAU;IACV,wBAAW,CAAA;AACb,CAAC,EA7CW,SAAS,yBAAT,SAAS,QA6CpB","sourcesContent":["import { Decimal } from 'decimal.js';\n\n// ============================================================================\n// Value Types\n// ============================================================================\n\nexport type ValueType = 'number' | 'decimal' | 'string' | 'boolean' | 'array' | 'object' | 'null' | 'any';\n\nexport type FormulaValue = Decimal | number | string | boolean | null | FormulaValue[] | { [key: string]: FormulaValue };\n\n// ============================================================================\n// Decimal Configuration\n// ============================================================================\n\nexport enum DecimalRoundingMode {\n  CEIL = 'CEIL',\n  FLOOR = 'FLOOR',\n  DOWN = 'DOWN',\n  UP = 'UP',\n  HALF_UP = 'HALF_UP',\n  HALF_DOWN = 'HALF_DOWN',\n  HALF_EVEN = 'HALF_EVEN',\n  HALF_ODD = 'HALF_ODD',\n}\n\nexport interface DecimalConfig {\n  precision?: number;\n  roundingMode?: DecimalRoundingMode;\n  divisionScale?: number;\n  preserveTrailingZeros?: boolean;\n  autoConvertFloats?: boolean;\n  maxExponent?: number;\n  minExponent?: number;\n}\n\n// ============================================================================\n// Rounding Configuration\n// ============================================================================\n\nexport interface RoundingConfig {\n  mode: 'HALF_UP' | 'HALF_DOWN' | 'FLOOR' | 'CEIL' | 'NONE';\n  precision: number;\n}\n\n// ============================================================================\n// Error Behavior\n// ============================================================================\n\nexport interface ErrorBehavior {\n  type: 'THROW' | 'NULL' | 'ZERO' | 'DEFAULT' | 'SKIP';\n  defaultValue?: unknown;\n}\n\n// ============================================================================\n// Formula Definition\n// ============================================================================\n\nexport interface FormulaDefinition {\n  id: string;\n  expression: string;\n  dependencies?: string[];\n  onError?: ErrorBehavior;\n  defaultValue?: unknown;\n  rounding?: RoundingConfig;\n  metadata?: Record<string, unknown>;\n}\n\n// ============================================================================\n// Evaluation Context\n// ============================================================================\n\nexport interface EvaluationContext {\n  variables: Record<string, unknown>;\n  collections?: Record<string, unknown[]>;\n  extra?: Record<string, unknown>;\n}\n\n// ============================================================================\n// Engine Configuration\n// ============================================================================\n\nexport interface OperatorDefinition {\n  symbol: string;\n  precedence: number;\n  associativity: 'left' | 'right';\n  handler: (left: unknown, right: unknown) => unknown;\n}\n\nexport interface SecurityConfig {\n  maxExpressionLength?: number;\n  maxRecursionDepth?: number;\n  maxIterations?: number;\n  maxExecutionTime?: number;\n  allowedFunctions?: string[];\n  blockedFunctions?: string[];\n}\n\nexport interface FormulaEngineConfig {\n  enableCache?: boolean;\n  maxCacheSize?: number;\n  defaultErrorBehavior?: ErrorBehavior;\n  defaultRounding?: RoundingConfig;\n  variablePrefix?: string;\n  contextPrefix?: string;\n  strictMode?: boolean;\n  operators?: OperatorDefinition[];\n  functions?: FunctionDefinition[];\n  decimal?: DecimalConfig;\n  security?: SecurityConfig;\n}\n\n// ============================================================================\n// AST Node Types\n// ============================================================================\n\nexport interface DecimalLiteral {\n  type: 'DecimalLiteral';\n  value: string;\n  raw: string;\n}\n\nexport interface NumberLiteral {\n  type: 'NumberLiteral';\n  value: number;\n}\n\nexport interface StringLiteral {\n  type: 'StringLiteral';\n  value: string;\n}\n\nexport interface BooleanLiteral {\n  type: 'BooleanLiteral';\n  value: boolean;\n}\n\nexport interface NullLiteral {\n  type: 'NullLiteral';\n}\n\nexport interface ArrayLiteral {\n  type: 'ArrayLiteral';\n  elements: ASTNode[];\n}\n\nexport interface VariableReference {\n  type: 'VariableReference';\n  prefix: '$' | '@';\n  name: string;\n}\n\nexport interface BinaryOperation {\n  type: 'BinaryOperation';\n  operator: string;\n  left: ASTNode;\n  right: ASTNode;\n}\n\nexport interface UnaryOperation {\n  type: 'UnaryOperation';\n  operator: string;\n  operand: ASTNode;\n}\n\nexport interface ConditionalExpression {\n  type: 'ConditionalExpression';\n  condition: ASTNode;\n  consequent: ASTNode;\n  alternate: ASTNode;\n}\n\nexport interface FunctionCall {\n  type: 'FunctionCall';\n  name: string;\n  arguments: ASTNode[];\n}\n\nexport interface MemberAccess {\n  type: 'MemberAccess';\n  object: ASTNode;\n  property: string;\n}\n\nexport interface IndexAccess {\n  type: 'IndexAccess';\n  object: ASTNode;\n  index: ASTNode;\n}\n\nexport type ASTNode =\n  | DecimalLiteral\n  | NumberLiteral\n  | StringLiteral\n  | BooleanLiteral\n  | NullLiteral\n  | ArrayLiteral\n  | VariableReference\n  | BinaryOperation\n  | UnaryOperation\n  | ConditionalExpression\n  | FunctionCall\n  | MemberAccess\n  | IndexAccess;\n\n// ============================================================================\n// Function Definition\n// ============================================================================\n\nexport interface ArgumentType {\n  name: string;\n  type: ValueType;\n  required: boolean;\n  default?: unknown;\n}\n\nexport type FunctionImplementation = (\n  args: unknown[],\n  context: EvaluationContext,\n  engine: unknown\n) => unknown;\n\nexport interface FunctionDefinition {\n  name: string;\n  minArgs: number;\n  maxArgs: number;\n  argTypes?: ArgumentType[];\n  returnType: ValueType;\n  implementation: FunctionImplementation;\n  description?: string;\n}\n\n// ============================================================================\n// Dependency Graph\n// ============================================================================\n\nexport interface DependencyGraph {\n  nodes: Set<string>;\n  edges: Map<string, Set<string>>;\n  hasCycles(): boolean;\n  getRoots(): Set<string>;\n  getDependents(nodeId: string): Set<string>;\n  getDependencies(nodeId: string): Set<string>;\n  getTransitiveDependencies(nodeId: string): Set<string>;\n  topologicalSort(): string[];\n}\n\n// ============================================================================\n// Evaluation Results\n// ============================================================================\n\nexport interface EvaluationResult {\n  value: unknown;\n  success: boolean;\n  error?: Error;\n  executionTimeMs: number;\n  accessedVariables: Set<string>;\n}\n\nexport interface EvaluationResultSet {\n  results: Map<string, EvaluationResult>;\n  success: boolean;\n  errors: Error[];\n  totalExecutionTimeMs: number;\n  evaluationOrder: string[];\n}\n\n// ============================================================================\n// Validation Result\n// ============================================================================\n\nexport interface ValidationResult {\n  valid: boolean;\n  errors: Error[];\n  warnings: string[];\n  dependencyGraph: DependencyGraph;\n  evaluationOrder: string[];\n}\n\n// ============================================================================\n// Cache Statistics\n// ============================================================================\n\nexport interface CacheStats {\n  size: number;\n  hits: number;\n  misses: number;\n  hitRate: number;\n}\n\n// ============================================================================\n// Token Types\n// ============================================================================\n\nexport enum TokenType {\n  // Literals\n  NUMBER = 'NUMBER',\n  STRING = 'STRING',\n  BOOLEAN = 'BOOLEAN',\n  NULL = 'NULL',\n\n  // Identifiers and Variables\n  IDENTIFIER = 'IDENTIFIER',\n  VARIABLE = 'VARIABLE',\n  CONTEXT_VAR = 'CONTEXT_VAR',\n\n  // Operators\n  PLUS = 'PLUS',\n  MINUS = 'MINUS',\n  MULTIPLY = 'MULTIPLY',\n  DIVIDE = 'DIVIDE',\n  MODULO = 'MODULO',\n  POWER = 'POWER',\n\n  // Comparison\n  EQ = 'EQ',\n  NEQ = 'NEQ',\n  LT = 'LT',\n  GT = 'GT',\n  LTE = 'LTE',\n  GTE = 'GTE',\n\n  // Logical\n  AND = 'AND',\n  OR = 'OR',\n  NOT = 'NOT',\n\n  // Punctuation\n  LPAREN = 'LPAREN',\n  RPAREN = 'RPAREN',\n  LBRACKET = 'LBRACKET',\n  RBRACKET = 'RBRACKET',\n  COMMA = 'COMMA',\n  DOT = 'DOT',\n  QUESTION = 'QUESTION',\n  COLON = 'COLON',\n\n  // Special\n  EOF = 'EOF',\n}\n\nexport interface Token {\n  type: TokenType;\n  value: string | number | boolean | null;\n  position: number;\n  line: number;\n  column: number;\n}\n"]}
package/jest.config.js ADDED
@@ -0,0 +1,23 @@
1
+ module.exports = {
2
+ preset: 'ts-jest',
3
+ testEnvironment: 'node',
4
+ roots: ['<rootDir>/src'],
5
+ testMatch: ['**/*.test.ts'],
6
+ collectCoverageFrom: [
7
+ 'src/**/*.ts',
8
+ '!src/**/*.test.ts',
9
+ '!src/index.ts'
10
+ ],
11
+ coverageThreshold: {
12
+ global: {
13
+ branches: 80,
14
+ functions: 80,
15
+ lines: 80,
16
+ statements: 80
17
+ }
18
+ },
19
+ moduleFileExtensions: ['ts', 'js', 'json'],
20
+ transform: {
21
+ '^.+\\.ts$': 'ts-jest'
22
+ }
23
+ };
package/package.json ADDED
@@ -0,0 +1,35 @@
1
+ {
2
+ "name": "@the-trybe/formula-engine",
3
+ "version": "1.0.0",
4
+ "description": "Configuration-driven expression evaluation system with dependency resolution and decimal precision",
5
+ "main": "dist/index.js",
6
+ "types": "dist/index.d.ts",
7
+ "scripts": {
8
+ "build": "tsc",
9
+ "test": "jest",
10
+ "test:watch": "jest --watch",
11
+ "test:coverage": "jest --coverage",
12
+ "lint": "eslint src --ext .ts",
13
+ "clean": "rm -rf dist"
14
+ },
15
+ "keywords": [
16
+ "formula",
17
+ "expression",
18
+ "parser",
19
+ "evaluator",
20
+ "dependency-graph",
21
+ "decimal"
22
+ ],
23
+ "author": "",
24
+ "license": "MIT",
25
+ "devDependencies": {
26
+ "@types/jest": "^29.5.11",
27
+ "@types/node": "^20.10.0",
28
+ "jest": "^29.7.0",
29
+ "ts-jest": "^29.1.1",
30
+ "typescript": "^5.3.0"
31
+ },
32
+ "dependencies": {
33
+ "decimal.js": "^10.4.3"
34
+ }
35
+ }
@@ -0,0 +1,408 @@
1
+ import Decimal from 'decimal.js';
2
+ import { DecimalConfig, DecimalRoundingMode } from './types';
3
+ import { InvalidDecimalError, DecimalDivisionByZeroError } from './errors';
4
+
5
+ // Map our rounding modes to decimal.js rounding modes
6
+ const ROUNDING_MODE_MAP: Record<DecimalRoundingMode, Decimal.Rounding> = {
7
+ [DecimalRoundingMode.CEIL]: Decimal.ROUND_CEIL,
8
+ [DecimalRoundingMode.FLOOR]: Decimal.ROUND_FLOOR,
9
+ [DecimalRoundingMode.DOWN]: Decimal.ROUND_DOWN,
10
+ [DecimalRoundingMode.UP]: Decimal.ROUND_UP,
11
+ [DecimalRoundingMode.HALF_UP]: Decimal.ROUND_HALF_UP,
12
+ [DecimalRoundingMode.HALF_DOWN]: Decimal.ROUND_HALF_DOWN,
13
+ [DecimalRoundingMode.HALF_EVEN]: Decimal.ROUND_HALF_EVEN,
14
+ [DecimalRoundingMode.HALF_ODD]: Decimal.ROUND_HALF_CEIL, // decimal.js doesn't have HALF_ODD, use HALF_CEIL as approximation
15
+ };
16
+
17
+ export type DecimalLike = Decimal | string | number | bigint;
18
+
19
+ export interface DecimalUtilsConfig {
20
+ precision: number;
21
+ roundingMode: DecimalRoundingMode;
22
+ divisionScale: number;
23
+ maxExponent: number;
24
+ minExponent: number;
25
+ }
26
+
27
+ const DEFAULT_CONFIG: DecimalUtilsConfig = {
28
+ precision: 20,
29
+ roundingMode: DecimalRoundingMode.HALF_UP,
30
+ divisionScale: 10,
31
+ maxExponent: 1000,
32
+ minExponent: -1000,
33
+ };
34
+
35
+ export class DecimalUtils {
36
+ private config: DecimalUtilsConfig;
37
+
38
+ constructor(config?: Partial<DecimalConfig>) {
39
+ this.config = {
40
+ ...DEFAULT_CONFIG,
41
+ ...config,
42
+ };
43
+
44
+ // Configure decimal.js globally
45
+ Decimal.set({
46
+ precision: this.config.precision,
47
+ rounding: ROUNDING_MODE_MAP[this.config.roundingMode],
48
+ toExpNeg: this.config.minExponent,
49
+ toExpPos: this.config.maxExponent,
50
+ });
51
+ }
52
+
53
+ /**
54
+ * Create a Decimal from various input types
55
+ */
56
+ from(value: DecimalLike): Decimal {
57
+ if (value instanceof Decimal) {
58
+ return value;
59
+ }
60
+
61
+ try {
62
+ if (typeof value === 'bigint') {
63
+ return new Decimal(value.toString());
64
+ }
65
+ return new Decimal(value);
66
+ } catch {
67
+ throw new InvalidDecimalError(String(value));
68
+ }
69
+ }
70
+
71
+ /**
72
+ * Check if a value is a Decimal
73
+ */
74
+ isDecimal(value: unknown): value is Decimal {
75
+ return value instanceof Decimal;
76
+ }
77
+
78
+ /**
79
+ * Convert a value to Decimal if it's numeric
80
+ */
81
+ toDecimal(value: unknown): Decimal | unknown {
82
+ if (value instanceof Decimal) {
83
+ return value;
84
+ }
85
+ if (typeof value === 'number' && !isNaN(value) && isFinite(value)) {
86
+ return new Decimal(value);
87
+ }
88
+ if (typeof value === 'string') {
89
+ const trimmed = value.trim();
90
+ if (/^-?\d+\.?\d*$/.test(trimmed) || /^-?\d*\.?\d+$/.test(trimmed)) {
91
+ try {
92
+ return new Decimal(trimmed);
93
+ } catch {
94
+ return value;
95
+ }
96
+ }
97
+ }
98
+ if (typeof value === 'bigint') {
99
+ return new Decimal(value.toString());
100
+ }
101
+ return value;
102
+ }
103
+
104
+ /**
105
+ * Addition
106
+ */
107
+ add(a: DecimalLike, b: DecimalLike): Decimal {
108
+ return this.from(a).plus(this.from(b));
109
+ }
110
+
111
+ /**
112
+ * Subtraction
113
+ */
114
+ subtract(a: DecimalLike, b: DecimalLike): Decimal {
115
+ return this.from(a).minus(this.from(b));
116
+ }
117
+
118
+ /**
119
+ * Multiplication
120
+ */
121
+ multiply(a: DecimalLike, b: DecimalLike): Decimal {
122
+ return this.from(a).times(this.from(b));
123
+ }
124
+
125
+ /**
126
+ * Division with scale
127
+ */
128
+ divide(a: DecimalLike, b: DecimalLike, scale?: number, roundingMode?: DecimalRoundingMode): Decimal {
129
+ const divisor = this.from(b);
130
+ if (divisor.isZero()) {
131
+ throw new DecimalDivisionByZeroError();
132
+ }
133
+
134
+ const result = this.from(a).dividedBy(divisor);
135
+
136
+ if (scale !== undefined) {
137
+ const rounding = roundingMode
138
+ ? ROUNDING_MODE_MAP[roundingMode]
139
+ : ROUNDING_MODE_MAP[this.config.roundingMode];
140
+ return result.toDecimalPlaces(scale, rounding);
141
+ }
142
+
143
+ return result.toDecimalPlaces(this.config.divisionScale, ROUNDING_MODE_MAP[this.config.roundingMode]);
144
+ }
145
+
146
+ /**
147
+ * Modulo
148
+ */
149
+ modulo(a: DecimalLike, b: DecimalLike): Decimal {
150
+ const divisor = this.from(b);
151
+ if (divisor.isZero()) {
152
+ throw new DecimalDivisionByZeroError();
153
+ }
154
+ return this.from(a).modulo(divisor);
155
+ }
156
+
157
+ /**
158
+ * Power
159
+ */
160
+ power(base: DecimalLike, exponent: number): Decimal {
161
+ return this.from(base).pow(exponent);
162
+ }
163
+
164
+ /**
165
+ * Negation
166
+ */
167
+ negate(value: DecimalLike): Decimal {
168
+ return this.from(value).negated();
169
+ }
170
+
171
+ /**
172
+ * Absolute value
173
+ */
174
+ abs(value: DecimalLike): Decimal {
175
+ return this.from(value).absoluteValue();
176
+ }
177
+
178
+ /**
179
+ * Round to specified decimal places
180
+ */
181
+ round(value: DecimalLike, scale: number, roundingMode?: DecimalRoundingMode): Decimal {
182
+ const rounding = roundingMode
183
+ ? ROUNDING_MODE_MAP[roundingMode]
184
+ : ROUNDING_MODE_MAP[this.config.roundingMode];
185
+ return this.from(value).toDecimalPlaces(scale, rounding);
186
+ }
187
+
188
+ /**
189
+ * Floor to specified decimal places
190
+ */
191
+ floor(value: DecimalLike, scale: number = 0): Decimal {
192
+ return this.from(value).toDecimalPlaces(scale, Decimal.ROUND_FLOOR);
193
+ }
194
+
195
+ /**
196
+ * Ceiling to specified decimal places
197
+ */
198
+ ceil(value: DecimalLike, scale: number = 0): Decimal {
199
+ return this.from(value).toDecimalPlaces(scale, Decimal.ROUND_CEIL);
200
+ }
201
+
202
+ /**
203
+ * Truncate to specified decimal places
204
+ */
205
+ truncate(value: DecimalLike, scale: number = 0): Decimal {
206
+ return this.from(value).toDecimalPlaces(scale, Decimal.ROUND_DOWN);
207
+ }
208
+
209
+ /**
210
+ * Square root
211
+ */
212
+ sqrt(value: DecimalLike): Decimal {
213
+ return this.from(value).sqrt();
214
+ }
215
+
216
+ /**
217
+ * Natural logarithm
218
+ */
219
+ ln(value: DecimalLike): Decimal {
220
+ return this.from(value).ln();
221
+ }
222
+
223
+ /**
224
+ * Base-10 logarithm
225
+ */
226
+ log10(value: DecimalLike): Decimal {
227
+ return this.from(value).log(10);
228
+ }
229
+
230
+ /**
231
+ * Comparison: returns -1, 0, or 1
232
+ */
233
+ compare(a: DecimalLike, b: DecimalLike): -1 | 0 | 1 {
234
+ const result = this.from(a).comparedTo(this.from(b));
235
+ return result as -1 | 0 | 1;
236
+ }
237
+
238
+ /**
239
+ * Equality check
240
+ */
241
+ equals(a: DecimalLike, b: DecimalLike): boolean {
242
+ return this.from(a).equals(this.from(b));
243
+ }
244
+
245
+ /**
246
+ * Greater than
247
+ */
248
+ greaterThan(a: DecimalLike, b: DecimalLike): boolean {
249
+ return this.from(a).greaterThan(this.from(b));
250
+ }
251
+
252
+ /**
253
+ * Greater than or equal
254
+ */
255
+ greaterThanOrEqual(a: DecimalLike, b: DecimalLike): boolean {
256
+ return this.from(a).greaterThanOrEqualTo(this.from(b));
257
+ }
258
+
259
+ /**
260
+ * Less than
261
+ */
262
+ lessThan(a: DecimalLike, b: DecimalLike): boolean {
263
+ return this.from(a).lessThan(this.from(b));
264
+ }
265
+
266
+ /**
267
+ * Less than or equal
268
+ */
269
+ lessThanOrEqual(a: DecimalLike, b: DecimalLike): boolean {
270
+ return this.from(a).lessThanOrEqualTo(this.from(b));
271
+ }
272
+
273
+ /**
274
+ * Check if zero
275
+ */
276
+ isZero(value: DecimalLike): boolean {
277
+ return this.from(value).isZero();
278
+ }
279
+
280
+ /**
281
+ * Check if positive
282
+ */
283
+ isPositive(value: DecimalLike): boolean {
284
+ return this.from(value).isPositive();
285
+ }
286
+
287
+ /**
288
+ * Check if negative
289
+ */
290
+ isNegative(value: DecimalLike): boolean {
291
+ return this.from(value).isNegative();
292
+ }
293
+
294
+ /**
295
+ * Check if integer
296
+ */
297
+ isInteger(value: DecimalLike): boolean {
298
+ return this.from(value).isInteger();
299
+ }
300
+
301
+ /**
302
+ * Get sign: -1, 0, or 1
303
+ */
304
+ sign(value: DecimalLike): -1 | 0 | 1 {
305
+ const d = this.from(value);
306
+ if (d.isZero()) return 0;
307
+ return d.isNegative() ? -1 : 1;
308
+ }
309
+
310
+ /**
311
+ * Get precision (total significant digits)
312
+ */
313
+ precision(value: DecimalLike): number {
314
+ return this.from(value).precision();
315
+ }
316
+
317
+ /**
318
+ * Get scale (decimal places)
319
+ */
320
+ scale(value: DecimalLike): number {
321
+ return this.from(value).decimalPlaces();
322
+ }
323
+
324
+ /**
325
+ * Convert to JavaScript number (may lose precision)
326
+ */
327
+ toNumber(value: DecimalLike): number {
328
+ return this.from(value).toNumber();
329
+ }
330
+
331
+ /**
332
+ * Convert to string
333
+ */
334
+ toString(value: DecimalLike): string {
335
+ return this.from(value).toString();
336
+ }
337
+
338
+ /**
339
+ * Convert to fixed decimal places string
340
+ */
341
+ toFixed(value: DecimalLike, scale: number): string {
342
+ return this.from(value).toFixed(scale);
343
+ }
344
+
345
+ /**
346
+ * Minimum of values
347
+ */
348
+ min(...values: DecimalLike[]): Decimal {
349
+ if (values.length === 0) {
350
+ throw new Error('min requires at least one argument');
351
+ }
352
+ return Decimal.min(...values.map(v => this.from(v)));
353
+ }
354
+
355
+ /**
356
+ * Maximum of values
357
+ */
358
+ max(...values: DecimalLike[]): Decimal {
359
+ if (values.length === 0) {
360
+ throw new Error('max requires at least one argument');
361
+ }
362
+ return Decimal.max(...values.map(v => this.from(v)));
363
+ }
364
+
365
+ /**
366
+ * Sum of values
367
+ */
368
+ sum(values: DecimalLike[]): Decimal {
369
+ return values.reduce<Decimal>((acc, v) => acc.plus(this.from(v)), new Decimal(0));
370
+ }
371
+
372
+ /**
373
+ * Average of values
374
+ */
375
+ avg(values: DecimalLike[]): Decimal {
376
+ if (values.length === 0) {
377
+ throw new Error('avg requires at least one value');
378
+ }
379
+ return this.sum(values).dividedBy(values.length);
380
+ }
381
+
382
+ /**
383
+ * Product of values
384
+ */
385
+ product(values: DecimalLike[]): Decimal {
386
+ return values.reduce<Decimal>((acc, v) => acc.times(this.from(v)), new Decimal(1));
387
+ }
388
+
389
+ /**
390
+ * Create zero
391
+ */
392
+ zero(): Decimal {
393
+ return new Decimal(0);
394
+ }
395
+
396
+ /**
397
+ * Create one
398
+ */
399
+ one(): Decimal {
400
+ return new Decimal(1);
401
+ }
402
+ }
403
+
404
+ // Export a default instance
405
+ export const decimalUtils = new DecimalUtils();
406
+
407
+ // Re-export Decimal type for convenience
408
+ export { Decimal };
@@ -0,0 +1,117 @@
1
+ import { ASTNode } from './types';
2
+ import { Parser } from './parser';
3
+
4
+ export class DependencyExtractor {
5
+ private parser: Parser;
6
+
7
+ constructor() {
8
+ this.parser = new Parser();
9
+ }
10
+
11
+ /**
12
+ * Extract all variable dependencies from an expression string
13
+ * Returns a set of variable names (without the $ or @ prefix)
14
+ */
15
+ extract(expression: string): Set<string> {
16
+ const ast = this.parser.parse(expression);
17
+ return this.extractFromNode(ast);
18
+ }
19
+
20
+ /**
21
+ * Extract dependencies from an AST node
22
+ */
23
+ extractFromNode(node: ASTNode): Set<string> {
24
+ const dependencies = new Set<string>();
25
+ this.visit(node, dependencies);
26
+ return dependencies;
27
+ }
28
+
29
+ private visit(node: ASTNode, dependencies: Set<string>): void {
30
+ switch (node.type) {
31
+ case 'VariableReference':
32
+ // Only extract $ prefixed variables (not @ context variables)
33
+ // Context variables are considered external and don't form part of the dependency graph
34
+ if (node.prefix === '$') {
35
+ dependencies.add(node.name);
36
+ }
37
+ break;
38
+
39
+ case 'BinaryOperation':
40
+ this.visit(node.left, dependencies);
41
+ this.visit(node.right, dependencies);
42
+ break;
43
+
44
+ case 'UnaryOperation':
45
+ this.visit(node.operand, dependencies);
46
+ break;
47
+
48
+ case 'ConditionalExpression':
49
+ this.visit(node.condition, dependencies);
50
+ this.visit(node.consequent, dependencies);
51
+ this.visit(node.alternate, dependencies);
52
+ break;
53
+
54
+ case 'FunctionCall':
55
+ // Visit all function arguments
56
+ for (const arg of node.arguments) {
57
+ this.visit(arg, dependencies);
58
+ }
59
+ break;
60
+
61
+ case 'MemberAccess':
62
+ // For member access like $product.price, we want to extract 'product'
63
+ // The root variable is what matters for dependency tracking
64
+ this.visitMemberRoot(node.object, dependencies);
65
+ break;
66
+
67
+ case 'IndexAccess':
68
+ // Similar to MemberAccess - extract the root variable
69
+ this.visitMemberRoot(node.object, dependencies);
70
+ // Also visit the index expression as it might contain variables
71
+ this.visit(node.index, dependencies);
72
+ break;
73
+
74
+ case 'ArrayLiteral':
75
+ for (const element of node.elements) {
76
+ this.visit(element, dependencies);
77
+ }
78
+ break;
79
+
80
+ // Literals don't have dependencies
81
+ case 'DecimalLiteral':
82
+ case 'NumberLiteral':
83
+ case 'StringLiteral':
84
+ case 'BooleanLiteral':
85
+ case 'NullLiteral':
86
+ break;
87
+ }
88
+ }
89
+
90
+ /**
91
+ * Visit the root of a member/index access chain
92
+ * For $product.price.value, we want to extract 'product'
93
+ */
94
+ private visitMemberRoot(node: ASTNode, dependencies: Set<string>): void {
95
+ switch (node.type) {
96
+ case 'VariableReference':
97
+ if (node.prefix === '$') {
98
+ dependencies.add(node.name);
99
+ }
100
+ break;
101
+
102
+ case 'MemberAccess':
103
+ this.visitMemberRoot(node.object, dependencies);
104
+ break;
105
+
106
+ case 'IndexAccess':
107
+ this.visitMemberRoot(node.object, dependencies);
108
+ this.visit(node.index, dependencies);
109
+ break;
110
+
111
+ default:
112
+ // For other node types (like function calls), visit normally
113
+ this.visit(node, dependencies);
114
+ break;
115
+ }
116
+ }
117
+ }