@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,252 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.DependencyGraphBuilder = exports.DependencyGraph = void 0;
|
|
4
|
+
const dependency_extractor_1 = require("./dependency-extractor");
|
|
5
|
+
const errors_1 = require("./errors");
|
|
6
|
+
class DependencyGraph {
|
|
7
|
+
constructor() {
|
|
8
|
+
this.nodes = new Set();
|
|
9
|
+
this.edges = new Map();
|
|
10
|
+
}
|
|
11
|
+
/**
|
|
12
|
+
* Add a node to the graph
|
|
13
|
+
*/
|
|
14
|
+
addNode(id) {
|
|
15
|
+
this.nodes.add(id);
|
|
16
|
+
if (!this.edges.has(id)) {
|
|
17
|
+
this.edges.set(id, new Set());
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* Add an edge from source to target (source depends on target)
|
|
22
|
+
*/
|
|
23
|
+
addEdge(source, target) {
|
|
24
|
+
this.addNode(source);
|
|
25
|
+
this.addNode(target);
|
|
26
|
+
this.edges.get(source).add(target);
|
|
27
|
+
}
|
|
28
|
+
/**
|
|
29
|
+
* Check if the graph has any cycles
|
|
30
|
+
*/
|
|
31
|
+
hasCycles() {
|
|
32
|
+
try {
|
|
33
|
+
this.topologicalSort();
|
|
34
|
+
return false;
|
|
35
|
+
}
|
|
36
|
+
catch (error) {
|
|
37
|
+
return error instanceof errors_1.CircularDependencyError;
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
/**
|
|
41
|
+
* Get all root nodes (nodes with no dependencies)
|
|
42
|
+
*/
|
|
43
|
+
getRoots() {
|
|
44
|
+
const roots = new Set();
|
|
45
|
+
for (const node of this.nodes) {
|
|
46
|
+
const deps = this.edges.get(node) || new Set();
|
|
47
|
+
if (deps.size === 0) {
|
|
48
|
+
roots.add(node);
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
return roots;
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* Get all nodes that depend on the given node
|
|
55
|
+
*/
|
|
56
|
+
getDependents(nodeId) {
|
|
57
|
+
const dependents = new Set();
|
|
58
|
+
for (const [node, deps] of this.edges) {
|
|
59
|
+
if (deps.has(nodeId)) {
|
|
60
|
+
dependents.add(node);
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
return dependents;
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* Get direct dependencies of a node
|
|
67
|
+
*/
|
|
68
|
+
getDependencies(nodeId) {
|
|
69
|
+
return this.edges.get(nodeId) || new Set();
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Get all transitive dependencies of a node
|
|
73
|
+
*/
|
|
74
|
+
getTransitiveDependencies(nodeId) {
|
|
75
|
+
const visited = new Set();
|
|
76
|
+
const result = new Set();
|
|
77
|
+
this.collectTransitiveDependencies(nodeId, visited, result);
|
|
78
|
+
result.delete(nodeId); // Don't include the node itself
|
|
79
|
+
return result;
|
|
80
|
+
}
|
|
81
|
+
collectTransitiveDependencies(nodeId, visited, result) {
|
|
82
|
+
if (visited.has(nodeId)) {
|
|
83
|
+
return;
|
|
84
|
+
}
|
|
85
|
+
visited.add(nodeId);
|
|
86
|
+
const deps = this.edges.get(nodeId);
|
|
87
|
+
if (deps) {
|
|
88
|
+
for (const dep of deps) {
|
|
89
|
+
result.add(dep);
|
|
90
|
+
this.collectTransitiveDependencies(dep, visited, result);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
/**
|
|
95
|
+
* Perform topological sort using Kahn's algorithm
|
|
96
|
+
* Returns nodes in evaluation order (dependencies first)
|
|
97
|
+
* Throws CircularDependencyError if a cycle is detected
|
|
98
|
+
*/
|
|
99
|
+
topologicalSort() {
|
|
100
|
+
// Calculate in-degree for each node
|
|
101
|
+
const inDegree = new Map();
|
|
102
|
+
for (const node of this.nodes) {
|
|
103
|
+
inDegree.set(node, 0);
|
|
104
|
+
}
|
|
105
|
+
// Count incoming edges for each node
|
|
106
|
+
for (const [, deps] of this.edges) {
|
|
107
|
+
for (const dep of deps) {
|
|
108
|
+
if (this.nodes.has(dep)) {
|
|
109
|
+
// We need to count how many nodes depend on this one
|
|
110
|
+
// This is actually reverse - we need dependents, not dependencies
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
// Actually, we need to build a reverse graph for proper topological sort
|
|
115
|
+
// edges: node -> dependencies means we need to evaluate dependencies first
|
|
116
|
+
// So the "in-degree" should count how many times a node is depended upon
|
|
117
|
+
const dependents = new Map();
|
|
118
|
+
for (const node of this.nodes) {
|
|
119
|
+
dependents.set(node, new Set());
|
|
120
|
+
}
|
|
121
|
+
for (const [node, deps] of this.edges) {
|
|
122
|
+
for (const dep of deps) {
|
|
123
|
+
if (dependents.has(dep)) {
|
|
124
|
+
dependents.get(dep).add(node);
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
// Calculate in-degree (number of dependencies)
|
|
129
|
+
for (const [node, deps] of this.edges) {
|
|
130
|
+
// Only count dependencies that are actual nodes in our graph
|
|
131
|
+
const validDeps = [...deps].filter(d => this.nodes.has(d));
|
|
132
|
+
inDegree.set(node, validDeps.length);
|
|
133
|
+
}
|
|
134
|
+
// Start with nodes that have no dependencies
|
|
135
|
+
const queue = [];
|
|
136
|
+
for (const [node, degree] of inDegree) {
|
|
137
|
+
if (degree === 0) {
|
|
138
|
+
queue.push(node);
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
const result = [];
|
|
142
|
+
const visited = new Set();
|
|
143
|
+
while (queue.length > 0) {
|
|
144
|
+
const node = queue.shift();
|
|
145
|
+
if (visited.has(node))
|
|
146
|
+
continue;
|
|
147
|
+
visited.add(node);
|
|
148
|
+
result.push(node);
|
|
149
|
+
// For each node that depends on this one, decrease its in-degree
|
|
150
|
+
const deps = dependents.get(node) || new Set();
|
|
151
|
+
for (const dependent of deps) {
|
|
152
|
+
const newDegree = (inDegree.get(dependent) || 0) - 1;
|
|
153
|
+
inDegree.set(dependent, newDegree);
|
|
154
|
+
if (newDegree === 0 && !visited.has(dependent)) {
|
|
155
|
+
queue.push(dependent);
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
// Check if all nodes were visited
|
|
160
|
+
if (result.length !== this.nodes.size) {
|
|
161
|
+
// There's a cycle - find it
|
|
162
|
+
const cycle = this.findCycle();
|
|
163
|
+
throw new errors_1.CircularDependencyError(cycle, [...this.nodes].filter(n => !visited.has(n)));
|
|
164
|
+
}
|
|
165
|
+
return result;
|
|
166
|
+
}
|
|
167
|
+
/**
|
|
168
|
+
* Find a cycle in the graph using DFS
|
|
169
|
+
*/
|
|
170
|
+
findCycle() {
|
|
171
|
+
const visited = new Set();
|
|
172
|
+
const recursionStack = new Set();
|
|
173
|
+
const path = [];
|
|
174
|
+
const dfs = (node) => {
|
|
175
|
+
visited.add(node);
|
|
176
|
+
recursionStack.add(node);
|
|
177
|
+
path.push(node);
|
|
178
|
+
const deps = this.edges.get(node) || new Set();
|
|
179
|
+
for (const dep of deps) {
|
|
180
|
+
if (!this.nodes.has(dep))
|
|
181
|
+
continue;
|
|
182
|
+
if (!visited.has(dep)) {
|
|
183
|
+
const cycle = dfs(dep);
|
|
184
|
+
if (cycle)
|
|
185
|
+
return cycle;
|
|
186
|
+
}
|
|
187
|
+
else if (recursionStack.has(dep)) {
|
|
188
|
+
// Found a cycle - extract it from the path
|
|
189
|
+
const cycleStart = path.indexOf(dep);
|
|
190
|
+
const cycle = path.slice(cycleStart);
|
|
191
|
+
cycle.push(dep); // Close the cycle
|
|
192
|
+
return cycle;
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
path.pop();
|
|
196
|
+
recursionStack.delete(node);
|
|
197
|
+
return null;
|
|
198
|
+
};
|
|
199
|
+
for (const node of this.nodes) {
|
|
200
|
+
if (!visited.has(node)) {
|
|
201
|
+
const cycle = dfs(node);
|
|
202
|
+
if (cycle)
|
|
203
|
+
return cycle;
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
return [];
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
exports.DependencyGraph = DependencyGraph;
|
|
210
|
+
/**
|
|
211
|
+
* Build a dependency graph from formula definitions
|
|
212
|
+
*/
|
|
213
|
+
class DependencyGraphBuilder {
|
|
214
|
+
constructor() {
|
|
215
|
+
this.extractor = new dependency_extractor_1.DependencyExtractor();
|
|
216
|
+
}
|
|
217
|
+
/**
|
|
218
|
+
* Build a dependency graph from formula definitions
|
|
219
|
+
*/
|
|
220
|
+
build(formulas) {
|
|
221
|
+
const graph = new DependencyGraph();
|
|
222
|
+
// Create a map of formula IDs for quick lookup
|
|
223
|
+
const formulaIds = new Set(formulas.map(f => f.id));
|
|
224
|
+
// Add all formula IDs as nodes
|
|
225
|
+
for (const formula of formulas) {
|
|
226
|
+
graph.addNode(formula.id);
|
|
227
|
+
}
|
|
228
|
+
// Extract dependencies and add edges
|
|
229
|
+
for (const formula of formulas) {
|
|
230
|
+
const deps = formula.dependencies
|
|
231
|
+
? new Set(formula.dependencies)
|
|
232
|
+
: this.extractor.extract(formula.expression);
|
|
233
|
+
for (const dep of deps) {
|
|
234
|
+
// Only add edges for dependencies that are other formulas
|
|
235
|
+
// (not external variables)
|
|
236
|
+
if (formulaIds.has(dep)) {
|
|
237
|
+
graph.addEdge(formula.id, dep);
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
return graph;
|
|
242
|
+
}
|
|
243
|
+
/**
|
|
244
|
+
* Get the evaluation order for a set of formulas
|
|
245
|
+
*/
|
|
246
|
+
getEvaluationOrder(formulas) {
|
|
247
|
+
const graph = this.build(formulas);
|
|
248
|
+
return graph.topologicalSort();
|
|
249
|
+
}
|
|
250
|
+
}
|
|
251
|
+
exports.DependencyGraphBuilder = DependencyGraphBuilder;
|
|
252
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dependency-graph.js","sourceRoot":"","sources":["../src/dependency-graph.ts"],"names":[],"mappings":"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","sourcesContent":["import { DependencyGraph as IDependencyGraph, FormulaDefinition } from './types';\nimport { DependencyExtractor } from './dependency-extractor';\nimport { CircularDependencyError } from './errors';\n\nexport class DependencyGraph implements IDependencyGraph {\n  nodes: Set<string>;\n  edges: Map<string, Set<string>>;\n\n  constructor() {\n    this.nodes = new Set();\n    this.edges = new Map();\n  }\n\n  /**\n   * Add a node to the graph\n   */\n  addNode(id: string): void {\n    this.nodes.add(id);\n    if (!this.edges.has(id)) {\n      this.edges.set(id, new Set());\n    }\n  }\n\n  /**\n   * Add an edge from source to target (source depends on target)\n   */\n  addEdge(source: string, target: string): void {\n    this.addNode(source);\n    this.addNode(target);\n    this.edges.get(source)!.add(target);\n  }\n\n  /**\n   * Check if the graph has any cycles\n   */\n  hasCycles(): boolean {\n    try {\n      this.topologicalSort();\n      return false;\n    } catch (error) {\n      return error instanceof CircularDependencyError;\n    }\n  }\n\n  /**\n   * Get all root nodes (nodes with no dependencies)\n   */\n  getRoots(): Set<string> {\n    const roots = new Set<string>();\n    for (const node of this.nodes) {\n      const deps = this.edges.get(node) || new Set();\n      if (deps.size === 0) {\n        roots.add(node);\n      }\n    }\n    return roots;\n  }\n\n  /**\n   * Get all nodes that depend on the given node\n   */\n  getDependents(nodeId: string): Set<string> {\n    const dependents = new Set<string>();\n    for (const [node, deps] of this.edges) {\n      if (deps.has(nodeId)) {\n        dependents.add(node);\n      }\n    }\n    return dependents;\n  }\n\n  /**\n   * Get direct dependencies of a node\n   */\n  getDependencies(nodeId: string): Set<string> {\n    return this.edges.get(nodeId) || new Set();\n  }\n\n  /**\n   * Get all transitive dependencies of a node\n   */\n  getTransitiveDependencies(nodeId: string): Set<string> {\n    const visited = new Set<string>();\n    const result = new Set<string>();\n    this.collectTransitiveDependencies(nodeId, visited, result);\n    result.delete(nodeId); // Don't include the node itself\n    return result;\n  }\n\n  private collectTransitiveDependencies(\n    nodeId: string,\n    visited: Set<string>,\n    result: Set<string>\n  ): void {\n    if (visited.has(nodeId)) {\n      return;\n    }\n    visited.add(nodeId);\n\n    const deps = this.edges.get(nodeId);\n    if (deps) {\n      for (const dep of deps) {\n        result.add(dep);\n        this.collectTransitiveDependencies(dep, visited, result);\n      }\n    }\n  }\n\n  /**\n   * Perform topological sort using Kahn's algorithm\n   * Returns nodes in evaluation order (dependencies first)\n   * Throws CircularDependencyError if a cycle is detected\n   */\n  topologicalSort(): string[] {\n    // Calculate in-degree for each node\n    const inDegree = new Map<string, number>();\n    for (const node of this.nodes) {\n      inDegree.set(node, 0);\n    }\n\n    // Count incoming edges for each node\n    for (const [, deps] of this.edges) {\n      for (const dep of deps) {\n        if (this.nodes.has(dep)) {\n          // We need to count how many nodes depend on this one\n          // This is actually reverse - we need dependents, not dependencies\n        }\n      }\n    }\n\n    // Actually, we need to build a reverse graph for proper topological sort\n    // edges: node -> dependencies means we need to evaluate dependencies first\n    // So the \"in-degree\" should count how many times a node is depended upon\n\n    const dependents = new Map<string, Set<string>>();\n    for (const node of this.nodes) {\n      dependents.set(node, new Set());\n    }\n\n    for (const [node, deps] of this.edges) {\n      for (const dep of deps) {\n        if (dependents.has(dep)) {\n          dependents.get(dep)!.add(node);\n        }\n      }\n    }\n\n    // Calculate in-degree (number of dependencies)\n    for (const [node, deps] of this.edges) {\n      // Only count dependencies that are actual nodes in our graph\n      const validDeps = [...deps].filter(d => this.nodes.has(d));\n      inDegree.set(node, validDeps.length);\n    }\n\n    // Start with nodes that have no dependencies\n    const queue: string[] = [];\n    for (const [node, degree] of inDegree) {\n      if (degree === 0) {\n        queue.push(node);\n      }\n    }\n\n    const result: string[] = [];\n    const visited = new Set<string>();\n\n    while (queue.length > 0) {\n      const node = queue.shift()!;\n      if (visited.has(node)) continue;\n\n      visited.add(node);\n      result.push(node);\n\n      // For each node that depends on this one, decrease its in-degree\n      const deps = dependents.get(node) || new Set();\n      for (const dependent of deps) {\n        const newDegree = (inDegree.get(dependent) || 0) - 1;\n        inDegree.set(dependent, newDegree);\n        if (newDegree === 0 && !visited.has(dependent)) {\n          queue.push(dependent);\n        }\n      }\n    }\n\n    // Check if all nodes were visited\n    if (result.length !== this.nodes.size) {\n      // There's a cycle - find it\n      const cycle = this.findCycle();\n      throw new CircularDependencyError(cycle, [...this.nodes].filter(n => !visited.has(n)));\n    }\n\n    return result;\n  }\n\n  /**\n   * Find a cycle in the graph using DFS\n   */\n  private findCycle(): string[] {\n    const visited = new Set<string>();\n    const recursionStack = new Set<string>();\n    const path: string[] = [];\n\n    const dfs = (node: string): string[] | null => {\n      visited.add(node);\n      recursionStack.add(node);\n      path.push(node);\n\n      const deps = this.edges.get(node) || new Set();\n      for (const dep of deps) {\n        if (!this.nodes.has(dep)) continue;\n\n        if (!visited.has(dep)) {\n          const cycle = dfs(dep);\n          if (cycle) return cycle;\n        } else if (recursionStack.has(dep)) {\n          // Found a cycle - extract it from the path\n          const cycleStart = path.indexOf(dep);\n          const cycle = path.slice(cycleStart);\n          cycle.push(dep); // Close the cycle\n          return cycle;\n        }\n      }\n\n      path.pop();\n      recursionStack.delete(node);\n      return null;\n    };\n\n    for (const node of this.nodes) {\n      if (!visited.has(node)) {\n        const cycle = dfs(node);\n        if (cycle) return cycle;\n      }\n    }\n\n    return [];\n  }\n}\n\n/**\n * Build a dependency graph from formula definitions\n */\nexport class DependencyGraphBuilder {\n  private extractor: DependencyExtractor;\n\n  constructor() {\n    this.extractor = new DependencyExtractor();\n  }\n\n  /**\n   * Build a dependency graph from formula definitions\n   */\n  build(formulas: FormulaDefinition[]): DependencyGraph {\n    const graph = new DependencyGraph();\n\n    // Create a map of formula IDs for quick lookup\n    const formulaIds = new Set(formulas.map(f => f.id));\n\n    // Add all formula IDs as nodes\n    for (const formula of formulas) {\n      graph.addNode(formula.id);\n    }\n\n    // Extract dependencies and add edges\n    for (const formula of formulas) {\n      const deps = formula.dependencies\n        ? new Set(formula.dependencies)\n        : this.extractor.extract(formula.expression);\n\n      for (const dep of deps) {\n        // Only add edges for dependencies that are other formulas\n        // (not external variables)\n        if (formulaIds.has(dep)) {\n          graph.addEdge(formula.id, dep);\n        }\n      }\n    }\n\n    return graph;\n  }\n\n  /**\n   * Get the evaluation order for a set of formulas\n   */\n  getEvaluationOrder(formulas: FormulaDefinition[]): string[] {\n    const graph = this.build(formulas);\n    return graph.topologicalSort();\n  }\n}\n"]}
|
package/dist/errors.d.ts
ADDED
|
@@ -0,0 +1,161 @@
|
|
|
1
|
+
import { ValueType } from './types';
|
|
2
|
+
export type ErrorCategory = 'PARSE' | 'VALIDATION' | 'EVALUATION' | 'CONFIGURATION';
|
|
3
|
+
export declare abstract class FormulaEngineError extends Error {
|
|
4
|
+
abstract readonly code: string;
|
|
5
|
+
abstract readonly category: ErrorCategory;
|
|
6
|
+
constructor(message: string);
|
|
7
|
+
}
|
|
8
|
+
export declare class SyntaxError extends FormulaEngineError {
|
|
9
|
+
position: number;
|
|
10
|
+
line: number;
|
|
11
|
+
column: number;
|
|
12
|
+
expression: string;
|
|
13
|
+
readonly code = "PARSE_SYNTAX_ERROR";
|
|
14
|
+
readonly category: ErrorCategory;
|
|
15
|
+
constructor(message: string, position: number, line: number, column: number, expression: string);
|
|
16
|
+
}
|
|
17
|
+
export declare class UnexpectedTokenError extends FormulaEngineError {
|
|
18
|
+
token: string;
|
|
19
|
+
expected: string[];
|
|
20
|
+
position: number;
|
|
21
|
+
readonly code = "PARSE_UNEXPECTED_TOKEN";
|
|
22
|
+
readonly category: ErrorCategory;
|
|
23
|
+
constructor(token: string, expected: string[], position: number);
|
|
24
|
+
}
|
|
25
|
+
export declare class UnterminatedStringError extends FormulaEngineError {
|
|
26
|
+
readonly code = "PARSE_UNTERMINATED_STRING";
|
|
27
|
+
readonly category: ErrorCategory;
|
|
28
|
+
constructor(position: number);
|
|
29
|
+
}
|
|
30
|
+
export declare class InvalidNumberError extends FormulaEngineError {
|
|
31
|
+
readonly code = "PARSE_INVALID_NUMBER";
|
|
32
|
+
readonly category: ErrorCategory;
|
|
33
|
+
constructor(value: string, position: number);
|
|
34
|
+
}
|
|
35
|
+
export declare class CircularDependencyError extends FormulaEngineError {
|
|
36
|
+
cycle: string[];
|
|
37
|
+
involvedFormulas: string[];
|
|
38
|
+
readonly code = "VALIDATION_CIRCULAR_DEPENDENCY";
|
|
39
|
+
readonly category: ErrorCategory;
|
|
40
|
+
constructor(cycle: string[], involvedFormulas: string[]);
|
|
41
|
+
}
|
|
42
|
+
export declare class UndefinedVariableError extends FormulaEngineError {
|
|
43
|
+
variableName: string;
|
|
44
|
+
expression: string;
|
|
45
|
+
readonly code = "VALIDATION_UNDEFINED_VARIABLE";
|
|
46
|
+
readonly category: ErrorCategory;
|
|
47
|
+
constructor(variableName: string, expression: string);
|
|
48
|
+
}
|
|
49
|
+
export declare class UndefinedFunctionError extends FormulaEngineError {
|
|
50
|
+
functionName: string;
|
|
51
|
+
readonly code = "VALIDATION_UNDEFINED_FUNCTION";
|
|
52
|
+
readonly category: ErrorCategory;
|
|
53
|
+
constructor(functionName: string);
|
|
54
|
+
}
|
|
55
|
+
export declare class DuplicateFormulaError extends FormulaEngineError {
|
|
56
|
+
formulaId: string;
|
|
57
|
+
readonly code = "VALIDATION_DUPLICATE_FORMULA";
|
|
58
|
+
readonly category: ErrorCategory;
|
|
59
|
+
constructor(formulaId: string);
|
|
60
|
+
}
|
|
61
|
+
export declare class DivisionByZeroError extends FormulaEngineError {
|
|
62
|
+
readonly code = "EVAL_DIVISION_BY_ZERO";
|
|
63
|
+
readonly category: ErrorCategory;
|
|
64
|
+
constructor();
|
|
65
|
+
}
|
|
66
|
+
export declare class TypeMismatchError extends FormulaEngineError {
|
|
67
|
+
expected: ValueType | ValueType[] | string;
|
|
68
|
+
actual: ValueType | string;
|
|
69
|
+
context: string;
|
|
70
|
+
readonly code = "EVAL_TYPE_MISMATCH";
|
|
71
|
+
readonly category: ErrorCategory;
|
|
72
|
+
constructor(expected: ValueType | ValueType[] | string, actual: ValueType | string, context: string);
|
|
73
|
+
}
|
|
74
|
+
export declare class ArgumentCountError extends FormulaEngineError {
|
|
75
|
+
functionName: string;
|
|
76
|
+
expected: {
|
|
77
|
+
min: number;
|
|
78
|
+
max: number;
|
|
79
|
+
};
|
|
80
|
+
actual: number;
|
|
81
|
+
readonly code = "EVAL_ARGUMENT_COUNT";
|
|
82
|
+
readonly category: ErrorCategory;
|
|
83
|
+
constructor(functionName: string, expected: {
|
|
84
|
+
min: number;
|
|
85
|
+
max: number;
|
|
86
|
+
}, actual: number);
|
|
87
|
+
}
|
|
88
|
+
export declare class InvalidOperationError extends FormulaEngineError {
|
|
89
|
+
operator: string;
|
|
90
|
+
operandTypes: string[];
|
|
91
|
+
readonly code = "EVAL_INVALID_OPERATION";
|
|
92
|
+
readonly category: ErrorCategory;
|
|
93
|
+
constructor(operator: string, operandTypes: string[]);
|
|
94
|
+
}
|
|
95
|
+
export declare class PropertyAccessError extends FormulaEngineError {
|
|
96
|
+
property: string;
|
|
97
|
+
objectType: string;
|
|
98
|
+
readonly code = "EVAL_PROPERTY_ACCESS";
|
|
99
|
+
readonly category: ErrorCategory;
|
|
100
|
+
constructor(property: string, objectType: string);
|
|
101
|
+
}
|
|
102
|
+
export declare class IndexAccessError extends FormulaEngineError {
|
|
103
|
+
index: unknown;
|
|
104
|
+
objectType: string;
|
|
105
|
+
readonly code = "EVAL_INDEX_ACCESS";
|
|
106
|
+
readonly category: ErrorCategory;
|
|
107
|
+
constructor(index: unknown, objectType: string);
|
|
108
|
+
}
|
|
109
|
+
export declare class DecimalError extends FormulaEngineError {
|
|
110
|
+
readonly code: string;
|
|
111
|
+
readonly category: ErrorCategory;
|
|
112
|
+
constructor(message: string);
|
|
113
|
+
}
|
|
114
|
+
export declare class DecimalOverflowError extends DecimalError {
|
|
115
|
+
value: string;
|
|
116
|
+
maxExponent: number;
|
|
117
|
+
readonly code: string;
|
|
118
|
+
constructor(value: string, maxExponent: number);
|
|
119
|
+
}
|
|
120
|
+
export declare class DecimalUnderflowError extends DecimalError {
|
|
121
|
+
value: string;
|
|
122
|
+
minExponent: number;
|
|
123
|
+
readonly code: string;
|
|
124
|
+
constructor(value: string, minExponent: number);
|
|
125
|
+
}
|
|
126
|
+
export declare class DecimalDivisionByZeroError extends DecimalError {
|
|
127
|
+
readonly code: string;
|
|
128
|
+
constructor();
|
|
129
|
+
}
|
|
130
|
+
export declare class InvalidDecimalError extends DecimalError {
|
|
131
|
+
input: string;
|
|
132
|
+
readonly code: string;
|
|
133
|
+
constructor(input: string);
|
|
134
|
+
}
|
|
135
|
+
export declare class ConfigurationError extends FormulaEngineError {
|
|
136
|
+
readonly code = "CONFIGURATION_ERROR";
|
|
137
|
+
readonly category: ErrorCategory;
|
|
138
|
+
constructor(message: string);
|
|
139
|
+
}
|
|
140
|
+
export declare class SecurityError extends FormulaEngineError {
|
|
141
|
+
readonly code: string;
|
|
142
|
+
readonly category: ErrorCategory;
|
|
143
|
+
constructor(message: string);
|
|
144
|
+
}
|
|
145
|
+
export declare class MaxIterationsError extends SecurityError {
|
|
146
|
+
readonly code: string;
|
|
147
|
+
constructor(limit: number);
|
|
148
|
+
}
|
|
149
|
+
export declare class MaxRecursionError extends SecurityError {
|
|
150
|
+
readonly code: string;
|
|
151
|
+
constructor(limit: number);
|
|
152
|
+
}
|
|
153
|
+
export declare class MaxExpressionLengthError extends SecurityError {
|
|
154
|
+
readonly code: string;
|
|
155
|
+
constructor(length: number, limit: number);
|
|
156
|
+
}
|
|
157
|
+
export declare class GeneralFormulaError extends FormulaEngineError {
|
|
158
|
+
readonly code: string;
|
|
159
|
+
readonly category: ErrorCategory;
|
|
160
|
+
constructor(message: string);
|
|
161
|
+
}
|