@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,625 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import {
3
+ constantFolding,
4
+ expressionInlining,
5
+ tableIndexing,
6
+ runOptimizations,
7
+ } from '../../../src/compiler/optimizer';
8
+ import { NodeType } from '../../../src/compiler/parser';
9
+
10
+ describe('Constant Folding', () => {
11
+ it('should handle empty cells and dash wildcards', () => {
12
+ const jdm = {
13
+ nodes: [
14
+ {
15
+ id: 'table1',
16
+ type: NodeType.DECISION_TABLE,
17
+ name: 'Wildcard Table',
18
+ position: { x: 0, y: 0 },
19
+ content: {
20
+ rules: [
21
+ { input1: '', output1: 'test1' },
22
+ { input1: '-', output1: 'test2' },
23
+ ],
24
+ inputs: [{ id: 'input1', field: 'input', type: 'string' }],
25
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
26
+ hitPolicy: 'first',
27
+ },
28
+ },
29
+ ],
30
+ edges: [],
31
+ };
32
+
33
+ const optimized = constantFolding(jdm);
34
+ // Empty and dash cells should remain unchanged
35
+ expect(optimized.nodes[0].content.rules[0].input1).toBe('');
36
+ expect(optimized.nodes[0].content.rules[1].input1).toBe('-');
37
+ });
38
+
39
+ it('should preserve non-constant expression references', () => {
40
+ const jdm = {
41
+ nodes: [
42
+ {
43
+ id: 'table1',
44
+ type: NodeType.DECISION_TABLE,
45
+ name: 'Variable Table',
46
+ position: { x: 0, y: 0 },
47
+ content: {
48
+ rules: [
49
+ { input1: 'admin', output1: 'test1' },
50
+ { input1: '> 100', output1: 'test2' },
51
+ ],
52
+ inputs: [{ id: 'input1', field: 'input', type: 'string' }],
53
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
54
+ hitPolicy: 'first',
55
+ },
56
+ },
57
+ ],
58
+ edges: [],
59
+ };
60
+
61
+ const optimized = constantFolding(jdm);
62
+ // Non-constant expressions should remain unchanged
63
+ expect(optimized.nodes[0].content.rules[0].input1).toBe('admin');
64
+ expect(optimized.nodes[0].content.rules[1].input1).toBe('> 100');
65
+ });
66
+ });
67
+
68
+ describe('Expression Inlining', () => {
69
+ it('should inline sum() function calls with constant arguments', () => {
70
+ const jdm = {
71
+ nodes: [
72
+ {
73
+ id: 'table1',
74
+ type: NodeType.DECISION_TABLE,
75
+ name: 'Inline Table',
76
+ position: { x: 0, y: 0 },
77
+ content: {
78
+ rules: [
79
+ { input1: 'sum(1, 2, 3)', output1: 'test1' },
80
+ { input1: 'sum(10, 20, 30)', output1: 'test2' },
81
+ ],
82
+ inputs: [{ id: 'input1', field: 'input', type: 'number' }],
83
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
84
+ hitPolicy: 'first',
85
+ },
86
+ },
87
+ ],
88
+ edges: [],
89
+ };
90
+
91
+ const optimized = expressionInlining(jdm);
92
+
93
+ expect(optimized.nodes[0].content.rules[0].input1).toBe('6');
94
+ expect(optimized.nodes[0].content.rules[1].input1).toBe('60');
95
+ });
96
+
97
+ it('should inline max() function calls', () => {
98
+ const jdm = {
99
+ nodes: [
100
+ {
101
+ id: 'table1',
102
+ type: NodeType.DECISION_TABLE,
103
+ name: 'Max Table',
104
+ position: { x: 0, y: 0 },
105
+ content: {
106
+ rules: [
107
+ { input1: 'max(10, 5)', output1: 'test1' },
108
+ { input1: 'max(3, 7)', output1: 'test2' },
109
+ ],
110
+ inputs: [{ id: 'input1', field: 'input', type: 'number' }],
111
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
112
+ hitPolicy: 'first',
113
+ },
114
+ },
115
+ ],
116
+ edges: [],
117
+ };
118
+
119
+ const optimized = expressionInlining(jdm);
120
+
121
+ expect(optimized.nodes[0].content.rules[0].input1).toBe('10');
122
+ expect(optimized.nodes[0].content.rules[1].input1).toBe('7');
123
+ });
124
+
125
+ it('should inline min() function calls', () => {
126
+ const jdm = {
127
+ nodes: [
128
+ {
129
+ id: 'table1',
130
+ type: NodeType.DECISION_TABLE,
131
+ name: 'Min Table',
132
+ position: { x: 0, y: 0 },
133
+ content: {
134
+ rules: [
135
+ { input1: 'min(10, 5)', output1: 'test1' },
136
+ { input1: 'min(3, 7)', output1: 'test2' },
137
+ ],
138
+ inputs: [{ id: 'input1', field: 'input', type: 'number' }],
139
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
140
+ hitPolicy: 'first',
141
+ },
142
+ },
143
+ ],
144
+ edges: [],
145
+ };
146
+
147
+ const optimized = expressionInlining(jdm);
148
+
149
+ expect(optimized.nodes[0].content.rules[0].input1).toBe('5');
150
+ expect(optimized.nodes[0].content.rules[1].input1).toBe('3');
151
+ });
152
+
153
+ it('should not inline non-constant function calls', () => {
154
+ const jdm = {
155
+ nodes: [
156
+ {
157
+ id: 'table1',
158
+ type: NodeType.DECISION_TABLE,
159
+ name: 'Variable Function Table',
160
+ position: { x: 0, y: 0 },
161
+ content: {
162
+ rules: [
163
+ { input1: 'sum(x, y, z)', output1: 'test1' },
164
+ { input1: 'max(price, discountedPrice)', output1: 'test2' },
165
+ ],
166
+ inputs: [{ id: 'input1', field: 'input', type: 'number' }],
167
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
168
+ hitPolicy: 'first',
169
+ },
170
+ },
171
+ ],
172
+ edges: [],
173
+ };
174
+
175
+ const optimized = expressionInlining(jdm);
176
+
177
+ // Non-constant function calls should remain unchanged
178
+ expect(optimized.nodes[0].content.rules[0].input1).toBe('sum(x, y, z)');
179
+ expect(optimized.nodes[0].content.rules[1].input1).toBe('max(price, discountedPrice)');
180
+ });
181
+
182
+ it('should inline abs() function calls with negative numbers', () => {
183
+ const jdm = {
184
+ nodes: [
185
+ {
186
+ id: 'table1',
187
+ type: NodeType.DECISION_TABLE,
188
+ name: 'Abs Table',
189
+ position: { x: 0, y: 0 },
190
+ content: {
191
+ rules: [
192
+ { input1: 'abs(-5)', output1: 'test1' },
193
+ { input1: 'abs(-3.14)', output1: 'test2' },
194
+ { input1: 'abs(10)', output1: 'test3' },
195
+ ],
196
+ inputs: [{ id: 'input1', field: 'input', type: 'number' }],
197
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
198
+ hitPolicy: 'first',
199
+ },
200
+ },
201
+ ],
202
+ edges: [],
203
+ };
204
+
205
+ const optimized = expressionInlining(jdm);
206
+
207
+ expect(optimized.nodes[0].content.rules[0].input1).toBe('5');
208
+ expect(optimized.nodes[0].content.rules[1].input1).toBe('3.14');
209
+ expect(optimized.nodes[0].content.rules[2].input1).toBe('10');
210
+ });
211
+
212
+ it('should inline floor() function calls with decimal numbers', () => {
213
+ const jdm = {
214
+ nodes: [
215
+ {
216
+ id: 'table1',
217
+ type: NodeType.DECISION_TABLE,
218
+ name: 'Floor Table',
219
+ position: { x: 0, y: 0 },
220
+ content: {
221
+ rules: [
222
+ { input1: 'floor(3.7)', output1: 'test1' },
223
+ { input1: 'floor(-2.3)', output1: 'test2' },
224
+ { input1: 'floor(10)', output1: 'test3' },
225
+ ],
226
+ inputs: [{ id: 'input1', field: 'input', type: 'number' }],
227
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
228
+ hitPolicy: 'first',
229
+ },
230
+ },
231
+ ],
232
+ edges: [],
233
+ };
234
+
235
+ const optimized = expressionInlining(jdm);
236
+
237
+ expect(optimized.nodes[0].content.rules[0].input1).toBe('3');
238
+ expect(optimized.nodes[0].content.rules[1].input1).toBe('-3');
239
+ expect(optimized.nodes[0].content.rules[2].input1).toBe('10');
240
+ });
241
+
242
+ it('should inline ceil() function calls with decimal numbers', () => {
243
+ const jdm = {
244
+ nodes: [
245
+ {
246
+ id: 'table1',
247
+ type: NodeType.DECISION_TABLE,
248
+ name: 'Ceil Table',
249
+ position: { x: 0, y: 0 },
250
+ content: {
251
+ rules: [
252
+ { input1: 'ceil(3.2)', output1: 'test1' },
253
+ { input1: 'ceil(-2.7)', output1: 'test2' },
254
+ { input1: 'ceil(10)', output1: 'test3' },
255
+ ],
256
+ inputs: [{ id: 'input1', field: 'input', type: 'number' }],
257
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
258
+ hitPolicy: 'first',
259
+ },
260
+ },
261
+ ],
262
+ edges: [],
263
+ };
264
+
265
+ const optimized = expressionInlining(jdm);
266
+
267
+ expect(optimized.nodes[0].content.rules[0].input1).toBe('4');
268
+ expect(optimized.nodes[0].content.rules[1].input1).toBe('-2');
269
+ expect(optimized.nodes[0].content.rules[2].input1).toBe('10');
270
+ });
271
+
272
+ it('should inline round() function calls with decimal numbers', () => {
273
+ const jdm = {
274
+ nodes: [
275
+ {
276
+ id: 'table1',
277
+ type: NodeType.DECISION_TABLE,
278
+ name: 'Round Table',
279
+ position: { x: 0, y: 0 },
280
+ content: {
281
+ rules: [
282
+ { input1: 'round(3.5)', output1: 'test1' },
283
+ { input1: 'round(3.4)', output1: 'test2' },
284
+ { input1: 'round(-2.5)', output1: 'test3' },
285
+ { input1: 'round(-2.4)', output1: 'test4' },
286
+ ],
287
+ inputs: [{ id: 'input1', field: 'input', type: 'number' }],
288
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
289
+ hitPolicy: 'first',
290
+ },
291
+ },
292
+ ],
293
+ edges: [],
294
+ };
295
+
296
+ const optimized = expressionInlining(jdm);
297
+
298
+ expect(optimized.nodes[0].content.rules[0].input1).toBe('4');
299
+ expect(optimized.nodes[0].content.rules[1].input1).toBe('3');
300
+ expect(optimized.nodes[0].content.rules[2].input1).toBe('-2'); // JavaScript rounds -2.5 to -2
301
+ expect(optimized.nodes[0].content.rules[3].input1).toBe('-2');
302
+ });
303
+
304
+ it('should inline upper() function calls with constant strings', () => {
305
+ const jdm = {
306
+ nodes: [
307
+ {
308
+ id: 'table1',
309
+ type: NodeType.DECISION_TABLE,
310
+ name: 'Upper Table',
311
+ position: { x: 0, y: 0 },
312
+ content: {
313
+ rules: [
314
+ { input1: 'upper("hello")', output1: 'test1' },
315
+ { input1: 'upper("WORLD")', output1: 'test2' },
316
+ { input1: 'upper("MiXeD")', output1: 'test3' },
317
+ ],
318
+ inputs: [{ id: 'input1', field: 'input', type: 'number' }],
319
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
320
+ hitPolicy: 'first',
321
+ },
322
+ },
323
+ ],
324
+ edges: [],
325
+ };
326
+
327
+ const optimized = expressionInlining(jdm);
328
+
329
+ expect(optimized.nodes[0].content.rules[0].input1).toBe('"HELLO"');
330
+ expect(optimized.nodes[0].content.rules[1].input1).toBe('"WORLD"');
331
+ expect(optimized.nodes[0].content.rules[2].input1).toBe('"MIXED"');
332
+ });
333
+
334
+ it('should inline lower() function calls with constant strings', () => {
335
+ const jdm = {
336
+ nodes: [
337
+ {
338
+ id: 'table1',
339
+ type: NodeType.DECISION_TABLE,
340
+ name: 'Lower Table',
341
+ position: { x: 0, y: 0 },
342
+ content: {
343
+ rules: [
344
+ { input1: 'lower("HELLO")', output1: 'test1' },
345
+ { input1: 'lower("world")', output1: 'test2' },
346
+ { input1: 'lower("MiXeD")', output1: 'test3' },
347
+ ],
348
+ inputs: [{ id: 'input1', field: 'input', type: 'number' }],
349
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
350
+ hitPolicy: 'first',
351
+ },
352
+ },
353
+ ],
354
+ edges: [],
355
+ };
356
+
357
+ const optimized = expressionInlining(jdm);
358
+
359
+ expect(optimized.nodes[0].content.rules[0].input1).toBe('"hello"');
360
+ expect(optimized.nodes[0].content.rules[1].input1).toBe('"world"');
361
+ expect(optimized.nodes[0].content.rules[2].input1).toBe('"mixed"');
362
+ });
363
+
364
+ it('should inline len() function calls with constant strings', () => {
365
+ const jdm = {
366
+ nodes: [
367
+ {
368
+ id: 'table1',
369
+ type: NodeType.DECISION_TABLE,
370
+ name: 'Len String Table',
371
+ position: { x: 0, y: 0 },
372
+ content: {
373
+ rules: [
374
+ { input1: 'len("hello")', output1: 'test1' },
375
+ { input1: 'len("")', output1: 'test2' },
376
+ { input1: 'len("ABCD")', output1: 'test3' },
377
+ ],
378
+ inputs: [{ id: 'input1', field: 'input', type: 'number' }],
379
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
380
+ hitPolicy: 'first',
381
+ },
382
+ },
383
+ ],
384
+ edges: [],
385
+ };
386
+
387
+ const optimized = expressionInlining(jdm);
388
+
389
+ expect(optimized.nodes[0].content.rules[0].input1).toBe('5');
390
+ expect(optimized.nodes[0].content.rules[1].input1).toBe('0');
391
+ expect(optimized.nodes[0].content.rules[2].input1).toBe('4');
392
+ });
393
+
394
+ it('should inline len() function calls with constant numbers', () => {
395
+ const jdm = {
396
+ nodes: [
397
+ {
398
+ id: 'table1',
399
+ type: NodeType.DECISION_TABLE,
400
+ name: 'Len Number Table',
401
+ position: { x: 0, y: 0 },
402
+ content: {
403
+ rules: [
404
+ { input1: 'len(123)', output1: 'test1' },
405
+ { input1: 'len(45678)', output1: 'test2' },
406
+ { input1: 'len(-987)', output1: 'test3' },
407
+ { input1: 'len(12.34)', output1: 'test4' },
408
+ ],
409
+ inputs: [{ id: 'input1', field: 'input', type: 'number' }],
410
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
411
+ hitPolicy: 'first',
412
+ },
413
+ },
414
+ ],
415
+ edges: [],
416
+ };
417
+
418
+ const optimized = expressionInlining(jdm);
419
+
420
+ expect(optimized.nodes[0].content.rules[0].input1).toBe('3');
421
+ expect(optimized.nodes[0].content.rules[1].input1).toBe('5');
422
+ expect(optimized.nodes[0].content.rules[2].input1).toBe('3'); // Sign doesn't count
423
+ expect(optimized.nodes[0].content.rules[3].input1).toBe('4'); // Decimal point doesn't count
424
+ });
425
+
426
+ it('should handle multiple function inlining in the same table', () => {
427
+ const jdm = {
428
+ nodes: [
429
+ {
430
+ id: 'table1',
431
+ type: NodeType.DECISION_TABLE,
432
+ name: 'Mixed Table',
433
+ position: { x: 0, y: 0 },
434
+ content: {
435
+ rules: [
436
+ { input1: 'sum(1, 2, 3)', input2: 'abs(-5)', output1: 'test1' },
437
+ { input1: 'upper("hi")', input2: 'floor(3.7)', output1: 'test2' },
438
+ { input1: 'len(123)', input2: 'max(10, 5)', output1: 'test3' },
439
+ ],
440
+ inputs: [
441
+ { id: 'input1', field: 'input1', type: 'number' },
442
+ { id: 'input2', field: 'input2', type: 'number' },
443
+ ],
444
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
445
+ hitPolicy: 'first',
446
+ },
447
+ },
448
+ ],
449
+ edges: [],
450
+ };
451
+
452
+ const optimized = expressionInlining(jdm);
453
+
454
+ expect(optimized.nodes[0].content.rules[0].input1).toBe('6');
455
+ expect(optimized.nodes[0].content.rules[0].input2).toBe('5');
456
+ expect(optimized.nodes[0].content.rules[1].input1).toBe('"HI"');
457
+ expect(optimized.nodes[0].content.rules[1].input2).toBe('3');
458
+ expect(optimized.nodes[0].content.rules[2].input1).toBe('3');
459
+ expect(optimized.nodes[0].content.rules[2].input2).toBe('10');
460
+ });
461
+ });
462
+
463
+ describe('Table Indexing', () => {
464
+ it('should mark inputs as indexable when they have unique literal values', () => {
465
+ const jdm = {
466
+ nodes: [
467
+ {
468
+ id: 'table1',
469
+ type: NodeType.DECISION_TABLE,
470
+ name: 'Indexing Table',
471
+ position: { x: 0, y: 0 },
472
+ content: {
473
+ rules: [
474
+ { input1: '"admin"', input2: '100', output1: 'result1' },
475
+ { input1: '"user"', input2: '200', output1: 'result2' },
476
+ { input1: '"guest"', input2: '300', output1: 'result3' },
477
+ ],
478
+ inputs: [
479
+ { id: 'input1', field: 'role', type: 'string' },
480
+ { id: 'input2', field: 'level', type: 'number' },
481
+ ],
482
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
483
+ hitPolicy: 'first',
484
+ },
485
+ },
486
+ ],
487
+ edges: [],
488
+ };
489
+
490
+ const optimized = tableIndexing(jdm);
491
+
492
+ // Both inputs should be marked as indexable
493
+ expect(optimized.nodes[0].content.inputs[0]._indexable).toBe(true);
494
+ expect(optimized.nodes[0].content.inputs[1]._indexable).toBe(true);
495
+
496
+ // Unique values should be captured
497
+ expect(optimized.nodes[0].content.inputs[0]._uniqueValues).toEqual([
498
+ '"admin"',
499
+ '"user"',
500
+ '"guest"',
501
+ ]);
502
+ expect(optimized.nodes[0].content.inputs[1]._uniqueValues).toEqual(['100', '200', '300']);
503
+ });
504
+
505
+ it('should not mark inputs with duplicate values as indexable', () => {
506
+ const jdm = {
507
+ nodes: [
508
+ {
509
+ id: 'table1',
510
+ type: NodeType.DECISION_TABLE,
511
+ name: 'Duplicate Table',
512
+ position: { x: 0, y: 0 },
513
+ content: {
514
+ rules: [
515
+ { input1: '"admin"', output1: 'result1' },
516
+ { input1: '"admin"', output1: 'result2' },
517
+ { input1: '"user"', output1: 'result3' },
518
+ ],
519
+ inputs: [{ id: 'input1', field: 'role', type: 'string' }],
520
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
521
+ hitPolicy: 'first',
522
+ },
523
+ },
524
+ ],
525
+ edges: [],
526
+ };
527
+
528
+ const optimized = tableIndexing(jdm);
529
+
530
+ // Input should NOT be indexable because of duplicates
531
+ expect(optimized.nodes[0].content.inputs[0]._indexable).toBe(false);
532
+ });
533
+
534
+ it('should not mark inputs with many unique values as indexable', () => {
535
+ const jdm = {
536
+ nodes: [
537
+ {
538
+ id: 'table1',
539
+ type: NodeType.DECISION_TABLE,
540
+ name: 'Large Value Table',
541
+ position: { x: 0, y: 0 },
542
+ content: {
543
+ rules: Array.from({ length: 60 }, (_, i) => ({
544
+ input1: `"value${i}"`,
545
+ output1: `result${i}`,
546
+ })),
547
+ inputs: [{ id: 'input1', field: 'key', type: 'string' }],
548
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
549
+ hitPolicy: 'first',
550
+ },
551
+ },
552
+ ],
553
+ edges: [],
554
+ };
555
+
556
+ const optimized = tableIndexing(jdm);
557
+
558
+ // Input should NOT be indexable because value count exceeds threshold
559
+ expect(optimized.nodes[0].content.inputs[0]._indexable).toBe(false);
560
+ });
561
+ });
562
+
563
+ describe('runOptimizations', () => {
564
+ it('should run all optimization passes in sequence', () => {
565
+ const jdm = {
566
+ nodes: [
567
+ {
568
+ id: 'table1',
569
+ type: NodeType.DECISION_TABLE,
570
+ name: 'Multi-Opt Table',
571
+ position: { x: 0, y: 0 },
572
+ content: {
573
+ rules: [
574
+ { input1: 'sum(1, 2)', output1: 'test1' },
575
+ { input1: 'sum(10, 20)', output1: 'test2' },
576
+ { input1: 'admin', output1: 'test3' },
577
+ ],
578
+ inputs: [{ id: 'input1', field: 'input', type: 'string' }],
579
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
580
+ hitPolicy: 'first',
581
+ },
582
+ },
583
+ ],
584
+ edges: [],
585
+ };
586
+
587
+ const optimized = runOptimizations(jdm);
588
+
589
+ // Expression inlining should turn sum(1, 2) into 3 and sum(10, 20) into 30
590
+ expect(optimized.nodes[0].content.rules.length).toBe(3);
591
+ expect(optimized.nodes[0].content.rules[0].input1).toBe('3');
592
+ expect(optimized.nodes[0].content.rules[1].input1).toBe('30');
593
+ expect(optimized.nodes[0].content.rules[2].input1).toBe('admin');
594
+ });
595
+
596
+ it('should preserve the original AST structure', () => {
597
+ const jdm = {
598
+ nodes: [
599
+ {
600
+ id: 'table1',
601
+ type: NodeType.DECISION_TABLE,
602
+ name: 'Structure Test',
603
+ position: { x: 0, y: 0 },
604
+ content: {
605
+ rules: [{ input1: 'admin', output1: 'test' }],
606
+ inputs: [{ id: 'input1', field: 'input', type: 'string' }],
607
+ outputs: [{ id: 'output1', field: 'output', type: 'string' }],
608
+ hitPolicy: 'first',
609
+ },
610
+ },
611
+ ],
612
+ edges: [],
613
+ };
614
+
615
+ const optimized = runOptimizations(jdm);
616
+
617
+ // Structure should be preserved
618
+ expect(optimized.nodes).toHaveLength(1);
619
+ expect(optimized.nodes[0].id).toBe('table1');
620
+ expect(optimized.nodes[0].type).toBe(NodeType.DECISION_TABLE);
621
+ expect(optimized.nodes[0].name).toBe('Structure Test');
622
+ expect(optimized.nodes[0].position).toEqual({ x: 0, y: 0 });
623
+ expect(optimized.edges).toEqual([]);
624
+ });
625
+ });