@divmain/jdm-asm 0.2.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 (397) hide show
  1. package/.github/workflows/ci.yml +53 -0
  2. package/.oxfmtrc.json +16 -0
  3. package/.oxlintrc.json +183 -0
  4. package/AGENTS.md +81 -0
  5. package/README.md +769 -0
  6. package/asconfig.json +23 -0
  7. package/benchmarks/fixtures.ts +111 -0
  8. package/benchmarks/input-fixtures.ts +80 -0
  9. package/benchmarks/run.ts +913 -0
  10. package/benchmarks/worker-pool.ts +223 -0
  11. package/benchmarks/worker.ts +374 -0
  12. package/dist/index.d.ts +996 -0
  13. package/dist/index.js +12239 -0
  14. package/dist/index.js.map +1 -0
  15. package/package.json +49 -0
  16. package/scripts/run-all-tests.ts +220 -0
  17. package/src/compiler/EXPRESSION_SUBSETS.md +228 -0
  18. package/src/compiler/asc-compiler.ts +315 -0
  19. package/src/compiler/ast-types.ts +215 -0
  20. package/src/compiler/build.ts +56 -0
  21. package/src/compiler/cache.ts +414 -0
  22. package/src/compiler/code-generators.ts +211 -0
  23. package/src/compiler/codegen/index.ts +15 -0
  24. package/src/compiler/codegen/js-marshal.ts +999 -0
  25. package/src/compiler/codegen/js-validation.ts +243 -0
  26. package/src/compiler/codegen.ts +19 -0
  27. package/src/compiler/compile-time-validation.ts +507 -0
  28. package/src/compiler/cst-visitor.ts +434 -0
  29. package/src/compiler/errors.ts +227 -0
  30. package/src/compiler/expression-parser.ts +536 -0
  31. package/src/compiler/graph.ts +197 -0
  32. package/src/compiler/index.ts +199 -0
  33. package/src/compiler/input-validation.ts +33 -0
  34. package/src/compiler/marshal-gen.ts +21 -0
  35. package/src/compiler/nodes/context-resolvers.ts +197 -0
  36. package/src/compiler/nodes/decision-table.ts +507 -0
  37. package/src/compiler/nodes/decision.ts +292 -0
  38. package/src/compiler/nodes/expression-compiler.ts +526 -0
  39. package/src/compiler/nodes/expression.ts +425 -0
  40. package/src/compiler/nodes/function.ts +316 -0
  41. package/src/compiler/nodes/input.ts +60 -0
  42. package/src/compiler/nodes/switch.ts +547 -0
  43. package/src/compiler/optimizer.ts +948 -0
  44. package/src/compiler/orchestrator.ts +352 -0
  45. package/src/compiler/parser.ts +115 -0
  46. package/src/compiler/result-selection.ts +161 -0
  47. package/src/compiler/runtime/index.ts +26 -0
  48. package/src/compiler/runtime-codegen.ts +211 -0
  49. package/src/compiler/runtime-validation-codegen.ts +294 -0
  50. package/src/compiler/runtime.ts +452 -0
  51. package/src/compiler/schema.ts +245 -0
  52. package/src/compiler/switch-branch-detection.ts +92 -0
  53. package/src/compiler/types.ts +136 -0
  54. package/src/compiler/unary-ast-transforms.ts +148 -0
  55. package/src/compiler/unary-parser.ts +301 -0
  56. package/src/compiler/unary-transform.ts +161 -0
  57. package/src/compiler/utils.ts +27 -0
  58. package/src/compiler/virtual-fs.ts +90 -0
  59. package/src/compiler/wasm-instantiate.ts +127 -0
  60. package/src/index.ts +1 -0
  61. package/src/runtime/arrays.ts +579 -0
  62. package/src/runtime/context.ts +189 -0
  63. package/src/runtime/expressions.ts +1811 -0
  64. package/src/runtime/index.ts +8 -0
  65. package/src/runtime/memory.ts +607 -0
  66. package/src/runtime/strings.ts +260 -0
  67. package/src/runtime/tables.ts +96 -0
  68. package/src/runtime/tsconfig.json +4 -0
  69. package/src/runtime/values.ts +209 -0
  70. package/test-data/README.md +83 -0
  71. package/test-data/decision-tables/basic/8k.json +87992 -0
  72. package/test-data/decision-tables/basic/affiliate-commission-calculator.json +228 -0
  73. package/test-data/decision-tables/basic/airline-loyalty-points-calculations.json +285 -0
  74. package/test-data/decision-tables/basic/airline-upgrade-eligibility.json +466 -0
  75. package/test-data/decision-tables/basic/auto-insurance-premium-calculator.json +412 -0
  76. package/test-data/decision-tables/basic/booking-personalization-system.json +553 -0
  77. package/test-data/decision-tables/basic/care-team-assignment-system.json +585 -0
  78. package/test-data/decision-tables/basic/claim-validation-system.json +307 -0
  79. package/test-data/decision-tables/basic/clinical-lab-result-interpreter.json +433 -0
  80. package/test-data/decision-tables/basic/clinical-treatment-protocol.json +474 -0
  81. package/test-data/decision-tables/basic/credit-limit-adjustment.json +479 -0
  82. package/test-data/decision-tables/basic/customer-eligibility-engine.json +551 -0
  83. package/test-data/decision-tables/basic/customer-lifetime-value.json +200 -0
  84. package/test-data/decision-tables/basic/customer-onboarding-kyc-verification.json +611 -0
  85. package/test-data/decision-tables/basic/customer-service-escalation.json +191 -0
  86. package/test-data/decision-tables/basic/decision-table-discounts.json +168 -0
  87. package/test-data/decision-tables/basic/decision-table-shipping.json +398 -0
  88. package/test-data/decision-tables/basic/delivery-route-optimizer.json +271 -0
  89. package/test-data/decision-tables/basic/device-compatibility-checker.json +303 -0
  90. package/test-data/decision-tables/basic/disaster-relief-fund-allocation.json +296 -0
  91. package/test-data/decision-tables/basic/dynamic-fx-rate-pricing-system.json +237 -0
  92. package/test-data/decision-tables/basic/dynamic-marketplace-comission-calculator.json +242 -0
  93. package/test-data/decision-tables/basic/dynamic-shipping-cost-calculator.json +378 -0
  94. package/test-data/decision-tables/basic/dynamic-tarrif-engine.json +289 -0
  95. package/test-data/decision-tables/basic/dynamic-ticket-pricing.json +325 -0
  96. package/test-data/decision-tables/basic/empty-column-with-space.json +100 -0
  97. package/test-data/decision-tables/basic/empty-column-without-space.json +100 -0
  98. package/test-data/decision-tables/basic/environment-compliance-assessment.json +386 -0
  99. package/test-data/decision-tables/basic/expression-table-map.json +313 -0
  100. package/test-data/decision-tables/basic/flash-sale-eligibility.json +366 -0
  101. package/test-data/decision-tables/basic/flight-dispatch-decision-system.json +455 -0
  102. package/test-data/decision-tables/basic/flight-rebooking-fee-calculator.json +406 -0
  103. package/test-data/decision-tables/basic/government-assistance.json +299 -0
  104. package/test-data/decision-tables/basic/grant-funding-distribution.json +307 -0
  105. package/test-data/decision-tables/basic/hazardous-materials-management-system.json +414 -0
  106. package/test-data/decision-tables/basic/immigration-eligibility-evaluator.json +765 -0
  107. package/test-data/decision-tables/basic/import-duties-calculator.json +318 -0
  108. package/test-data/decision-tables/basic/insurance-agent-commission.json +228 -0
  109. package/test-data/decision-tables/basic/insurance-coverage-calculator.json +362 -0
  110. package/test-data/decision-tables/basic/insurance-underwriting-risk.json +321 -0
  111. package/test-data/decision-tables/basic/international-roaming-policy-manager.json +199 -0
  112. package/test-data/decision-tables/basic/legacy-plan-management.json +434 -0
  113. package/test-data/decision-tables/basic/marketplace-listing-verification-system.json +334 -0
  114. package/test-data/decision-tables/basic/medication-dosage-calculator.json +318 -0
  115. package/test-data/decision-tables/basic/merch-bags.json +171 -0
  116. package/test-data/decision-tables/basic/municipal-permit-evaluation-system.json +364 -0
  117. package/test-data/decision-tables/basic/mvno-partner-enablement.json +313 -0
  118. package/test-data/decision-tables/basic/partner-revenue-sharing.json +244 -0
  119. package/test-data/decision-tables/basic/payment-routing-and-fee-calculator.json +475 -0
  120. package/test-data/decision-tables/basic/policy-discount-calculator.json +307 -0
  121. package/test-data/decision-tables/basic/policy-eligibility-analyzer.json +299 -0
  122. package/test-data/decision-tables/basic/product-listing-scoring.json +358 -0
  123. package/test-data/decision-tables/basic/realtime-fraud-detection.json +235 -0
  124. package/test-data/decision-tables/basic/regional-compliance-manager.json +278 -0
  125. package/test-data/decision-tables/basic/returns-and-refund-policy.json +366 -0
  126. package/test-data/decision-tables/basic/returns-processing-system.json +448 -0
  127. package/test-data/decision-tables/basic/school-district-resource-allocation.json +282 -0
  128. package/test-data/decision-tables/basic/seat-map-optimization.json +325 -0
  129. package/test-data/decision-tables/basic/seller-fee-calculator.json +307 -0
  130. package/test-data/decision-tables/basic/service-level-agreement-enforcement.json +575 -0
  131. package/test-data/decision-tables/basic/smart-financial-product-matcher.json +249 -0
  132. package/test-data/decision-tables/basic/supply-chain-risk.json +316 -0
  133. package/test-data/decision-tables/basic/table-loop.json +93 -0
  134. package/test-data/decision-tables/basic/table.json +76 -0
  135. package/test-data/decision-tables/basic/traffic-violation-penalty-calculator.json +436 -0
  136. package/test-data/decision-tables/basic/transaction-compliance-classifier.json +525 -0
  137. package/test-data/decision-tables/basic/vehicle-claims-resolution.json +310 -0
  138. package/test-data/decision-tables/basic/warehouse-storage-location.json +345 -0
  139. package/test-data/decision-tables/hit-policy-collect/collect-multiple-matches.json +127 -0
  140. package/test-data/decision-tables/hit-policy-collect/collect-no-match.json +95 -0
  141. package/test-data/decision-tables/hit-policy-first/first-match.json +103 -0
  142. package/test-data/decision-tables/hit-policy-first/no-match.json +95 -0
  143. package/test-data/decision-tables/hit-policy-output-order/output-order-respected.json +94 -0
  144. package/test-data/decision-tables/hit-policy-output-order/string-output-order.json +94 -0
  145. package/test-data/decision-tables/hit-policy-priority/priority-respected.json +86 -0
  146. package/test-data/decision-tables/hit-policy-rule-order/rule-order-respected.json +94 -0
  147. package/test-data/decision-tables/hit-policy-unique/all-match-error.json +89 -0
  148. package/test-data/decision-tables/hit-policy-unique/multiple-match-error.json +89 -0
  149. package/test-data/decision-tables/hit-policy-unique/no-match.json +88 -0
  150. package/test-data/decision-tables/hit-policy-unique/unique-match.json +99 -0
  151. package/test-data/expressions/arithmetic/error-cyclic.json +114 -0
  152. package/test-data/expressions/arithmetic/error-missing-input.json +54 -0
  153. package/test-data/expressions/arithmetic/error-missing-output.json +54 -0
  154. package/test-data/expressions/arithmetic/expression-default.json +93 -0
  155. package/test-data/expressions/arithmetic/expression-fields.json +94 -0
  156. package/test-data/expressions/arithmetic/expression-loop.json +94 -0
  157. package/test-data/expressions/arithmetic/expression-passthrough.json +108 -0
  158. package/test-data/expressions/arithmetic/expression.json +69 -0
  159. package/test-data/expressions/arithmetic/nested-request.json +125 -0
  160. package/test-data/expressions/arithmetic/number-function.json +58 -0
  161. package/test-data/expressions/arithmetic/test-number-functions.json +68 -0
  162. package/test-data/expressions/functions/all.json +149 -0
  163. package/test-data/expressions/functions/avg.json +89 -0
  164. package/test-data/expressions/functions/filter.json +109 -0
  165. package/test-data/expressions/functions/flat.json +167 -0
  166. package/test-data/expressions/functions/map-strings.json +65 -0
  167. package/test-data/expressions/functions/map.json +73 -0
  168. package/test-data/expressions/functions/reduce.json +49 -0
  169. package/test-data/expressions/functions/some.json +175 -0
  170. package/test-data/expressions/functions/sort-strings.json +97 -0
  171. package/test-data/expressions/functions/sort.json +97 -0
  172. package/test-data/expressions/logical/logical-and.json +116 -0
  173. package/test-data/expressions/logical/logical-complex.json +260 -0
  174. package/test-data/expressions/logical/logical-not.json +111 -0
  175. package/test-data/expressions/logical/logical-or.json +123 -0
  176. package/test-data/expressions/string/string-comparison.json +128 -0
  177. package/test-data/expressions/string/string-concat.json +106 -0
  178. package/test-data/expressions/string/string-contains.json +125 -0
  179. package/test-data/expressions/string/string-endsWith.json +113 -0
  180. package/test-data/expressions/string/string-indexOf.json +131 -0
  181. package/test-data/expressions/string/string-join.json +92 -0
  182. package/test-data/expressions/string/string-lower.json +94 -0
  183. package/test-data/expressions/string/string-replace.json +130 -0
  184. package/test-data/expressions/string/string-split.json +101 -0
  185. package/test-data/expressions/string/string-startsWith.json +113 -0
  186. package/test-data/expressions/string/string-substring.json +138 -0
  187. package/test-data/expressions/string/string-trim.json +100 -0
  188. package/test-data/expressions/string/string-upper.json +94 -0
  189. package/test-data/other/custom.json +51 -0
  190. package/test-data/other/customer-input-schema.json +34 -0
  191. package/test-data/other/customer-output-schema.json +34 -0
  192. package/test-data/other/passthrough.json +31 -0
  193. package/test-data/sub-decisions/basic/$nodes-child.json +31 -0
  194. package/test-data/sub-decisions/basic/$nodes-parent.json +49 -0
  195. package/test-data/sub-decisions/basic/recursive-table1.json +49 -0
  196. package/test-data/sub-decisions/basic/recursive-table2.json +49 -0
  197. package/test-data/sub-decisions/complex-multi/approval-decision.json +31 -0
  198. package/test-data/sub-decisions/complex-multi/complex-dag.json +175 -0
  199. package/test-data/sub-decisions/complex-multi/credit-check.json +31 -0
  200. package/test-data/sub-decisions/complex-multi/customer-segmentation.json +31 -0
  201. package/test-data/sub-decisions/complex-multi/discount-eligibility.json +31 -0
  202. package/test-data/sub-decisions/complex-multi/eligibility-check.json +31 -0
  203. package/test-data/sub-decisions/complex-multi/final-offer.json +31 -0
  204. package/test-data/sub-decisions/complex-multi/income-verification.json +31 -0
  205. package/test-data/sub-decisions/complex-multi/linear-chain.json +121 -0
  206. package/test-data/sub-decisions/complex-multi/pricing-calculation.json +31 -0
  207. package/test-data/sub-decisions/complex-multi/product-eligibility.json +31 -0
  208. package/test-data/sub-decisions/complex-multi/risk-assessment.json +31 -0
  209. package/test-data/sub-decisions/complex-multi/shared-validation.json +31 -0
  210. package/test-data/sub-decisions/complex-multi/validation.json +31 -0
  211. package/test-data/sub-decisions/diamond/decision-a.json +31 -0
  212. package/test-data/sub-decisions/diamond/decision-b.json +31 -0
  213. package/test-data/sub-decisions/diamond/decision-c.json +31 -0
  214. package/test-data/sub-decisions/diamond/decision-shared.json +31 -0
  215. package/test-data/sub-decisions/diamond/diamond-pattern.json +109 -0
  216. package/test-data/sub-decisions/error-propagation/parent-calls-error.json +44 -0
  217. package/test-data/sub-decisions/error-propagation/sub-decision-with-error.json +60 -0
  218. package/test-data/switch-nodes/basic/account-dormancy-management.json +245 -0
  219. package/test-data/switch-nodes/basic/application-risk-assessment.json +474 -0
  220. package/test-data/switch-nodes/basic/cellular-data-rollover-system.json +281 -0
  221. package/test-data/switch-nodes/basic/clinical-pathway-selection.json +454 -0
  222. package/test-data/switch-nodes/basic/insurance-prior-authorization.json +467 -0
  223. package/test-data/switch-nodes/basic/last-mile-delivery-assignment.json +373 -0
  224. package/test-data/switch-nodes/basic/loan-approval.json +469 -0
  225. package/test-data/switch-nodes/basic/multi-switch.json +498 -0
  226. package/test-data/switch-nodes/basic/online-checkin-eligibility.json +285 -0
  227. package/test-data/switch-nodes/basic/order-consolidation-system.json +493 -0
  228. package/test-data/switch-nodes/basic/seller-approval-workflow.json +383 -0
  229. package/test-data/switch-nodes/basic/set-fee.json +243 -0
  230. package/test-data/switch-nodes/basic/shipping-carrier-selector.json +379 -0
  231. package/test-data/switch-nodes/basic/switch-node.json +167 -0
  232. package/test-data/switch-nodes/basic/switch-performance-2.json +1307 -0
  233. package/test-data/switch-nodes/basic/switch-performance.json +691 -0
  234. package/test-data/switch-nodes/basic/tax-exemption.json +295 -0
  235. package/test-data/switch-nodes/basic/warehouse-cross-docking.json +313 -0
  236. package/test-data/switch-nodes/default-cases/switch-with-default.json +134 -0
  237. package/test-data/zen-reference/$nodes-child.json +69 -0
  238. package/test-data/zen-reference/$nodes-parent.json +34 -0
  239. package/test-data/zen-reference/8k.json +87992 -0
  240. package/test-data/zen-reference/credit-analysis.json +324 -0
  241. package/test-data/zen-reference/custom.json +51 -0
  242. package/test-data/zen-reference/customer-input-schema.json +34 -0
  243. package/test-data/zen-reference/customer-output-schema.json +34 -0
  244. package/test-data/zen-reference/error-cyclic.json +114 -0
  245. package/test-data/zen-reference/error-missing-input.json +54 -0
  246. package/test-data/zen-reference/error-missing-output.json +54 -0
  247. package/test-data/zen-reference/expression.json +69 -0
  248. package/test-data/zen-reference/function-v2.json +48 -0
  249. package/test-data/zen-reference/function.json +46 -0
  250. package/test-data/zen-reference/graphs/account-dormancy-management.json +245 -0
  251. package/test-data/zen-reference/graphs/affiliate-commission-calculator.json +228 -0
  252. package/test-data/zen-reference/graphs/airline-loyalty-points-calculations.json +285 -0
  253. package/test-data/zen-reference/graphs/airline-upgrade-eligibility.json +466 -0
  254. package/test-data/zen-reference/graphs/aml.json +537 -0
  255. package/test-data/zen-reference/graphs/application-risk-assessment.json +474 -0
  256. package/test-data/zen-reference/graphs/auto-insurance-premium-calculator.json +412 -0
  257. package/test-data/zen-reference/graphs/booking-personalization-system.json +553 -0
  258. package/test-data/zen-reference/graphs/care-team-assignment-system.json +585 -0
  259. package/test-data/zen-reference/graphs/cellular-data-rollover-system.json +281 -0
  260. package/test-data/zen-reference/graphs/claim-validation-system.json +307 -0
  261. package/test-data/zen-reference/graphs/clinical-lab-result-interpreter.json +433 -0
  262. package/test-data/zen-reference/graphs/clinical-pathway-selection.json +454 -0
  263. package/test-data/zen-reference/graphs/clinical-treatment-protocol.json +474 -0
  264. package/test-data/zen-reference/graphs/company-analysis.json +390 -0
  265. package/test-data/zen-reference/graphs/credit-limit-adjustment.json +479 -0
  266. package/test-data/zen-reference/graphs/customer-eligibility-engine.json +551 -0
  267. package/test-data/zen-reference/graphs/customer-lifetime-value.json +200 -0
  268. package/test-data/zen-reference/graphs/customer-onboarding-kyc-verification.json +611 -0
  269. package/test-data/zen-reference/graphs/customer-service-escalation.json +191 -0
  270. package/test-data/zen-reference/graphs/decision-table-discounts.json +168 -0
  271. package/test-data/zen-reference/graphs/decision-table-shipping.json +398 -0
  272. package/test-data/zen-reference/graphs/delivery-route-optimizer.json +271 -0
  273. package/test-data/zen-reference/graphs/device-compatibility-checker.json +303 -0
  274. package/test-data/zen-reference/graphs/disaster-relief-fund-allocation.json +296 -0
  275. package/test-data/zen-reference/graphs/dynamic-fx-rate-pricing-system.json +237 -0
  276. package/test-data/zen-reference/graphs/dynamic-marketplace-comission-calculator.json +242 -0
  277. package/test-data/zen-reference/graphs/dynamic-shipping-cost-calculator.json +378 -0
  278. package/test-data/zen-reference/graphs/dynamic-tarrif-engine.json +289 -0
  279. package/test-data/zen-reference/graphs/dynamic-ticket-pricing.json +325 -0
  280. package/test-data/zen-reference/graphs/empty-column-with-space.json +100 -0
  281. package/test-data/zen-reference/graphs/empty-column-without-space.json +100 -0
  282. package/test-data/zen-reference/graphs/environment-compliance-assessment.json +386 -0
  283. package/test-data/zen-reference/graphs/expression-default.json +93 -0
  284. package/test-data/zen-reference/graphs/expression-fields.json +94 -0
  285. package/test-data/zen-reference/graphs/expression-loop.json +94 -0
  286. package/test-data/zen-reference/graphs/expression-passthrough.json +108 -0
  287. package/test-data/zen-reference/graphs/expression-table-map.json +313 -0
  288. package/test-data/zen-reference/graphs/flash-sale-eligibility.json +366 -0
  289. package/test-data/zen-reference/graphs/flight-dispatch-decision-system.json +455 -0
  290. package/test-data/zen-reference/graphs/flight-rebooking-fee-calculator.json +406 -0
  291. package/test-data/zen-reference/graphs/government-assistance.json +299 -0
  292. package/test-data/zen-reference/graphs/grant-funding-distribution.json +307 -0
  293. package/test-data/zen-reference/graphs/hazardous-materials-management-system.json +414 -0
  294. package/test-data/zen-reference/graphs/immigration-eligibility-evaluator.json +765 -0
  295. package/test-data/zen-reference/graphs/import-duties-calculator.json +318 -0
  296. package/test-data/zen-reference/graphs/insurance-agent-commission.json +228 -0
  297. package/test-data/zen-reference/graphs/insurance-breakdown.json +421 -0
  298. package/test-data/zen-reference/graphs/insurance-coverage-calculator.json +362 -0
  299. package/test-data/zen-reference/graphs/insurance-prior-authorization.json +467 -0
  300. package/test-data/zen-reference/graphs/insurance-underwriting-risk.json +321 -0
  301. package/test-data/zen-reference/graphs/international-roaming-policy-manager.json +199 -0
  302. package/test-data/zen-reference/graphs/last-mile-delivery-assignment.json +373 -0
  303. package/test-data/zen-reference/graphs/legacy-plan-management.json +434 -0
  304. package/test-data/zen-reference/graphs/loan-approval.json +469 -0
  305. package/test-data/zen-reference/graphs/marketplace-listing-verification-system.json +334 -0
  306. package/test-data/zen-reference/graphs/medication-dosage-calculator.json +318 -0
  307. package/test-data/zen-reference/graphs/merch-bags.json +171 -0
  308. package/test-data/zen-reference/graphs/multi-switch.json +498 -0
  309. package/test-data/zen-reference/graphs/municipal-permit-evaluation-system.json +364 -0
  310. package/test-data/zen-reference/graphs/mvno-partner-enablement.json +313 -0
  311. package/test-data/zen-reference/graphs/nested-request.json +125 -0
  312. package/test-data/zen-reference/graphs/online-checkin-eligibility.json +285 -0
  313. package/test-data/zen-reference/graphs/order-consolidation-system.json +493 -0
  314. package/test-data/zen-reference/graphs/partner-revenue-sharing.json +244 -0
  315. package/test-data/zen-reference/graphs/payment-routing-and-fee-calculator.json +475 -0
  316. package/test-data/zen-reference/graphs/policy-discount-calculator.json +307 -0
  317. package/test-data/zen-reference/graphs/policy-eligibility-analyzer.json +299 -0
  318. package/test-data/zen-reference/graphs/product-listing-scoring.json +358 -0
  319. package/test-data/zen-reference/graphs/realtime-fraud-detection.json +235 -0
  320. package/test-data/zen-reference/graphs/regional-compliance-manager.json +278 -0
  321. package/test-data/zen-reference/graphs/returns-and-refund-policy.json +366 -0
  322. package/test-data/zen-reference/graphs/returns-processing-system.json +448 -0
  323. package/test-data/zen-reference/graphs/school-district-resource-allocation.json +282 -0
  324. package/test-data/zen-reference/graphs/seat-map-optimization.json +325 -0
  325. package/test-data/zen-reference/graphs/seller-approval-workflow.json +383 -0
  326. package/test-data/zen-reference/graphs/seller-fee-calculator.json +307 -0
  327. package/test-data/zen-reference/graphs/service-level-agreement-enforcement.json +575 -0
  328. package/test-data/zen-reference/graphs/set-fee.json +243 -0
  329. package/test-data/zen-reference/graphs/shipping-carrier-selector.json +379 -0
  330. package/test-data/zen-reference/graphs/smart-financial-product-matcher.json +249 -0
  331. package/test-data/zen-reference/graphs/supply-chain-risk.json +316 -0
  332. package/test-data/zen-reference/graphs/table-loop.json +93 -0
  333. package/test-data/zen-reference/graphs/tax-exemption.json +295 -0
  334. package/test-data/zen-reference/graphs/traffic-violation-penalty-calculator.json +436 -0
  335. package/test-data/zen-reference/graphs/transaction-compliance-classifier.json +525 -0
  336. package/test-data/zen-reference/graphs/vehicle-claims-resolution.json +310 -0
  337. package/test-data/zen-reference/graphs/warehouse-cross-docking.json +313 -0
  338. package/test-data/zen-reference/graphs/warehouse-storage-location.json +345 -0
  339. package/test-data/zen-reference/http-function.json +34 -0
  340. package/test-data/zen-reference/infinite-function.json +46 -0
  341. package/test-data/zen-reference/js/imports.js +25 -0
  342. package/test-data/zen-reference/passthrough.json +31 -0
  343. package/test-data/zen-reference/recursive-table1.json +49 -0
  344. package/test-data/zen-reference/recursive-table2.json +49 -0
  345. package/test-data/zen-reference/sleep-function.json +34 -0
  346. package/test-data/zen-reference/switch-node.json +167 -0
  347. package/test-data/zen-reference/switch-performance-2.json +1307 -0
  348. package/test-data/zen-reference/switch-performance.json +691 -0
  349. package/test-data/zen-reference/table.json +76 -0
  350. package/tests/helpers/index.ts +73 -0
  351. package/tests/helpers/mock-context.ts +231 -0
  352. package/tests/helpers/round-trip.ts +398 -0
  353. package/tests/helpers/test-harness-comparison.ts +325 -0
  354. package/tests/helpers/test-harness-wasm.ts +710 -0
  355. package/tests/helpers/test-harness.ts +28 -0
  356. package/tests/helpers/wasm-test.ts +659 -0
  357. package/tests/integration/compilation-errors.test.ts +864 -0
  358. package/tests/integration/decision-tables.test.ts +531 -0
  359. package/tests/integration/edge-cases.test.ts +787 -0
  360. package/tests/integration/expressions.test.ts +513 -0
  361. package/tests/integration/function-node-integration.test.ts +182 -0
  362. package/tests/integration/sub-decisions.test.ts +108 -0
  363. package/tests/integration/switch-nodes.test.ts +399 -0
  364. package/tests/integration/unary-or-matching.test.ts +53 -0
  365. package/tests/integration/wasm-data-types.test.ts +398 -0
  366. package/tests/integration/wasm-errors.test.ts +199 -0
  367. package/tests/integration/wasm-execution.test.ts +348 -0
  368. package/tests/integration/wasm-memory.test.ts +228 -0
  369. package/tests/scripts/analyze-coverage.ts +166 -0
  370. package/tests/scripts/categorize-tests.ts +396 -0
  371. package/tests/scripts/coverage-analysis.ts +836 -0
  372. package/tests/unit/compiler/cache.test.ts +238 -0
  373. package/tests/unit/compiler/errors.test.ts +316 -0
  374. package/tests/unit/compiler/graph-scalability.test.ts +510 -0
  375. package/tests/unit/compiler/graph.test.ts +878 -0
  376. package/tests/unit/compiler/input-validation.test.ts +447 -0
  377. package/tests/unit/compiler/logical-and-parser.test.ts +143 -0
  378. package/tests/unit/compiler/logical-not-parser.test.ts +107 -0
  379. package/tests/unit/compiler/logical-or-parser.test.ts +236 -0
  380. package/tests/unit/compiler/marshal-gen/marshal-gen.test.ts +97 -0
  381. package/tests/unit/compiler/nodes/decision-table.test.ts +103 -0
  382. package/tests/unit/compiler/nodes/decision.test.ts +182 -0
  383. package/tests/unit/compiler/nodes/function-compile.test.ts +204 -0
  384. package/tests/unit/compiler/nodes/function.test.ts +176 -0
  385. package/tests/unit/compiler/nodes/input.test.ts +30 -0
  386. package/tests/unit/compiler/nodes/switch.test.ts +127 -0
  387. package/tests/unit/compiler/optimizer-cache.test.ts +327 -0
  388. package/tests/unit/compiler/optimizer-implementation.test.ts +625 -0
  389. package/tests/unit/compiler/parser.test.ts +508 -0
  390. package/tests/unit/compiler/runtime-error-cleanup.test.ts +426 -0
  391. package/tests/unit/compiler/runtime-validation.test.ts +303 -0
  392. package/tests/unit/compiler/runtime.test.ts +221 -0
  393. package/tests/unit/compiler/schema/schema.test.ts +248 -0
  394. package/tests/unit/compiler/unary-ast-transforms.test.ts +245 -0
  395. package/tsconfig.json +27 -0
  396. package/tsup.config.ts +11 -0
  397. package/vitest.config.ts +12 -0
