@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,510 @@
1
+ /**
2
+ * Unit tests for graph module scalability and complex topologies
3
+ */
4
+
5
+ import { describe, it, expect } from 'vitest';
6
+ import { buildGraph, validateGraph, getEvaluationOrder } from '../../../src/compiler/graph';
7
+ import type { JDMDecision, JDMNode, JDMEdge } from '../../../src/compiler/parser';
8
+ import { NodeType } from '../../../src/compiler/parser';
9
+
10
+ function createTestGraph(
11
+ nodes: Array<{ id: string }>,
12
+ edges: Array<{ id: string; sourceId: string; targetId: string }>,
13
+ options?: { hasInput?: boolean; hasOutput?: boolean },
14
+ ): JDMDecision {
15
+ const jdmNodes: JDMNode[] = [];
16
+
17
+ // Add input node first if needed
18
+ if (options?.hasInput !== false) {
19
+ jdmNodes.unshift({
20
+ id: 'input',
21
+ type: NodeType.INPUT,
22
+ name: 'Input',
23
+ position: { x: 0, y: 0 },
24
+ });
25
+ }
26
+
27
+ // Add user-provided nodes as decision nodes
28
+ for (const n of nodes) {
29
+ jdmNodes.push({
30
+ id: n.id,
31
+ type: NodeType.DECISION,
32
+ name: n.id,
33
+ position: { x: 0, y: 0 },
34
+ });
35
+ }
36
+
37
+ // Add output node last if needed
38
+ if (options?.hasOutput !== false) {
39
+ jdmNodes.push({
40
+ id: 'output',
41
+ type: NodeType.OUTPUT,
42
+ name: 'Output',
43
+ position: { x: 0, y: 0 },
44
+ });
45
+ }
46
+
47
+ const jdmEdges: JDMEdge[] = edges.map((e) => ({
48
+ id: e.id,
49
+ type: 'edge',
50
+ sourceId: e.sourceId,
51
+ targetId: e.targetId,
52
+ }));
53
+
54
+ return {
55
+ nodes: jdmNodes,
56
+ edges: jdmEdges,
57
+ };
58
+ }
59
+
60
+ describe('Graph Module - Scaling Tests', () => {
61
+ describe('Large Graph Support (5+ decision nodes)', () => {
62
+ it('should handle linear chain of 10 decision nodes', () => {
63
+ const nodes: Array<{ id: string }> = [
64
+ { id: 'node1' },
65
+ { id: 'node2' },
66
+ { id: 'node3' },
67
+ { id: 'node4' },
68
+ { id: 'node5' },
69
+ { id: 'node6' },
70
+ { id: 'node7' },
71
+ { id: 'node8' },
72
+ { id: 'node9' },
73
+ { id: 'node10' },
74
+ ];
75
+
76
+ const edges: Array<{ id: string; sourceId: string; targetId: string }> = [
77
+ { id: 'e1', sourceId: 'input', targetId: 'node1' },
78
+ { id: 'e2', sourceId: 'node1', targetId: 'node2' },
79
+ { id: 'e3', sourceId: 'node2', targetId: 'node3' },
80
+ { id: 'e4', sourceId: 'node3', targetId: 'node4' },
81
+ { id: 'e5', sourceId: 'node4', targetId: 'node5' },
82
+ { id: 'e6', sourceId: 'node5', targetId: 'node6' },
83
+ { id: 'e7', sourceId: 'node6', targetId: 'node7' },
84
+ { id: 'e8', sourceId: 'node7', targetId: 'node8' },
85
+ { id: 'e9', sourceId: 'node8', targetId: 'node9' },
86
+ { id: 'e10', sourceId: 'node9', targetId: 'node10' },
87
+ { id: 'e11', sourceId: 'node10', targetId: 'output' },
88
+ ];
89
+
90
+ const jdm = createTestGraph(nodes, edges);
91
+ const graph = buildGraph(jdm);
92
+
93
+ validateGraph(graph);
94
+
95
+ const order = getEvaluationOrder(graph);
96
+ expect(order.length).toBeGreaterThan(nodes.length);
97
+
98
+ // Verify linear ordering
99
+ expect(order.indexOf('input')).toBeLessThan(order.indexOf('node1'));
100
+ expect(order.indexOf('node1')).toBeLessThan(order.indexOf('node2'));
101
+ expect(order.indexOf('node9')).toBeLessThan(order.indexOf('node10'));
102
+ expect(order.indexOf('node10')).toBeLessThan(order.indexOf('output'));
103
+ });
104
+
105
+ it('should handle complex DAG with 20+ nodes', () => {
106
+ const nodeCount = 20;
107
+ const nodes: Array<{ id: string }> = [];
108
+ for (let i = 0; i < nodeCount; i++) {
109
+ nodes.push({ id: `node${i}` });
110
+ }
111
+
112
+ const edges: Array<{ id: string; sourceId: string; targetId: string }> = [
113
+ { id: 'e-in', sourceId: 'input', targetId: 'node0' },
114
+ ];
115
+
116
+ // Create a mesh-like structure where each node connects to the next few nodes
117
+ for (let i = 0; i < nodeCount; i++) {
118
+ const connections = Math.min(3, nodeCount - i - 1);
119
+ for (let j = 1; j <= connections; j++) {
120
+ edges.push({
121
+ id: `e-${i}-${i + j}`,
122
+ sourceId: `node${i}`,
123
+ targetId: `node${i + j}`,
124
+ });
125
+ }
126
+ }
127
+
128
+ edges.push({ id: 'e-out', sourceId: `node${nodeCount - 1}`, targetId: 'output' });
129
+
130
+ const jdm = createTestGraph(nodes, edges);
131
+ const graph = buildGraph(jdm);
132
+
133
+ // Should validate without errors (no cycles, proper structure)
134
+ validateGraph(graph);
135
+
136
+ const order = getEvaluationOrder(graph);
137
+ expect(order.length).toBeGreaterThan(nodeCount);
138
+
139
+ // Topological ordering is respected
140
+ expect(order.indexOf('input')).toBeLessThan(order.indexOf('node0'));
141
+ expect(order.indexOf(`node${nodeCount - 1}`)).toBeLessThan(order.indexOf('output'));
142
+ });
143
+
144
+ it('should handle tree-like structure with 15 nodes', () => {
145
+ const nodes: Array<{ id: string }> = [
146
+ { id: 'root' },
147
+ // Level 1
148
+ { id: 'branch1' },
149
+ { id: 'branch2' },
150
+ { id: 'branch3' },
151
+ // Level 2
152
+ { id: 'leaf1-1' },
153
+ { id: 'leaf1-2' },
154
+ { id: 'leaf2-1' },
155
+ { id: 'leaf2-2' },
156
+ { id: 'leaf3-1' },
157
+ // Level 3
158
+ { id: 'subleaf1-1-1' },
159
+ { id: 'subleaf1-1-2' },
160
+ { id: 'subleaf2-1-1' },
161
+ { id: 'subleaf2-1-2' },
162
+ { id: 'subleaf3-1-1' },
163
+ ];
164
+
165
+ const edges: Array<{ id: string; sourceId: string; targetId: string }> = [
166
+ // Input to root
167
+ { id: 'e-in', sourceId: 'input', targetId: 'root' },
168
+ // Root to branches
169
+ { id: 'e-r-1', sourceId: 'root', targetId: 'branch1' },
170
+ { id: 'e-r-2', sourceId: 'root', targetId: 'branch2' },
171
+ { id: 'e-r-3', sourceId: 'root', targetId: 'branch3' },
172
+ // Branch 1 to leaves
173
+ { id: 'e-b1-l1', sourceId: 'branch1', targetId: 'leaf1-1' },
174
+ { id: 'e-b1-l2', sourceId: 'branch1', targetId: 'leaf1-2' },
175
+ // Branch 2 to leaves
176
+ { id: 'e-b2-l1', sourceId: 'branch2', targetId: 'leaf2-1' },
177
+ { id: 'e-b2-l2', sourceId: 'branch2', targetId: 'leaf2-2' },
178
+ // Branch 3 to leaf
179
+ { id: 'e-b3-l1', sourceId: 'branch3', targetId: 'leaf3-1' },
180
+ // Leaves to subleaves
181
+ { id: 'e-l1-1-s1', sourceId: 'leaf1-1', targetId: 'subleaf1-1-1' },
182
+ { id: 'e-l1-1-s2', sourceId: 'leaf1-1', targetId: 'subleaf1-1-2' },
183
+ { id: 'e-l2-1-s1', sourceId: 'leaf2-1', targetId: 'subleaf2-1-1' },
184
+ { id: 'e-l2-1-s2', sourceId: 'leaf2-1', targetId: 'subleaf2-1-2' },
185
+ { id: 'e-l3-1-s1', sourceId: 'leaf3-1', targetId: 'subleaf3-1-1' },
186
+ // Subleaves to output
187
+ { id: 'e-out1', sourceId: 'subleaf1-1-1', targetId: 'output' },
188
+ { id: 'e-out2', sourceId: 'subleaf2-1-1', targetId: 'output' },
189
+ { id: 'e-out3', sourceId: 'subleaf3-1-1', targetId: 'output' },
190
+ ];
191
+
192
+ const jdm = createTestGraph(nodes, edges);
193
+ const graph = buildGraph(jdm);
194
+
195
+ validateGraph(graph);
196
+
197
+ const order = getEvaluationOrder(graph);
198
+
199
+ // Verify tree ordering is respected
200
+ expect(order.indexOf('input')).toBeLessThan(order.indexOf('root'));
201
+ expect(order.indexOf('root')).toBeLessThan(order.indexOf('branch1'));
202
+ expect(order.indexOf('root')).toBeLessThan(order.indexOf('branch2'));
203
+ expect(order.indexOf('branch1')).toBeLessThan(order.indexOf('leaf1-1'));
204
+ expect(order.indexOf('leaf1-1')).toBeLessThan(order.indexOf('subleaf1-1-1'));
205
+ expect(order.indexOf('subleaf1-1-1')).toBeLessThan(order.indexOf('output'));
206
+ });
207
+ });
208
+
209
+ describe('Various Graph Topologies', () => {
210
+ it('should correctly topologically sort diamond pattern (4 decision nodes)', () => {
211
+ const nodes: Array<{ id: string }> = [
212
+ { id: 'node-a' },
213
+ { id: 'node-b' },
214
+ { id: 'node-c' },
215
+ { id: 'node-shared' },
216
+ ];
217
+
218
+ const edges: Array<{ id: string; sourceId: string; targetId: string }> = [
219
+ { id: 'e-1', sourceId: 'input', targetId: 'node-a' },
220
+ { id: 'e-2', sourceId: 'node-a', targetId: 'node-b' },
221
+ { id: 'e-3', sourceId: 'node-a', targetId: 'node-c' },
222
+ { id: 'e-4', sourceId: 'node-b', targetId: 'node-shared' },
223
+ { id: 'e-5', sourceId: 'node-c', targetId: 'node-shared' },
224
+ { id: 'e-6', sourceId: 'node-shared', targetId: 'output' },
225
+ ];
226
+
227
+ const jdm = createTestGraph(nodes, edges);
228
+ const graph = buildGraph(jdm);
229
+
230
+ const order = getEvaluationOrder(graph);
231
+
232
+ // Diamond ordering: a comes before both b and c, both come before shared
233
+ expect(order.indexOf('node-a')).toBeLessThan(order.indexOf('node-b'));
234
+ expect(order.indexOf('node-a')).toBeLessThan(order.indexOf('node-c'));
235
+ expect(order.indexOf('node-b')).toBeLessThan(order.indexOf('node-shared'));
236
+ expect(order.indexOf('node-c')).toBeLessThan(order.indexOf('node-shared'));
237
+ expect(order.indexOf('node-shared')).toBeLessThan(order.indexOf('output'));
238
+ });
239
+
240
+ it('should correctly handle star pattern (fan-out then fan-in)', () => {
241
+ const nodes: Array<{ id: string }> = [
242
+ { id: 'start' },
243
+ { id: 'worker1' },
244
+ { id: 'worker2' },
245
+ { id: 'worker3' },
246
+ { id: 'worker4' },
247
+ { id: 'worker5' },
248
+ { id: 'end' },
249
+ ];
250
+
251
+ const edges: Array<{ id: string; sourceId: string; targetId: string }> = [
252
+ { id: 'e-in', sourceId: 'input', targetId: 'start' },
253
+ { id: 'e-1', sourceId: 'start', targetId: 'worker1' },
254
+ { id: 'e-2', sourceId: 'start', targetId: 'worker2' },
255
+ { id: 'e-3', sourceId: 'start', targetId: 'worker3' },
256
+ { id: 'e-4', sourceId: 'start', targetId: 'worker4' },
257
+ { id: 'e-5', sourceId: 'start', targetId: 'worker5' },
258
+ { id: 'e-6', sourceId: 'worker1', targetId: 'end' },
259
+ { id: 'e-7', sourceId: 'worker2', targetId: 'end' },
260
+ { id: 'e-8', sourceId: 'worker3', targetId: 'end' },
261
+ { id: 'e-9', sourceId: 'worker4', targetId: 'end' },
262
+ { id: 'e-10', sourceId: 'worker5', targetId: 'end' },
263
+ { id: 'e-out', sourceId: 'end', targetId: 'output' },
264
+ ];
265
+
266
+ const jdm = createTestGraph(nodes, edges);
267
+ const graph = buildGraph(jdm);
268
+
269
+ validateGraph(graph);
270
+
271
+ const order = getEvaluationOrder(graph);
272
+
273
+ // All workers come after start, and end comes after all workers
274
+ expect(order.indexOf('start')).toBeLessThan(order.indexOf('worker1'));
275
+ expect(order.indexOf('start')).toBeLessThan(order.indexOf('worker2'));
276
+ expect(order.indexOf('start')).toBeLessThan(order.indexOf('worker3'));
277
+ expect(order.indexOf('start')).toBeLessThan(order.indexOf('worker4'));
278
+ expect(order.indexOf('start')).toBeLessThan(order.indexOf('worker5'));
279
+
280
+ expect(order.indexOf('worker1')).toBeLessThan(order.indexOf('end'));
281
+ expect(order.indexOf('worker2')).toBeLessThan(order.indexOf('end'));
282
+ expect(order.indexOf('worker3')).toBeLessThan(order.indexOf('end'));
283
+ expect(order.indexOf('worker4')).toBeLessThan(order.indexOf('end'));
284
+ expect(order.indexOf('worker5')).toBeLessThan(order.indexOf('end'));
285
+ });
286
+
287
+ it('should handle multiple diamond patterns merged together', () => {
288
+ const nodes: Array<{ id: string }> = [
289
+ { id: 'stage1' },
290
+ // First diamond
291
+ { id: 'd1-a' },
292
+ { id: 'd1-b' },
293
+ { id: 'd1-shared' },
294
+ // Second diamond
295
+ { id: 'd2-a' },
296
+ { id: 'd2-b' },
297
+ { id: 'd2-shared' },
298
+ // Final stage
299
+ { id: 'stage2' },
300
+ ];
301
+
302
+ const edges: Array<{ id: string; sourceId: string; targetId: string }> = [
303
+ // Input to stage1
304
+ { id: 'e-in1', sourceId: 'input', targetId: 'stage1' },
305
+
306
+ // First diamond
307
+ { id: 'e-d1-a', sourceId: 'stage1', targetId: 'd1-a' },
308
+ { id: 'e-d1-b', sourceId: 'stage1', targetId: 'd1-b' },
309
+ { id: 'e-d1-in', sourceId: 'd1-a', targetId: 'd1-shared' },
310
+ { id: 'e-d1-in2', sourceId: 'd1-b', targetId: 'd1-shared' },
311
+
312
+ // Second diamond (feeds from first)
313
+ { id: 'e-d2-a', sourceId: 'd1-shared', targetId: 'd2-a' },
314
+ { id: 'e-d2-b', sourceId: 'd1-shared', targetId: 'd2-b' },
315
+ { id: 'e-d2-in', sourceId: 'd2-a', targetId: 'd2-shared' },
316
+ { id: 'e-d2-in2', sourceId: 'd2-b', targetId: 'd2-shared' },
317
+
318
+ // To final stage
319
+ { id: 'e-stage2', sourceId: 'd2-shared', targetId: 'stage2' },
320
+ { id: 'e-out', sourceId: 'stage2', targetId: 'output' },
321
+ ];
322
+
323
+ const jdm = createTestGraph(nodes, edges);
324
+ const graph = buildGraph(jdm);
325
+
326
+ validateGraph(graph);
327
+
328
+ const order = getEvaluationOrder(graph);
329
+
330
+ // Verify ordering through both diamonds
331
+ expect(order.indexOf('stage1')).toBeLessThan(order.indexOf('d1-a'));
332
+ expect(order.indexOf('d1-b')).toBeLessThan(order.indexOf('d1-shared'));
333
+ expect(order.indexOf('d1-shared')).toBeLessThan(order.indexOf('d2-a'));
334
+ expect(order.indexOf('d2-shared')).toBeLessThan(order.indexOf('stage2'));
335
+ });
336
+ });
337
+
338
+ describe('Error Messages at Scale', () => {
339
+ it('should provide clear cycle detection message for large graph', () => {
340
+ // Create a cycle in a larger graph
341
+ const nodes: Array<{ id: string }> = [
342
+ { id: 'n1' },
343
+ { id: 'n2' },
344
+ { id: 'n3' },
345
+ { id: 'n4' },
346
+ { id: 'n5' },
347
+ ];
348
+
349
+ const edges: Array<{ id: string; sourceId: string; targetId: string }> = [
350
+ { id: 'e1', sourceId: 'input', targetId: 'n1' },
351
+ { id: 'e2', sourceId: 'n1', targetId: 'n2' },
352
+ { id: 'e3', sourceId: 'n2', targetId: 'n3' },
353
+ { id: 'e4', sourceId: 'n3', targetId: 'n4' },
354
+ { id: 'e5', sourceId: 'n4', targetId: 'n5' },
355
+ // Add a cycle back to n2
356
+ { id: 'e6', sourceId: 'n5', targetId: 'n2' },
357
+ { id: 'e7', sourceId: 'n5', targetId: 'output' },
358
+ ];
359
+
360
+ const jdm = createTestGraph(nodes, edges);
361
+ const graph = buildGraph(jdm);
362
+
363
+ // Should throw with clear cycle message
364
+ expect(() => validateGraph(graph)).toThrow('Circular dependency detected');
365
+
366
+ try {
367
+ validateGraph(graph);
368
+ } catch (error) {
369
+ const message = (error as Error).message;
370
+ // Should include the cycle path
371
+ expect(message).toContain('Circular dependency detected');
372
+ }
373
+ });
374
+
375
+ it('should allow multiple output nodes', () => {
376
+ const jdm: JDMDecision = {
377
+ nodes: [
378
+ { id: 'input', type: NodeType.INPUT, name: 'Input', position: { x: 0, y: 0 } },
379
+ { id: 'n1', type: NodeType.DECISION, name: 'n1', position: { x: 0, y: 0 } },
380
+ { id: 'n2', type: NodeType.DECISION, name: 'n2', position: { x: 0, y: 0 } },
381
+ { id: 'n3', type: NodeType.DECISION, name: 'n3', position: { x: 0, y: 0 } },
382
+ { id: 'output1', type: NodeType.OUTPUT, name: 'Output1', position: { x: 0, y: 0 } },
383
+ { id: 'output2', type: NodeType.OUTPUT, name: 'Output2', position: { x: 0, y: 0 } },
384
+ ],
385
+ edges: [
386
+ { id: 'e1', type: 'edge', sourceId: 'input', targetId: 'n1' },
387
+ { id: 'e2', type: 'edge', sourceId: 'n1', targetId: 'output1' },
388
+ { id: 'e3', type: 'edge', sourceId: 'n1', targetId: 'output2' },
389
+ ],
390
+ };
391
+
392
+ const graph = buildGraph(jdm);
393
+
394
+ // Should not throw - multiple outputs are allowed (only single input is enforced)
395
+ validateGraph(graph);
396
+ });
397
+
398
+ it('should error on multiple input nodes', () => {
399
+ const jdm: JDMDecision = {
400
+ nodes: [
401
+ { id: 'input1', type: NodeType.INPUT, name: 'Input1', position: { x: 0, y: 0 } },
402
+ { id: 'input2', type: NodeType.INPUT, name: 'Input2', position: { x: 0, y: 0 } },
403
+ { id: 'n1', type: NodeType.DECISION, name: 'n1', position: { x: 0, y: 0 } },
404
+ { id: 'output', type: NodeType.OUTPUT, name: 'Output', position: { x: 0, y: 0 } },
405
+ ],
406
+ edges: [
407
+ { id: 'e1', type: 'edge', sourceId: 'input1', targetId: 'n1' },
408
+ { id: 'e2', type: 'edge', sourceId: 'input2', targetId: 'n1' },
409
+ { id: 'e3', type: 'edge', sourceId: 'n1', targetId: 'output' },
410
+ ],
411
+ };
412
+
413
+ const graph = buildGraph(jdm);
414
+
415
+ // Should throw on multiple input nodes
416
+ expect(() => validateGraph(graph)).toThrow('Expected exactly 1 input node');
417
+ try {
418
+ validateGraph(graph);
419
+ } catch (error) {
420
+ const message = (error as Error).message;
421
+ expect(message).toContain('Expected exactly 1 input node');
422
+ expect(message).toContain('found 2');
423
+ }
424
+ });
425
+ });
426
+
427
+ describe('Performance Considerations', () => {
428
+ it('should build 50-node graph', () => {
429
+ const nodeCount = 50;
430
+ const nodes: Array<{ id: string }> = [];
431
+ for (let i = 0; i < nodeCount; i++) {
432
+ nodes.push({ id: `node${i}` });
433
+ }
434
+
435
+ const edges: Array<{ id: string; sourceId: string; targetId: string }> = [
436
+ { id: 'e-in', sourceId: 'input', targetId: 'node0' },
437
+ ];
438
+
439
+ // Create a chain with some branching
440
+ for (let i = 0; i < nodeCount - 1; i++) {
441
+ edges.push({
442
+ id: `e-${i}-${i + 1}`,
443
+ sourceId: `node${i}`,
444
+ targetId: `node${i + 1}`,
445
+ });
446
+
447
+ // Add some cross-links for complexity
448
+ if (i > 0 && i % 5 === 0) {
449
+ edges.push({
450
+ id: `e-cross-${i}`,
451
+ sourceId: `node${i - 2}`,
452
+ targetId: `node${i + 1}`,
453
+ });
454
+ }
455
+ }
456
+
457
+ edges.push({ id: 'e-out', sourceId: `node${nodeCount - 1}`, targetId: 'output' });
458
+
459
+ const jdm = createTestGraph(nodes, edges);
460
+
461
+ const graph = buildGraph(jdm);
462
+ validateGraph(graph);
463
+
464
+ // Verify the graph was built correctly
465
+ const order = getEvaluationOrder(graph);
466
+ expect(order.length).toBeGreaterThan(nodeCount);
467
+ expect(order).toContain('input');
468
+ expect(order).toContain('output');
469
+ expect(order).toContain('node0');
470
+ expect(order).toContain(`node${nodeCount - 1}`);
471
+ });
472
+
473
+ it('should handle dense graph (many edges between many nodes)', () => {
474
+ const nodeCount = 30;
475
+ const nodes: Array<{ id: string }> = [];
476
+ for (let i = 0; i < nodeCount; i++) {
477
+ nodes.push({ id: `node${i}` });
478
+ }
479
+
480
+ const edges: Array<{ id: string; sourceId: string; targetId: string }> = [
481
+ { id: 'e-in', sourceId: 'input', targetId: 'node0' },
482
+ ];
483
+
484
+ // Create a complete graph where each node connects to all higher-numbered nodes
485
+ let edgeId = 1;
486
+ for (let i = 0; i < nodeCount; i++) {
487
+ for (let j = i + 1; j < Math.min(i + 5, nodeCount); j++) {
488
+ edges.push({
489
+ id: `e-${edgeId++}`,
490
+ sourceId: `node${i}`,
491
+ targetId: `node${j}`,
492
+ });
493
+ }
494
+ }
495
+
496
+ edges.push({ id: `e-${edgeId}`, sourceId: `node${nodeCount - 1}`, targetId: 'output' });
497
+
498
+ const jdm = createTestGraph(nodes, edges);
499
+
500
+ const graph = buildGraph(jdm);
501
+ validateGraph(graph);
502
+ const order = getEvaluationOrder(graph);
503
+
504
+ // All nodes should be in the order
505
+ for (const node of nodes) {
506
+ expect(order).toContain(node.id);
507
+ }
508
+ });
509
+ });
510
+ });