@the-trybe/formula-engine 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.claude/settings.local.json +6 -0
- package/PRD_FORMULA_ENGINE.md +1863 -0
- package/README.md +382 -0
- package/dist/decimal-utils.d.ts +180 -0
- package/dist/decimal-utils.js +355 -0
- package/dist/dependency-extractor.d.ts +20 -0
- package/dist/dependency-extractor.js +103 -0
- package/dist/dependency-graph.d.ts +60 -0
- package/dist/dependency-graph.js +252 -0
- package/dist/errors.d.ts +161 -0
- package/dist/errors.js +260 -0
- package/dist/evaluator.d.ts +51 -0
- package/dist/evaluator.js +494 -0
- package/dist/formula-engine.d.ts +79 -0
- package/dist/formula-engine.js +355 -0
- package/dist/functions.d.ts +3 -0
- package/dist/functions.js +720 -0
- package/dist/index.d.ts +10 -0
- package/dist/index.js +61 -0
- package/dist/lexer.d.ts +25 -0
- package/dist/lexer.js +357 -0
- package/dist/parser.d.ts +32 -0
- package/dist/parser.js +372 -0
- package/dist/types.d.ts +228 -0
- package/dist/types.js +62 -0
- package/jest.config.js +23 -0
- package/package.json +35 -0
- package/src/decimal-utils.ts +408 -0
- package/src/dependency-extractor.ts +117 -0
- package/src/dependency-graph.test.ts +238 -0
- package/src/dependency-graph.ts +288 -0
- package/src/errors.ts +296 -0
- package/src/evaluator.ts +604 -0
- package/src/formula-engine.test.ts +660 -0
- package/src/formula-engine.ts +430 -0
- package/src/functions.ts +770 -0
- package/src/index.ts +103 -0
- package/src/lexer.test.ts +288 -0
- package/src/lexer.ts +394 -0
- package/src/parser.test.ts +349 -0
- package/src/parser.ts +449 -0
- package/src/types.ts +347 -0
- package/tsconfig.json +29 -0
package/dist/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
|
+
}
|