@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,878 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import {
3
+ buildGraph,
4
+ validateGraph,
5
+ getEvaluationOrder,
6
+ getInputNodes,
7
+ getOutputNodes,
8
+ } from '../../../src/compiler/graph';
9
+ import { NodeType, type JDMDecision } from '../../../src/compiler/parser';
10
+ import { CompilationError } from '../../../src/compiler/errors';
11
+
12
+ describe('Graph Construction & Validation', () => {
13
+ describe('buildGraph', () => {
14
+ const validJDM: JDMDecision = {
15
+ nodes: [
16
+ {
17
+ id: 'input1',
18
+ type: NodeType.INPUT,
19
+ name: 'Input Node',
20
+ position: { x: 0, y: 0 },
21
+ },
22
+ {
23
+ id: 'expr1',
24
+ type: NodeType.EXPRESSION,
25
+ name: 'Expression Node',
26
+ position: { x: 100, y: 0 },
27
+ content: { expression: 'a + b' },
28
+ },
29
+ {
30
+ id: 'output1',
31
+ type: NodeType.OUTPUT,
32
+ name: 'Output Node',
33
+ position: { x: 200, y: 0 },
34
+ },
35
+ ],
36
+ edges: [
37
+ {
38
+ id: 'edge1',
39
+ type: 'direct',
40
+ sourceId: 'input1',
41
+ targetId: 'expr1',
42
+ },
43
+ {
44
+ id: 'edge2',
45
+ type: 'direct',
46
+ sourceId: 'expr1',
47
+ targetId: 'output1',
48
+ },
49
+ ],
50
+ };
51
+
52
+ it('should build a graph from a valid JDM', () => {
53
+ const graph = buildGraph(validJDM);
54
+ expect(graph.order).toBe(3); // 3 nodes
55
+ expect(graph.size).toBe(2); // 2 edges
56
+ });
57
+
58
+ it('should throw error for edge referencing non-existent node', () => {
59
+ const invalidJDMSource: JDMDecision = {
60
+ nodes: [
61
+ {
62
+ id: 'output1',
63
+ type: NodeType.OUTPUT,
64
+ name: 'Output Node',
65
+ position: { x: 0, y: 0 },
66
+ },
67
+ ],
68
+ edges: [
69
+ {
70
+ id: 'edge1',
71
+ type: 'direct',
72
+ sourceId: 'nonexistent',
73
+ targetId: 'output1',
74
+ },
75
+ ],
76
+ };
77
+
78
+ expect(() => buildGraph(invalidJDMSource)).toThrow(CompilationError);
79
+ expect(() => buildGraph(invalidJDMSource)).toThrow('Invalid edge');
80
+ expect(() => buildGraph(invalidJDMSource)).toThrow('references non-existent node');
81
+
82
+ const invalidJDMTarget: JDMDecision = {
83
+ nodes: [
84
+ {
85
+ id: 'input1',
86
+ type: NodeType.INPUT,
87
+ name: 'Input Node',
88
+ position: { x: 0, y: 0 },
89
+ },
90
+ ],
91
+ edges: [
92
+ {
93
+ id: 'edge1',
94
+ type: 'direct',
95
+ sourceId: 'input1',
96
+ targetId: 'nonexistent',
97
+ },
98
+ ],
99
+ };
100
+
101
+ expect(() => buildGraph(invalidJDMTarget)).toThrow(CompilationError);
102
+ expect(() => buildGraph(invalidJDMTarget)).toThrow('Invalid edge');
103
+ expect(() => buildGraph(invalidJDMTarget)).toThrow('references non-existent node');
104
+ });
105
+
106
+ it('should handle JDM with no edges', () => {
107
+ const jdmNoEdges: JDMDecision = {
108
+ nodes: [
109
+ {
110
+ id: 'input1',
111
+ type: NodeType.INPUT,
112
+ name: 'Input Node',
113
+ position: { x: 0, y: 0 },
114
+ },
115
+ {
116
+ id: 'output1',
117
+ type: NodeType.OUTPUT,
118
+ name: 'Output Node',
119
+ position: { x: 100, y: 0 },
120
+ },
121
+ ],
122
+ edges: [],
123
+ };
124
+
125
+ const graph = buildGraph(jdmNoEdges);
126
+ expect(graph.order).toBe(2);
127
+ expect(graph.size).toBe(0);
128
+ });
129
+ });
130
+
131
+ describe('validateGraph', () => {
132
+ it('should validate a graph with exactly one input and at least one output', () => {
133
+ const jdm: JDMDecision = {
134
+ nodes: [
135
+ {
136
+ id: 'input1',
137
+ type: NodeType.INPUT,
138
+ name: 'Input Node',
139
+ position: { x: 0, y: 0 },
140
+ },
141
+ {
142
+ id: 'output1',
143
+ type: NodeType.OUTPUT,
144
+ name: 'Output Node',
145
+ position: { x: 100, y: 0 },
146
+ },
147
+ ],
148
+ edges: [
149
+ {
150
+ id: 'edge1',
151
+ type: 'direct',
152
+ sourceId: 'input1',
153
+ targetId: 'output1',
154
+ },
155
+ ],
156
+ };
157
+
158
+ const graph = buildGraph(jdm);
159
+ expect(() => validateGraph(graph)).not.toThrow();
160
+ });
161
+
162
+ it('should validate graph with single input and multiple outputs', () => {
163
+ const jdm: JDMDecision = {
164
+ nodes: [
165
+ {
166
+ id: 'input1',
167
+ type: NodeType.INPUT,
168
+ name: 'Input Node',
169
+ position: { x: 0, y: 0 },
170
+ },
171
+ {
172
+ id: 'output1',
173
+ type: NodeType.OUTPUT,
174
+ name: 'Output Node 1',
175
+ position: { x: 100, y: 0 },
176
+ },
177
+ {
178
+ id: 'output2',
179
+ type: NodeType.OUTPUT,
180
+ name: 'Output Node 2',
181
+ position: { x: 200, y: 0 },
182
+ },
183
+ ],
184
+ edges: [
185
+ {
186
+ id: 'edge1',
187
+ type: 'direct',
188
+ sourceId: 'input1',
189
+ targetId: 'output1',
190
+ },
191
+ {
192
+ id: 'edge2',
193
+ type: 'direct',
194
+ sourceId: 'input1',
195
+ targetId: 'output2',
196
+ },
197
+ ],
198
+ };
199
+
200
+ const graph = buildGraph(jdm);
201
+ expect(() => validateGraph(graph)).not.toThrow();
202
+ });
203
+
204
+ it('should throw CompilationError when graph contains cycles', () => {
205
+ const jdmWithCycle: JDMDecision = {
206
+ nodes: [
207
+ {
208
+ id: 'input1',
209
+ type: NodeType.INPUT,
210
+ name: 'Input Node',
211
+ position: { x: 0, y: 0 },
212
+ },
213
+ {
214
+ id: 'expr1',
215
+ type: NodeType.EXPRESSION,
216
+ name: 'Expression Node 1',
217
+ position: { x: 100, y: 0 },
218
+ },
219
+ {
220
+ id: 'expr2',
221
+ type: NodeType.EXPRESSION,
222
+ name: 'Expression Node 2',
223
+ position: { x: 200, y: 0 },
224
+ },
225
+ {
226
+ id: 'output1',
227
+ type: NodeType.OUTPUT,
228
+ name: 'Output Node',
229
+ position: { x: 300, y: 0 },
230
+ },
231
+ ],
232
+ edges: [
233
+ {
234
+ id: 'edge1',
235
+ type: 'direct',
236
+ sourceId: 'input1',
237
+ targetId: 'expr1',
238
+ },
239
+ {
240
+ id: 'edge2',
241
+ type: 'direct',
242
+ sourceId: 'expr1',
243
+ targetId: 'expr2',
244
+ },
245
+ {
246
+ id: 'edge3',
247
+ type: 'direct',
248
+ sourceId: 'expr2',
249
+ targetId: 'expr1', // Creates a cycle: expr1 -> expr2 -> expr1
250
+ },
251
+ ],
252
+ };
253
+
254
+ const graph = buildGraph(jdmWithCycle);
255
+ expect(() => validateGraph(graph)).toThrow(CompilationError);
256
+ expect(() => validateGraph(graph)).toThrow('Circular dependency detected');
257
+ expect(() => validateGraph(graph)).toThrow(
258
+ 'Expression Node 1 -> Expression Node 2 -> Expression Node 1',
259
+ );
260
+ });
261
+
262
+ it('should throw CompilationError for self-referencing nodes', () => {
263
+ const jdmWithSelfRef: JDMDecision = {
264
+ nodes: [
265
+ {
266
+ id: 'input1',
267
+ type: NodeType.INPUT,
268
+ name: 'Input Node',
269
+ position: { x: 0, y: 0 },
270
+ },
271
+ {
272
+ id: 'expr1',
273
+ type: NodeType.EXPRESSION,
274
+ name: 'Expression Node',
275
+ position: { x: 100, y: 0 },
276
+ },
277
+ {
278
+ id: 'output1',
279
+ type: NodeType.OUTPUT,
280
+ name: 'Output Node',
281
+ position: { x: 200, y: 0 },
282
+ },
283
+ ],
284
+ edges: [
285
+ {
286
+ id: 'edge1',
287
+ type: 'direct',
288
+ sourceId: 'input1',
289
+ targetId: 'expr1',
290
+ },
291
+ {
292
+ id: 'edge2',
293
+ type: 'direct',
294
+ sourceId: 'expr1',
295
+ targetId: 'expr1', // Self-reference: expr1 -> expr1
296
+ },
297
+ {
298
+ id: 'edge3',
299
+ type: 'direct',
300
+ sourceId: 'expr1',
301
+ targetId: 'output1',
302
+ },
303
+ ],
304
+ };
305
+
306
+ expect(() => buildGraph(jdmWithSelfRef)).toThrow(CompilationError);
307
+ expect(() => buildGraph(jdmWithSelfRef)).toThrow('Self-reference detected');
308
+ expect(() => buildGraph(jdmWithSelfRef)).toThrow('references itself');
309
+ expect(() => buildGraph(jdmWithSelfRef)).toThrow('Expression Node');
310
+ });
311
+
312
+ it('should throw CompilationError when graph has no input nodes', () => {
313
+ const jdmNoInput: JDMDecision = {
314
+ nodes: [
315
+ {
316
+ id: 'expr1',
317
+ type: NodeType.EXPRESSION,
318
+ name: 'Expression Node',
319
+ position: { x: 0, y: 0 },
320
+ },
321
+ {
322
+ id: 'output1',
323
+ type: NodeType.OUTPUT,
324
+ name: 'Output Node',
325
+ position: { x: 100, y: 0 },
326
+ },
327
+ ],
328
+ edges: [
329
+ {
330
+ id: 'edge1',
331
+ type: 'direct',
332
+ sourceId: 'expr1',
333
+ targetId: 'output1',
334
+ },
335
+ ],
336
+ };
337
+
338
+ const graph = buildGraph(jdmNoInput);
339
+ expect(() => validateGraph(graph)).toThrow(CompilationError);
340
+ expect(() => validateGraph(graph)).toThrow('Expected exactly 1 input node, found 0');
341
+ });
342
+
343
+ it('should throw CompilationError when graph has multiple input nodes', () => {
344
+ const jdmMultipleInputs: JDMDecision = {
345
+ nodes: [
346
+ {
347
+ id: 'input1',
348
+ type: NodeType.INPUT,
349
+ name: 'Input Node 1',
350
+ position: { x: 0, y: 0 },
351
+ },
352
+ {
353
+ id: 'input2',
354
+ type: NodeType.INPUT,
355
+ name: 'Input Node 2',
356
+ position: { x: 100, y: 0 },
357
+ },
358
+ {
359
+ id: 'output1',
360
+ type: NodeType.OUTPUT,
361
+ name: 'Output Node',
362
+ position: { x: 200, y: 0 },
363
+ },
364
+ ],
365
+ edges: [
366
+ {
367
+ id: 'edge1',
368
+ type: 'direct',
369
+ sourceId: 'input1',
370
+ targetId: 'output1',
371
+ },
372
+ {
373
+ id: 'edge2',
374
+ type: 'direct',
375
+ sourceId: 'input2',
376
+ targetId: 'output1',
377
+ },
378
+ ],
379
+ };
380
+
381
+ const graph = buildGraph(jdmMultipleInputs);
382
+ expect(() => validateGraph(graph)).toThrow(CompilationError);
383
+ expect(() => validateGraph(graph)).toThrow('Expected exactly 1 input node, found 2');
384
+ });
385
+
386
+ it('should throw CompilationError when graph has no output nodes', () => {
387
+ const jdmNoOutput: JDMDecision = {
388
+ nodes: [
389
+ {
390
+ id: 'input1',
391
+ type: NodeType.INPUT,
392
+ name: 'Input Node',
393
+ position: { x: 0, y: 0 },
394
+ },
395
+ {
396
+ id: 'expr1',
397
+ type: NodeType.EXPRESSION,
398
+ name: 'Expression Node',
399
+ position: { x: 100, y: 0 },
400
+ },
401
+ ],
402
+ edges: [
403
+ {
404
+ id: 'edge1',
405
+ type: 'direct',
406
+ sourceId: 'input1',
407
+ targetId: 'expr1',
408
+ },
409
+ ],
410
+ };
411
+
412
+ const graph = buildGraph(jdmNoOutput);
413
+ expect(() => validateGraph(graph)).toThrow(CompilationError);
414
+ expect(() => validateGraph(graph)).toThrow('Expected at least 1 output node, found 0');
415
+ });
416
+ });
417
+
418
+ describe('getEvaluationOrder', () => {
419
+ it('should return nodes in topological order', () => {
420
+ const jdm: JDMDecision = {
421
+ nodes: [
422
+ {
423
+ id: 'input1',
424
+ type: NodeType.INPUT,
425
+ name: 'Input Node',
426
+ position: { x: 0, y: 0 },
427
+ },
428
+ {
429
+ id: 'expr1',
430
+ type: NodeType.EXPRESSION,
431
+ name: 'Expression Node 1',
432
+ position: { x: 100, y: 0 },
433
+ },
434
+ {
435
+ id: 'expr2',
436
+ type: NodeType.EXPRESSION,
437
+ name: 'Expression Node 2',
438
+ position: { x: 200, y: 0 },
439
+ },
440
+ {
441
+ id: 'output1',
442
+ type: NodeType.OUTPUT,
443
+ name: 'Output Node',
444
+ position: { x: 300, y: 0 },
445
+ },
446
+ ],
447
+ edges: [
448
+ {
449
+ id: 'edge1',
450
+ type: 'direct',
451
+ sourceId: 'input1',
452
+ targetId: 'expr1',
453
+ },
454
+ {
455
+ id: 'edge2',
456
+ type: 'direct',
457
+ sourceId: 'expr1',
458
+ targetId: 'expr2',
459
+ },
460
+ {
461
+ id: 'edge3',
462
+ type: 'direct',
463
+ sourceId: 'expr2',
464
+ targetId: 'output1',
465
+ },
466
+ ],
467
+ };
468
+
469
+ const graph = buildGraph(jdm);
470
+ const order = getEvaluationOrder(graph);
471
+
472
+ // Input must come before expressions
473
+ const inputIndex = order.indexOf('input1');
474
+ const expr1Index = order.indexOf('expr1');
475
+ const expr2Index = order.indexOf('expr2');
476
+ const outputIndex = order.indexOf('output1');
477
+
478
+ expect(inputIndex).toBeLessThan(expr1Index);
479
+ expect(expr1Index).toBeLessThan(expr2Index);
480
+ expect(expr2Index).toBeLessThan(outputIndex);
481
+ });
482
+
483
+ it('should handle diamond topology correctly', () => {
484
+ const jdm: JDMDecision = {
485
+ nodes: [
486
+ {
487
+ id: 'input1',
488
+ type: NodeType.INPUT,
489
+ name: 'Input Node',
490
+ position: { x: 0, y: 0 },
491
+ },
492
+ {
493
+ id: 'expr1',
494
+ type: NodeType.EXPRESSION,
495
+ name: 'Expression Node 1',
496
+ position: { x: 100, y: -50 },
497
+ },
498
+ {
499
+ id: 'expr2',
500
+ type: NodeType.EXPRESSION,
501
+ name: 'Expression Node 2',
502
+ position: { x: 100, y: 50 },
503
+ },
504
+ {
505
+ id: 'expr3',
506
+ type: NodeType.EXPRESSION,
507
+ name: 'Expression Node 3',
508
+ position: { x: 200, y: 0 },
509
+ },
510
+ {
511
+ id: 'output1',
512
+ type: NodeType.OUTPUT,
513
+ name: 'Output Node',
514
+ position: { x: 300, y: 0 },
515
+ },
516
+ ],
517
+ edges: [
518
+ {
519
+ id: 'edge1',
520
+ type: 'direct',
521
+ sourceId: 'input1',
522
+ targetId: 'expr1',
523
+ },
524
+ {
525
+ id: 'edge2',
526
+ type: 'direct',
527
+ sourceId: 'input1',
528
+ targetId: 'expr2',
529
+ },
530
+ {
531
+ id: 'edge3',
532
+ type: 'direct',
533
+ sourceId: 'expr1',
534
+ targetId: 'expr3',
535
+ },
536
+ {
537
+ id: 'edge4',
538
+ type: 'direct',
539
+ sourceId: 'expr2',
540
+ targetId: 'expr3',
541
+ },
542
+ {
543
+ id: 'edge5',
544
+ type: 'direct',
545
+ sourceId: 'expr3',
546
+ targetId: 'output1',
547
+ },
548
+ ],
549
+ };
550
+
551
+ const graph = buildGraph(jdm);
552
+ const order = getEvaluationOrder(graph);
553
+
554
+ const inputIndex = order.indexOf('input1');
555
+ const expr1Index = order.indexOf('expr1');
556
+ const expr2Index = order.indexOf('expr2');
557
+ const expr3Index = order.indexOf('expr3');
558
+ const outputIndex = order.indexOf('output1');
559
+
560
+ // Input must come before expr1 and expr2
561
+ expect(inputIndex).toBeLessThan(expr1Index);
562
+ expect(inputIndex).toBeLessThan(expr2Index);
563
+ // Both expr1 and expr2 must come before expr3
564
+ expect(expr1Index).toBeLessThan(expr3Index);
565
+ expect(expr2Index).toBeLessThan(expr3Index);
566
+ // expr3 must come before output
567
+ expect(expr3Index).toBeLessThan(outputIndex);
568
+ });
569
+ });
570
+
571
+ describe('getInputNodes', () => {
572
+ const jdm: JDMDecision = {
573
+ nodes: [
574
+ {
575
+ id: 'input1',
576
+ type: NodeType.INPUT,
577
+ name: 'Input Node',
578
+ position: { x: 0, y: 0 },
579
+ },
580
+ {
581
+ id: 'expr1',
582
+ type: NodeType.EXPRESSION,
583
+ name: 'Expression Node',
584
+ position: { x: 100, y: 0 },
585
+ },
586
+ {
587
+ id: 'output1',
588
+ type: NodeType.OUTPUT,
589
+ name: 'Output Node',
590
+ position: { x: 200, y: 0 },
591
+ },
592
+ ],
593
+ edges: [
594
+ {
595
+ id: 'edge1',
596
+ type: 'direct',
597
+ sourceId: 'input1',
598
+ targetId: 'expr1',
599
+ },
600
+ {
601
+ id: 'edge2',
602
+ type: 'direct',
603
+ sourceId: 'expr1',
604
+ targetId: 'output1',
605
+ },
606
+ ],
607
+ };
608
+
609
+ it('should return all nodes that feed into a target node', () => {
610
+ const graph = buildGraph(jdm);
611
+ const inputs = getInputNodes(graph, 'expr1');
612
+
613
+ expect(inputs).toHaveLength(1);
614
+ expect(inputs[0].id).toBe('input1');
615
+ });
616
+
617
+ it('should return empty array for input node (no incoming edges)', () => {
618
+ const graph = buildGraph(jdm);
619
+ const inputs = getInputNodes(graph, 'input1');
620
+
621
+ expect(inputs).toHaveLength(0);
622
+ });
623
+
624
+ it('should return multiple input nodes for a node with multiple incoming edges', () => {
625
+ const jdmMultiInput: JDMDecision = {
626
+ nodes: [
627
+ {
628
+ id: 'expr1',
629
+ type: NodeType.EXPRESSION,
630
+ name: 'Expression Node 1',
631
+ position: { x: 0, y: -50 },
632
+ },
633
+ {
634
+ id: 'expr2',
635
+ type: NodeType.EXPRESSION,
636
+ name: 'Expression Node 2',
637
+ position: { x: 0, y: 50 },
638
+ },
639
+ {
640
+ id: 'output1',
641
+ type: NodeType.OUTPUT,
642
+ name: 'Output Node',
643
+ position: { x: 100, y: 0 },
644
+ },
645
+ ],
646
+ edges: [
647
+ {
648
+ id: 'edge1',
649
+ type: 'direct',
650
+ sourceId: 'expr1',
651
+ targetId: 'output1',
652
+ },
653
+ {
654
+ id: 'edge2',
655
+ type: 'direct',
656
+ sourceId: 'expr2',
657
+ targetId: 'output1',
658
+ },
659
+ ],
660
+ };
661
+
662
+ const graph = buildGraph(jdmMultiInput);
663
+ const inputs = getInputNodes(graph, 'output1');
664
+
665
+ expect(inputs).toHaveLength(2);
666
+ expect(inputs.map((n) => n.id).sort()).toEqual(['expr1', 'expr2']);
667
+ });
668
+ });
669
+
670
+ describe('getOutputNodes', () => {
671
+ const jdm: JDMDecision = {
672
+ nodes: [
673
+ {
674
+ id: 'input1',
675
+ type: NodeType.INPUT,
676
+ name: 'Input Node',
677
+ position: { x: 0, y: 0 },
678
+ },
679
+ {
680
+ id: 'expr1',
681
+ type: NodeType.EXPRESSION,
682
+ name: 'Expression Node',
683
+ position: { x: 100, y: 0 },
684
+ },
685
+ {
686
+ id: 'output1',
687
+ type: NodeType.OUTPUT,
688
+ name: 'Output Node',
689
+ position: { x: 200, y: 0 },
690
+ },
691
+ ],
692
+ edges: [
693
+ {
694
+ id: 'edge1',
695
+ type: 'direct',
696
+ sourceId: 'input1',
697
+ targetId: 'expr1',
698
+ },
699
+ {
700
+ id: 'edge2',
701
+ type: 'direct',
702
+ sourceId: 'expr1',
703
+ targetId: 'output1',
704
+ },
705
+ ],
706
+ };
707
+
708
+ it('should return all nodes that a source node feeds into', () => {
709
+ const graph = buildGraph(jdm);
710
+ const outputs = getOutputNodes(graph, 'expr1');
711
+
712
+ expect(outputs).toHaveLength(1);
713
+ expect(outputs[0].id).toBe('output1');
714
+ });
715
+
716
+ it('should return empty array for output node (no outgoing edges)', () => {
717
+ const graph = buildGraph(jdm);
718
+ const outputs = getOutputNodes(graph, 'output1');
719
+
720
+ expect(outputs).toHaveLength(0);
721
+ });
722
+
723
+ it('should return multiple output nodes for a node with multiple outgoing edges', () => {
724
+ const jdmMultiOutput: JDMDecision = {
725
+ nodes: [
726
+ {
727
+ id: 'input1',
728
+ type: NodeType.INPUT,
729
+ name: 'Input Node',
730
+ position: { x: 0, y: 0 },
731
+ },
732
+ {
733
+ id: 'expr1',
734
+ type: NodeType.EXPRESSION,
735
+ name: 'Expression Node',
736
+ position: { x: 100, y: 0 },
737
+ },
738
+ {
739
+ id: 'expr2',
740
+ type: NodeType.EXPRESSION,
741
+ name: 'Expression Node 2',
742
+ position: { x: 200, y: -50 },
743
+ },
744
+ {
745
+ id: 'expr3',
746
+ type: NodeType.EXPRESSION,
747
+ name: 'Expression Node 3',
748
+ position: { x: 200, y: 50 },
749
+ },
750
+ ],
751
+ edges: [
752
+ {
753
+ id: 'edge1',
754
+ type: 'direct',
755
+ sourceId: 'input1',
756
+ targetId: 'expr1',
757
+ },
758
+ {
759
+ id: 'edge2',
760
+ type: 'direct',
761
+ sourceId: 'expr1',
762
+ targetId: 'expr2',
763
+ },
764
+ {
765
+ id: 'edge3',
766
+ type: 'direct',
767
+ sourceId: 'expr1',
768
+ targetId: 'expr3',
769
+ },
770
+ ],
771
+ };
772
+
773
+ const graph = buildGraph(jdmMultiOutput);
774
+ const outputs = getOutputNodes(graph, 'expr1');
775
+
776
+ expect(outputs).toHaveLength(2);
777
+ expect(outputs.map((n) => n.id).sort()).toEqual(['expr2', 'expr3']);
778
+ });
779
+ });
780
+
781
+ describe('Integration - End-to-end graph construction and validation', () => {
782
+ it('should successfully build and validate a complex decision graph', () => {
783
+ const jdm: JDMDecision = {
784
+ nodes: [
785
+ {
786
+ id: 'input',
787
+ type: NodeType.INPUT,
788
+ name: 'Request',
789
+ position: { x: 0, y: 0 },
790
+ },
791
+ {
792
+ id: 'risk-calc',
793
+ type: NodeType.EXPRESSION,
794
+ name: 'Risk Assessment',
795
+ position: { x: 100, y: 0 },
796
+ content: { expression: 'request.score > 0.7' },
797
+ },
798
+ {
799
+ id: 'decision-table',
800
+ type: NodeType.DECISION_TABLE,
801
+ name: 'Decision Table',
802
+ position: { x: 200, y: -50 },
803
+ },
804
+ {
805
+ id: 'switch',
806
+ type: NodeType.SWITCH,
807
+ name: 'Region Switch',
808
+ position: { x: 200, y: 50 },
809
+ },
810
+ {
811
+ id: 'output1',
812
+ type: NodeType.OUTPUT,
813
+ name: 'Approval',
814
+ position: { x: 300, y: -50 },
815
+ },
816
+ {
817
+ id: 'output2',
818
+ type: NodeType.OUTPUT,
819
+ name: 'Rejection',
820
+ position: { x: 300, y: 50 },
821
+ },
822
+ ],
823
+ edges: [
824
+ {
825
+ id: 'e1',
826
+ type: 'direct',
827
+ sourceId: 'input',
828
+ targetId: 'risk-calc',
829
+ },
830
+ {
831
+ id: 'e2',
832
+ type: 'direct',
833
+ sourceId: 'risk-calc',
834
+ targetId: 'decision-table',
835
+ },
836
+ {
837
+ id: 'e3',
838
+ type: 'direct',
839
+ sourceId: 'risk-calc',
840
+ targetId: 'switch',
841
+ },
842
+ {
843
+ id: 'e4',
844
+ type: 'direct',
845
+ sourceId: 'decision-table',
846
+ targetId: 'output1',
847
+ },
848
+ {
849
+ id: 'e5',
850
+ type: 'direct',
851
+ sourceId: 'switch',
852
+ targetId: 'output2',
853
+ },
854
+ ],
855
+ };
856
+
857
+ const graph = buildGraph(jdm);
858
+ expect(() => validateGraph(graph)).not.toThrow();
859
+
860
+ const order = getEvaluationOrder(graph);
861
+ const inputIndex = order.indexOf('input');
862
+ const riskCalcIndex = order.indexOf('risk-calc');
863
+ const decisionTableIndex = order.indexOf('decision-table');
864
+ const switchIndex = order.indexOf('switch');
865
+ const output1Index = order.indexOf('output1');
866
+ const output2Index = order.indexOf('output2');
867
+
868
+ // Verify input comes first
869
+ expect(inputIndex).toBeLessThan(riskCalcIndex);
870
+ // Risk calculation comes before its dependents
871
+ expect(riskCalcIndex).toBeLessThan(decisionTableIndex);
872
+ expect(riskCalcIndex).toBeLessThan(switchIndex);
873
+ // Intermediate nodes come before outputs
874
+ expect(decisionTableIndex).toBeLessThan(output1Index);
875
+ expect(switchIndex).toBeLessThan(output2Index);
876
+ });
877
+ });
878
+ });