@@ -0,0 +1,536 @@
1
+ import { createToken, CstParser, Lexer, type IToken } from 'chevrotain';
2
+ import { CompilationError, ErrorCode, type SourceLocation } from './errors';
3
+
4
+ // === TOKENS (order matters - longer matches first) ===
5
+
6
+ const WhiteSpace = createToken({
7
+ name: 'WhiteSpace',
8
+ pattern: /\s+/,
9
+ group: Lexer.SKIPPED,
10
+ });
11
+
12
+ const DotDot = createToken({ name: 'DotDot', pattern: /\.\./ });
13
+ const Dot = createToken({ name: 'Dot', pattern: /\./ });
14
+
15
+ const Eq = createToken({ name: 'Eq', pattern: /==/ });
16
+ const NotEq = createToken({ name: 'NotEq', pattern: /!=/ });
17
+ const EqEq = createToken({ name: 'EqEq', pattern: /===/ });
18
+ const NotEqEq = createToken({ name: 'NotEqEq', pattern: /!==/ });
19
+ const LtEq = createToken({ name: 'LtEq', pattern: /<=/ });
20
+ const GtEq = createToken({ name: 'GtEq', pattern: />=/ });
21
+ const Lt = createToken({ name: 'Lt', pattern: /</ });
22
+ const Gt = createToken({ name: 'Gt', pattern: />/ });
23
+
24
+ const AndAnd = createToken({ name: 'AndAnd', pattern: /&&/ });
25
+ const OrOr = createToken({ name: 'OrOr', pattern: /\|\|/ });
26
+ const NotNot = createToken({ name: 'NotNot', pattern: /!/ });
27
+
28
+ // Keywords use longer_alt to ensure 'and'/'or' are recognized as operators rather than
29
+ // variable names - Chevrotain tries longer_alt tokens first, preventing 'android' from
30
+ // incorrectly tokenizing as 'and' + 'roid'.
31
+ const Identifier = createToken({ name: 'Identifier', pattern: /[$a-zA-Z_]\w*/ });
32
+
33
+ const And = createToken({ name: 'And', pattern: /and/, longer_alt: Identifier });
34
+ const Or = createToken({ name: 'Or', pattern: /or/, longer_alt: Identifier });
35
+ const Not = createToken({ name: 'Not', pattern: /not/, longer_alt: Identifier });
36
+ const In = createToken({ name: 'In', pattern: /in/, longer_alt: Identifier });
37
+
38
+ const True = createToken({ name: 'True', pattern: /true/, longer_alt: Identifier });
39
+ const False = createToken({ name: 'False', pattern: /false/, longer_alt: Identifier });
40
+ const Null = createToken({ name: 'Null', pattern: /null/, longer_alt: Identifier });
41
+
42
+ const NumberLiteral = createToken({
43
+ name: 'NumberLiteral',
44
+ pattern: /\d[\d_]*(\.\d+)?/,
45
+ });
46
+
47
+ const StringLiteral = createToken({
48
+ name: 'StringLiteral',
49
+ pattern: /"([^"\\]|\\.)*"|'([^'\\]|\\.)*'/,
50
+ });
51
+
52
+ // Template strings
53
+ // Token order critical: TemplateHead/Middle/Tail must precede TemplateNoSub because Chevrotain matches first token that fits.
54
+ // Without this order, '`foo${x}`' would incorrectly match TemplateNoSub instead of TemplateHead + expr + TemplateTail.
55
+ // TemplateNoSub pattern must explicitly exclude ${ to prevent ambiguity between templates with/without interpolation.
56
+ const TemplateHead = createToken({
57
+ name: 'TemplateHead',
58
+ pattern: /`(?:[^`\\$]|\\.|\$(?!\{))*?\$\{/,
59
+ });
60
+ const TemplateMiddle = createToken({
61
+ name: 'TemplateMiddle',
62
+ pattern: /\}(?:[^`\\$]|\\.|\$(?!\{))*?\$\{/,
63
+ });
64
+ const TemplateTail = createToken({
65
+ name: 'TemplateTail',
66
+ pattern: /\}(?:[^`\\$]|\\.|\$(?!\{))*?`/,
67
+ });
68
+ // TemplateNoSub should NOT match strings that contain ${, so we exclude them explicitly
69
+ const TemplateNoSub = createToken({
70
+ name: 'TemplateNoSub',
71
+ pattern: /`(?:[^`\\$]|\\.|\$(?!\{))*?`/,
72
+ });
73
+
74
+ const Hash = createToken({ name: 'Hash', pattern: /#/ }); // Current element in higher-order functions
75
+ // Dollar ($) represents explicit reference in unary mode. Uses longer_alt: Identifier to ensure $-prefixed identifiers like '$myVar'
76
+ // are tokenized as a single Identifier token rather than separate Dollar + Identifier tokens, since Identifier pattern includes '$' prefix.
77
+ const Dollar = createToken({ name: 'Dollar', pattern: /\$/, longer_alt: Identifier });
78
+
79
+ // Operators and delimiters
80
+ const Plus = createToken({ name: 'Plus', pattern: /\+/ });
81
+ const Minus = createToken({ name: 'Minus', pattern: /-/ });
82
+ const Mult = createToken({ name: 'Mult', pattern: /\*/ });
83
+ const Div = createToken({ name: 'Div', pattern: /\// });
84
+ const Mod = createToken({ name: 'Mod', pattern: /%/ });
85
+ const Power = createToken({ name: 'Power', pattern: /\^/ });
86
+
87
+ const LParen = createToken({ name: 'LParen', pattern: /\(/ });
88
+ const RParen = createToken({ name: 'RParen', pattern: /\)/ });
89
+ const LBracket = createToken({ name: 'LBracket', pattern: /\[/ });
90
+ const RBracket = createToken({ name: 'RBracket', pattern: /\]/ });
91
+
92
+ const LBrace = createToken({ name: 'LBrace', pattern: /\{/ });
93
+ const RBrace = createToken({ name: 'RBrace', pattern: /\}/ });
94
+
95
+ const Comma = createToken({ name: 'Comma', pattern: /,/ });
96
+ const Question = createToken({ name: 'Question', pattern: /\?/, longer_alt: Identifier });
97
+ const QuestionQuestion = createToken({ name: 'QuestionQuestion', pattern: /\?\?/ });
98
+ const Colon = createToken({ name: 'Colon', pattern: /:/ });
99
+
100
+ const allTokens = [
101
+ WhiteSpace,
102
+ DotDot,
103
+ Dot,
104
+ EqEq,
105
+ NotEqEq,
106
+ Eq,
107
+ NotEq,
108
+ LtEq,
109
+ GtEq,
110
+ Lt,
111
+ Gt,
112
+ AndAnd,
113
+ OrOr,
114
+ NotNot,
115
+ And,
116
+ Or,
117
+ Not,
118
+ In,
119
+ True,
120
+ False,
121
+ Null,
122
+ Plus,
123
+ Minus,
124
+ Mult,
125
+ Div,
126
+ Mod,
127
+ Power,
128
+ LParen,
129
+ RParen,
130
+ LBracket,
131
+ RBracket,
132
+ LBrace,
133
+ // Template tokens must come before RBrace so that TemplateTail/TemplateMiddle can match }
134
+ TemplateHead,
135
+ TemplateMiddle,
136
+ TemplateTail,
137
+ TemplateNoSub,
138
+ RBrace,
139
+ Comma,
140
+ QuestionQuestion,
141
+ Question,
142
+ Colon,
143
+ Hash,
144
+ Dollar,
145
+ NumberLiteral,
146
+ StringLiteral,
147
+ Identifier,
148
+ ];
149
+
150
+ // === PARSER (precedence via rule layering) ===
151
+ // Precedence (lowest to highest):
152
+ // 1. Ternary: ? :
153
+ // 2. Null coalescing: ??
154
+ // 3. Logical OR: or
155
+ // 4. Logical AND: and
156
+ // 5. Comparison: ==, !=, <, >, <=, >=, in, not in
157
+ // 6. Additive: +, -
158
+ // 7. Multiplicative: *, /, %
159
+ // 8. Power: ^ (RIGHT-ASSOCIATIVE)
160
+ // 9. Unary: not, -
161
+ // 10. Postfix: .property, [index], (call)
162
+ // 11. Primary: literals, identifiers, parentheses
163
+
164
+ class ZenExpressionParser extends CstParser {
165
+ constructor() {
166
+ super(allTokens);
167
+ this.performSelfAnalysis();
168
+ }
169
+
170
+ // Entry - ternary (lowest precedence)
171
+ expression = this.RULE('expression', () => {
172
+ this.SUBRULE(this.nullCoalescingExpression, { LABEL: 'condition' });
173
+ this.OPTION(() => {
174
+ this.CONSUME(Question);
175
+ this.SUBRULE2(this.expression, { LABEL: 'consequent' });
176
+ this.CONSUME(Colon);
177
+ this.SUBRULE3(this.expression, { LABEL: 'alternate' });
178
+ });
179
+ });
180
+
181
+ nullCoalescingExpression = this.RULE('nullCoalescingExpression', () => {
182
+ this.SUBRULE(this.orExpression, { LABEL: 'lhs' });
183
+ this.MANY(() => {
184
+ this.CONSUME(QuestionQuestion);
185
+ this.SUBRULE2(this.orExpression, { LABEL: 'rhs' });
186
+ });
187
+ });
188
+
189
+ orExpression = this.RULE('orExpression', () => {
190
+ this.SUBRULE(this.andExpression, { LABEL: 'lhs' });
191
+ this.MANY(() => {
192
+ this.OR([{ ALT: () => this.CONSUME(Or) }, { ALT: () => this.CONSUME(OrOr) }]);
193
+ this.SUBRULE2(this.andExpression, { LABEL: 'rhs' });
194
+ });
195
+ });
196
+
197
+ andExpression = this.RULE('andExpression', () => {
198
+ this.SUBRULE(this.comparisonExpression, { LABEL: 'lhs' });
199
+ this.MANY(() => {
200
+ this.OR([{ ALT: () => this.CONSUME(And) }, { ALT: () => this.CONSUME(AndAnd) }]);
201
+ this.SUBRULE2(this.comparisonExpression, { LABEL: 'rhs' });
202
+ });
203
+ });
204
+
205
+ comparisonExpression = this.RULE('comparisonExpression', () => {
206
+ this.SUBRULE(this.additiveExpression, { LABEL: 'lhs' });
207
+ this.OPTION(() => {
208
+ this.OR([
209
+ { ALT: () => this.CONSUME(Eq) },
210
+ { ALT: () => this.CONSUME(NotEq) },
211
+ { ALT: () => this.CONSUME(Lt) },
212
+ { ALT: () => this.CONSUME(Gt) },
213
+ { ALT: () => this.CONSUME(LtEq) },
214
+ { ALT: () => this.CONSUME(GtEq) },
215
+ { ALT: () => this.CONSUME(In) },
216
+ {
217
+ ALT: () => {
218
+ this.CONSUME(Not);
219
+ this.CONSUME2(In);
220
+ },
221
+ }, // not in
222
+ ]);
223
+ this.SUBRULE2(this.additiveExpression, { LABEL: 'rhs' });
224
+ });
225
+ });
226
+
227
+ additiveExpression = this.RULE('additiveExpression', () => {
228
+ this.SUBRULE(this.multiplicativeExpression, { LABEL: 'lhs' });
229
+ this.MANY(() => {
230
+ this.OR([{ ALT: () => this.CONSUME(Plus) }, { ALT: () => this.CONSUME(Minus) }]);
231
+ this.SUBRULE2(this.multiplicativeExpression, { LABEL: 'rhs' });
232
+ });
233
+ });
234
+
235
+ multiplicativeExpression = this.RULE('multiplicativeExpression', () => {
236
+ this.SUBRULE(this.powerExpression, { LABEL: 'lhs' });
237
+ this.MANY(() => {
238
+ this.OR([
239
+ { ALT: () => this.CONSUME(Mult) },
240
+ { ALT: () => this.CONSUME(Div) },
241
+ { ALT: () => this.CONSUME(Mod) },
242
+ ]);
243
+ this.SUBRULE2(this.powerExpression, { LABEL: 'rhs' });
244
+ });
245
+ });
246
+
247
+ // Power is RIGHT-ASSOCIATIVE to match mathematical convention:
248
+ // e.g. 2^3^4 = 2^(3^4) = 2^81, not (2^3)^4 = 8^4 = 4096.
249
+ // Use self-recursion instead of MANY for right-associativity
250
+ powerExpression = this.RULE('powerExpression', () => {
251
+ this.SUBRULE(this.unaryExpression, { LABEL: 'base' });
252
+ this.OPTION(() => {
253
+ this.CONSUME(Power);
254
+ this.SUBRULE(this.powerExpression, { LABEL: 'exponent' }); // Self-recursion!
255
+ });
256
+ });
257
+
258
+ unaryExpression = this.RULE('unaryExpression', () => {
259
+ this.OR([
260
+ {
261
+ ALT: () => {
262
+ this.CONSUME(Not);
263
+ this.SUBRULE(this.unaryExpression, { LABEL: 'operand' });
264
+ },
265
+ },
266
+ {
267
+ ALT: () => {
268
+ this.CONSUME(NotNot);
269
+ this.SUBRULE2(this.unaryExpression, { LABEL: 'operand' });
270
+ },
271
+ },
272
+ {
273
+ ALT: () => {
274
+ this.CONSUME(Minus);
275
+ this.SUBRULE3(this.unaryExpression, { LABEL: 'operand' });
276
+ },
277
+ },
278
+ { ALT: () => this.SUBRULE(this.postfixExpression) },
279
+ ]);
280
+ });
281
+
282
+ // Postfix: .property, [index], (args)
283
+ postfixExpression = this.RULE('postfixExpression', () => {
284
+ this.SUBRULE(this.primaryExpression);
285
+ this.MANY(() => {
286
+ this.OR([
287
+ {
288
+ ALT: () => {
289
+ this.CONSUME(Dot);
290
+ this.CONSUME(Identifier, { LABEL: 'property' });
291
+ },
292
+ },
293
+ {
294
+ ALT: () => {
295
+ this.CONSUME(LBracket);
296
+ this.SUBRULE(this.expression, { LABEL: 'index' });
297
+ this.CONSUME(RBracket);
298
+ },
299
+ },
300
+ {
301
+ ALT: () => {
302
+ this.CONSUME(LParen);
303
+ this.OPTION(() => this.SUBRULE(this.argumentList));
304
+ this.CONSUME(RParen);
305
+ },
306
+ },
307
+ ]);
308
+ });
309
+ });
310
+
311
+ primaryExpression = this.RULE('primaryExpression', () => {
312
+ this.OR([
313
+ { ALT: () => this.CONSUME(NumberLiteral) },
314
+ { ALT: () => this.CONSUME(StringLiteral) },
315
+ { ALT: () => this.SUBRULE(this.templateLiteral) },
316
+ { ALT: () => this.CONSUME(True) },
317
+ { ALT: () => this.CONSUME(False) },
318
+ { ALT: () => this.CONSUME(Null) },
319
+ { ALT: () => this.CONSUME(Hash) }, // # for current element in higher-order functions
320
+ { ALT: () => this.CONSUME(Dollar) }, // $ for explicit reference in unary mode
321
+ { ALT: () => this.CONSUME(Identifier) },
322
+ // Array literal: [1, 2, 3]
323
+ {
324
+ GATE: () => this.isArrayLiteralStart(),
325
+ ALT: () => this.SUBRULE(this.arrayLiteralExpression),
326
+ },
327
+ // Object literal: {key: value, ...}
328
+ {
329
+ GATE: () => this.isObjectLiteralStart(),
330
+ ALT: () => this.SUBRULE(this.objectLiteralExpression),
331
+ },
332
+ {
333
+ ALT: () => {
334
+ this.CONSUME(LParen);
335
+ this.SUBRULE(this.expression);
336
+ this.CONSUME(RParen);
337
+ },
338
+ },
339
+ // Interval: use GATE to check for DotDot lookahead
340
+ {
341
+ GATE: () => this.isIntervalStart(),
342
+ ALT: () => this.SUBRULE(this.intervalExpression),
343
+ },
344
+ ]);
345
+ });
346
+
347
+ // Template string: `Hello ${name}, you have ${count} items`
348
+ templateLiteral = this.RULE('templateLiteral', () => {
349
+ this.OR([
350
+ // Simple template with no interpolation
351
+ { ALT: () => this.CONSUME(TemplateNoSub) },
352
+ // Template with interpolation
353
+ {
354
+ ALT: () => {
355
+ this.CONSUME(TemplateHead);
356
+ this.SUBRULE(this.expression, { LABEL: 'expr' });
357
+ this.MANY(() => {
358
+ this.CONSUME(TemplateMiddle);
359
+ this.SUBRULE2(this.expression, { LABEL: 'expr' });
360
+ });
361
+ this.CONSUME(TemplateTail);
362
+ },
363
+ },
364
+ ]);
365
+ });
366
+
367
+ // Interval: [a..b] or (a..b) with open/closed variations
368
+ intervalExpression = this.RULE('intervalExpression', () => {
369
+ this.OR([
370
+ { ALT: () => this.CONSUME(LBracket, { LABEL: 'startBracket' }) },
371
+ { ALT: () => this.CONSUME(LParen, { LABEL: 'startBracket' }) },
372
+ ]);
373
+
374
+ this.SUBRULE(this.expression, { LABEL: 'start' });
375
+ this.CONSUME(DotDot);
376
+ this.SUBRULE2(this.expression, { LABEL: 'end' });
377
+
378
+ this.OR2([
379
+ { ALT: () => this.CONSUME(RBracket, { LABEL: 'endBracket' }) },
380
+ { ALT: () => this.CONSUME(RParen, { LABEL: 'endBracket' }) },
381
+ ]);
382
+ });
383
+
384
+ // Array literal: [1, 2, 3] or []
385
+ arrayLiteralExpression = this.RULE('arrayLiteralExpression', () => {
386
+ this.CONSUME(LBracket);
387
+ this.OPTION(() => this.SUBRULE(this.argumentList));
388
+ this.CONSUME(RBracket);
389
+ });
390
+
391
+ // Object literal: {key: value, ...} or {}
392
+ objectLiteralExpression = this.RULE('objectLiteralExpression', () => {
393
+ this.CONSUME(LBrace);
394
+ this.OPTION(() => this.SUBRULE(this.objectPropertyList));
395
+ this.CONSUME(RBrace);
396
+ });
397
+
398
+ // Object property list for object literals
399
+ objectPropertyList = this.RULE('objectPropertyList', () => {
400
+ this.SUBRULE(this.objectProperty, { LABEL: 'prop' });
401
+ this.MANY(() => {
402
+ this.CONSUME(Comma);
403
+ this.SUBRULE2(this.objectProperty, { LABEL: 'prop' });
404
+ });
405
+ });
406
+
407
+ // Single object property: key: value
408
+ objectProperty = this.RULE('objectProperty', () => {
409
+ this.OR([
410
+ { ALT: () => this.CONSUME(StringLiteral, { LABEL: 'key' }) },
411
+ { ALT: () => this.CONSUME(Identifier, { LABEL: 'key' }) },
412
+ ]);
413
+ this.CONSUME(Colon);
414
+ this.SUBRULE(this.expression, { LABEL: 'value' });
415
+ });
416
+
417
+ argumentList = this.RULE('argumentList', () => {
418
+ this.SUBRULE(this.expression, { LABEL: 'arg' });
419
+ this.MANY(() => {
420
+ this.CONSUME(Comma);
421
+ this.SUBRULE2(this.expression, { LABEL: 'arg' });
422
+ });
423
+ });
424
+
425
+ // Helper: Check if current position starts an array literal
426
+ // An array literal is [ followed by expressions and commas, then ]
427
+ // We must distinguish [1,2] from [a..b]. An interval ALWAYS contains .. before closing
428
+ private isArrayLiteralStart(): boolean {
429
+ const lookahead = this.LA(1);
430
+ if (!lookahead || lookahead.tokenType !== LBracket) {
431
+ return false;
432
+ }
433
+
434
+ // Check if we have .. before a closing bracket (interval)
435
+ // If yes, this is not an array literal
436
+ return !this.isIntervalStart();
437
+ }
438
+
439
+ // Helper: Check if current position starts an object literal
440
+ // An object literal is { followed by key:value pairs and commas, then }
441
+ private isObjectLiteralStart(): boolean {
442
+ const lookahead = this.LA(1);
443
+ if (!lookahead || lookahead.tokenType !== LBrace) {
444
+ return false;
445
+ }
446
+ return true;
447
+ }
448
+
449
+ // Helper: Check if current position starts an interval (lookahead for ..)
450
+ // An interval is [expr..expr] or (expr..expr)
451
+ //
452
+ // We verify that after the start bracket, we'll eventually see .. before a closing bracket
453
+ // O(n) lookahead required because interval syntax '[1..10]' is ambiguous with array literals
454
+ // '[1,2,3]' - we must scan ahead to find '..' token to disambiguate, potentially examining
455
+ // entire token stream for malformed input.
456
+ private isIntervalStart(): boolean {
457
+ const lookahead = this.LA(1);
458
+ if (!lookahead) {
459
+ return false;
460
+ }
461
+ const tokenType = lookahead.tokenType;
462
+
463
+ // Only intervals start with [ or ( and must contain .. before closing
464
+ if (tokenType !== LBracket && tokenType !== LParen) {
465
+ return false;
466
+ }
467
+
468
+ // Look ahead through tokens to see if we find DotDot before a closing bracket
469
+ // Note: Chevrotain's LA() returns EOF token (not null) when past end of input
470
+ let depth = 0;
471
+ let i = 0;
472
+ while (true) {
473
+ const token = this.LA(++i);
474
+ // Check for end of input - LA() returns EOF token, not null
475
+ if (!token || token.tokenType.name === 'EOF') {
476
+ break;
477
+ }
478
+
479
+ const tt = token.tokenType;
480
+ if (tt === LBracket || tt === LParen || tt === LBrace) {
481
+ depth++;
482
+ } else if (tt === RBracket || tt === RParen || tt === RBrace) {
483
+ if (depth === 0) {
484
+ // Found end before DotDot - not an interval
485
+ return false;
486
+ }
487
+ depth--;
488
+ } else if (tt === DotDot) {
489
+ // Found DotDot before closing - this is an interval!
490
+ return true;
491
+ }
492
+ }
493
+
494
+ return false;
495
+ }
496
+ }
497
+
498
+ export const parser = new ZenExpressionParser();
499
+ export const lexer = new Lexer(allTokens);
500
+
501
+ /**
502
+ * Parse an expression string and return the CST.
503
+ *
504
+ * @param input - The expression string to parse
505
+ * @param location - Optional source location for error reporting
506
+ * @returns The Concrete Syntax Tree (CST)
507
+ * @throws Error if the parse fails
508
+ */
509
+ export function parseExpression(input: string, location?: SourceLocation): any {
510
+ const lexResult = lexer.tokenize(input);
511
+
512
+ if (lexResult.errors.length > 0) {
513
+ throw new CompilationError(
514
+ `Lexer error: ${lexResult.errors.map((e) => e.message).join(', ')}`,
515
+ ErrorCode.PARSE_ERROR,
516
+ location,
517
+ { expression: input },
518
+ );
519
+ }
520
+
521
+ parser.input = lexResult.tokens;
522
+ const cst = parser.expression();
523
+
524
+ if (parser.errors.length > 0) {
525
+ throw new CompilationError(
526
+ `Parse error: ${parser.errors.map((e) => e.message).join(', ')}`,
527
+ ErrorCode.PARSE_ERROR,
528
+ location,
529
+ { expression: input },
530
+ );
531
+ }
532
+
533
+ return cst;
534
+ }
535
+
536
+ export type { IToken };