@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,447 @@
1
+ /**
2
+ * Tests for input validation functionality.
3
+ *
4
+ * This file tests both compile-time and runtime validation:
5
+ * - compile-time-validation.ts: collectIdentifiers, validateInputReferences
6
+ * - runtime-validation-codegen.ts: validateRuntimeInputs, validateRuntimeInputsDetailed
7
+ */
8
+
9
+ import { describe, it, expect } from 'vitest';
10
+ import {
11
+ collectIdentifiers,
12
+ validateInputReferences,
13
+ } from '../../../src/compiler/compile-time-validation';
14
+ import {
15
+ validateRuntimeInputs,
16
+ validateRuntimeInputsDetailed,
17
+ validateRuntimeOutputsDetailed,
18
+ } from '../../../src/compiler/runtime-validation-codegen';
19
+ import type { CompilationContext } from '../../../src/compiler/types';
20
+ import type { ExpressionAST } from '../../../src/compiler/ast-types';
21
+ import { CompilationError } from '../../../src/compiler/errors';
22
+ import { createMockContext as createBaseMockContext } from '../../helpers';
23
+
24
+ describe('validateRuntimeInputs', () => {
25
+ it('should return empty array when all required inputs are present', () => {
26
+ const inputLayout = {
27
+ fields: [
28
+ { path: 'age', flatName: 'age', type: 'f64' as any, offset: 0, size: 8 },
29
+ { path: 'name', flatName: 'name', type: 'string' as any, offset: 8, size: -1 },
30
+ ],
31
+ };
32
+
33
+ const inputData = { age: 25, name: 'John' };
34
+ const missing = validateRuntimeInputs(inputData, inputLayout);
35
+
36
+ expect(missing).toEqual([]);
37
+ });
38
+
39
+ it('should detect missing top-level fields', () => {
40
+ const inputLayout = {
41
+ fields: [
42
+ { path: 'age', flatName: 'age', type: 'f64' as any, offset: 0, size: 8 },
43
+ { path: 'name', flatName: 'name', type: 'string' as any, offset: 8, size: -1 },
44
+ ],
45
+ };
46
+
47
+ const inputData = { age: 25 };
48
+ const missing = validateRuntimeInputs(inputData, inputLayout);
49
+
50
+ expect(missing).toEqual(['name']);
51
+ });
52
+
53
+ it('should detect missing nested fields', () => {
54
+ const inputLayout = {
55
+ fields: [
56
+ { path: 'customer.age', flatName: 'customer_age', type: 'f64' as any, offset: 0, size: 8 },
57
+ {
58
+ path: 'customer.name',
59
+ flatName: 'customer_name',
60
+ type: 'string' as any,
61
+ offset: 8,
62
+ size: -1,
63
+ },
64
+ ],
65
+ };
66
+
67
+ const inputData = { customer: { age: 25 } };
68
+ const missing = validateRuntimeInputs(inputData, inputLayout);
69
+
70
+ expect(missing).toEqual(['customer.name']);
71
+ });
72
+
73
+ it('should detect all missing fields', () => {
74
+ const inputLayout = {
75
+ fields: [
76
+ { path: 'age', flatName: 'age', type: 'f64' as any, offset: 0, size: 8 },
77
+ { path: 'name', flatName: 'name', type: 'string' as any, offset: 8, size: -1 },
78
+ { path: 'email', flatName: 'email', type: 'string' as any, offset: 8, size: -1 },
79
+ ],
80
+ };
81
+
82
+ const inputData = {};
83
+ const missing = validateRuntimeInputs(inputData, inputLayout);
84
+
85
+ expect(missing.sort()).toEqual(['age', 'email', 'name']);
86
+ });
87
+
88
+ it('should detect type mismatch - string instead of number', () => {
89
+ const inputLayout = {
90
+ fields: [{ path: 'age', flatName: 'age', type: 'f64' as any, offset: 0, size: 8 }],
91
+ };
92
+
93
+ const inputData = { age: '25' }; // String instead of number
94
+ const errors = validateRuntimeInputs(inputData, inputLayout);
95
+
96
+ expect(errors).toEqual(['age']);
97
+ });
98
+
99
+ it('should detect type mismatch - number instead of string', () => {
100
+ const inputLayout = {
101
+ fields: [{ path: 'name', flatName: 'name', type: 'string' as any, offset: 0, size: -1 }],
102
+ };
103
+
104
+ const inputData = { name: 123 }; // Number instead of string
105
+ const errors = validateRuntimeInputs(inputData, inputLayout);
106
+
107
+ expect(errors).toEqual(['name']);
108
+ });
109
+
110
+ it('should detect type mismatch - object instead of array', () => {
111
+ const inputLayout = {
112
+ fields: [{ path: 'items', flatName: 'items', type: 'array' as any, offset: 0, size: -1 }],
113
+ };
114
+
115
+ const inputData = { items: { a: 1 } }; // Object instead of array
116
+ const errors = validateRuntimeInputs(inputData, inputLayout);
117
+
118
+ expect(errors).toEqual(['items']);
119
+ });
120
+ });
121
+
122
+ describe('validateRuntimeInputsDetailed', () => {
123
+ it('should return valid result for correct input types', () => {
124
+ const inputLayout = {
125
+ fields: [
126
+ { path: 'age', flatName: 'age', type: 'f64' as any, offset: 0, size: 8 },
127
+ { path: 'name', flatName: 'name', type: 'string' as any, offset: 8, size: -1 },
128
+ { path: 'active', flatName: 'active', type: 'bool' as any, offset: 16, size: 4 },
129
+ { path: 'items', flatName: 'items', type: 'array' as any, offset: 20, size: -1 },
130
+ ],
131
+ };
132
+
133
+ const inputData = { age: 25, name: 'John', active: true, items: [1, 2, 3] };
134
+ const result = validateRuntimeInputsDetailed(inputData, inputLayout);
135
+
136
+ expect(result.valid).toBe(true);
137
+ expect(result.errors).toEqual([]);
138
+ });
139
+
140
+ it('should detect string instead of number with detailed error', () => {
141
+ const inputLayout = {
142
+ fields: [{ path: 'age', flatName: 'age', type: 'f64' as any, offset: 0, size: 8 }],
143
+ };
144
+
145
+ const inputData = { age: 'not a number' };
146
+ const result = validateRuntimeInputsDetailed(inputData, inputLayout);
147
+
148
+ expect(result.valid).toBe(false);
149
+ expect(result.errors).toHaveLength(1);
150
+ expect(result.errors[0].path).toBe('age');
151
+ expect(result.errors[0].message).toContain('Expected f64');
152
+ expect(result.errors[0].expected).toBe('f64');
153
+ expect(result.errors[0].actual).toBe('not a number');
154
+ });
155
+
156
+ it('should detect NaN as invalid number', () => {
157
+ const inputLayout = {
158
+ fields: [{ path: 'value', flatName: 'value', type: 'f64' as any, offset: 0, size: 8 }],
159
+ };
160
+
161
+ const inputData = { value: NaN };
162
+ const result = validateRuntimeInputsDetailed(inputData, inputLayout);
163
+
164
+ expect(result.valid).toBe(false);
165
+ expect(result.errors).toHaveLength(1);
166
+ expect(result.errors[0].path).toBe('value');
167
+ expect(result.errors[0].message).toContain('NaN');
168
+ });
169
+
170
+ it('should detect float instead of integer for i32', () => {
171
+ const inputLayout = {
172
+ fields: [{ path: 'count', flatName: 'count', type: 'i32' as any, offset: 0, size: 4 }],
173
+ };
174
+
175
+ const inputData = { count: 3.14 }; // Float instead of integer
176
+ const result = validateRuntimeInputsDetailed(inputData, inputLayout);
177
+
178
+ expect(result.valid).toBe(false);
179
+ expect(result.errors).toHaveLength(1);
180
+ expect(result.errors[0].path).toBe('count');
181
+ expect(result.errors[0].message).toContain('integer');
182
+ expect(result.errors[0].message).toContain('floating-point');
183
+ });
184
+
185
+ it('should accept integer for i64', () => {
186
+ const inputLayout = {
187
+ fields: [
188
+ { path: 'bigNumber', flatName: 'bigNumber', type: 'i64' as any, offset: 0, size: 8 },
189
+ ],
190
+ };
191
+
192
+ const inputData = { bigNumber: 42 };
193
+ const result = validateRuntimeInputsDetailed(inputData, inputLayout);
194
+
195
+ expect(result.valid).toBe(true);
196
+ expect(result.errors).toEqual([]);
197
+ });
198
+
199
+ it('should accept bigint for i64', () => {
200
+ const inputLayout = {
201
+ fields: [
202
+ { path: 'bigNumber', flatName: 'bigNumber', type: 'i64' as any, offset: 0, size: 8 },
203
+ ],
204
+ };
205
+
206
+ const inputData = { bigNumber: 9007199254740993n };
207
+ const result = validateRuntimeInputsDetailed(inputData, inputLayout);
208
+
209
+ expect(result.valid).toBe(true);
210
+ expect(result.errors).toEqual([]);
211
+ });
212
+
213
+ it('should detect string instead of boolean', () => {
214
+ const inputLayout = {
215
+ fields: [{ path: 'active', flatName: 'active', type: 'bool' as any, offset: 0, size: 4 }],
216
+ };
217
+
218
+ const inputData = { active: 'true' }; // String instead of boolean
219
+ const result = validateRuntimeInputsDetailed(inputData, inputLayout);
220
+
221
+ expect(result.valid).toBe(false);
222
+ expect(result.errors).toHaveLength(1);
223
+ expect(result.errors[0].path).toBe('active');
224
+ expect(result.errors[0].message).toContain('Expected boolean');
225
+ expect(result.errors[0].expected).toBe('boolean');
226
+ });
227
+
228
+ it('should detect number instead of string', () => {
229
+ const inputLayout = {
230
+ fields: [{ path: 'name', flatName: 'name', type: 'string' as any, offset: 0, size: -1 }],
231
+ };
232
+
233
+ const inputData = { name: 123 };
234
+ const result = validateRuntimeInputsDetailed(inputData, inputLayout);
235
+
236
+ expect(result.valid).toBe(false);
237
+ expect(result.errors).toHaveLength(1);
238
+ expect(result.errors[0].path).toBe('name');
239
+ expect(result.errors[0].message).toContain('Expected string');
240
+ expect(result.errors[0].expected).toBe('string');
241
+ });
242
+
243
+ it('should detect object instead of array', () => {
244
+ const inputLayout = {
245
+ fields: [{ path: 'items', flatName: 'items', type: 'array' as any, offset: 0, size: -1 }],
246
+ };
247
+
248
+ const inputData = { items: { length: 3 } }; // Object instead of array
249
+ const result = validateRuntimeInputsDetailed(inputData, inputLayout);
250
+
251
+ expect(result.valid).toBe(false);
252
+ expect(result.errors).toHaveLength(1);
253
+ expect(result.errors[0].path).toBe('items');
254
+ expect(result.errors[0].message).toContain('Expected array');
255
+ });
256
+
257
+ it('should detect nested field type mismatches', () => {
258
+ const inputLayout = {
259
+ fields: [
260
+ { path: 'customer.age', flatName: 'customer_age', type: 'f64' as any, offset: 0, size: 8 },
261
+ {
262
+ path: 'customer.name',
263
+ flatName: 'customer_name',
264
+ type: 'string' as any,
265
+ offset: 8,
266
+ size: -1,
267
+ },
268
+ ],
269
+ };
270
+
271
+ const inputData = { customer: { age: 'thirty', name: 456 } };
272
+ const result = validateRuntimeInputsDetailed(inputData, inputLayout);
273
+
274
+ expect(result.valid).toBe(false);
275
+ expect(result.errors).toHaveLength(2);
276
+
277
+ const paths = result.errors.map((e) => e.path).sort();
278
+ expect(paths).toEqual(['customer.age', 'customer.name']);
279
+ });
280
+
281
+ it('should report both missing and type errors', () => {
282
+ const inputLayout = {
283
+ fields: [
284
+ { path: 'age', flatName: 'age', type: 'f64' as any, offset: 0, size: 8 },
285
+ { path: 'name', flatName: 'name', type: 'string' as any, offset: 8, size: -1 },
286
+ { path: 'email', flatName: 'email', type: 'string' as any, offset: 16, size: -1 },
287
+ ],
288
+ };
289
+
290
+ const inputData = { age: 'not a number' }; // Wrong type for age, missing name and email
291
+ const result = validateRuntimeInputsDetailed(inputData, inputLayout);
292
+
293
+ expect(result.valid).toBe(false);
294
+ expect(result.errors).toHaveLength(3);
295
+
296
+ // Find each error by path
297
+ const ageError = result.errors.find((e) => e.path === 'age');
298
+ const nameError = result.errors.find((e) => e.path === 'name');
299
+ const emailError = result.errors.find((e) => e.path === 'email');
300
+
301
+ expect(ageError?.message).toContain('Expected f64');
302
+ expect(nameError?.message).toContain('missing');
303
+ expect(emailError?.message).toContain('missing');
304
+ });
305
+ });
306
+
307
+ describe('validateRuntimeOutputsDetailed', () => {
308
+ it('should validate output types correctly', () => {
309
+ const outputLayout = {
310
+ fields: [
311
+ { path: 'result', flatName: 'result', type: 'f64' as any, offset: 0, size: 8 },
312
+ { path: 'message', flatName: 'message', type: 'string' as any, offset: 8, size: -1 },
313
+ ],
314
+ };
315
+
316
+ const outputData = { result: 42, message: 'Success' };
317
+ const result = validateRuntimeOutputsDetailed(outputData, outputLayout);
318
+
319
+ expect(result.valid).toBe(true);
320
+ expect(result.errors).toEqual([]);
321
+ });
322
+
323
+ it('should detect type mismatches in output', () => {
324
+ const outputLayout = {
325
+ fields: [{ path: 'result', flatName: 'result', type: 'f64' as any, offset: 0, size: 8 }],
326
+ };
327
+
328
+ const outputData = { result: 'not a number' };
329
+ const result = validateRuntimeOutputsDetailed(outputData, outputLayout);
330
+
331
+ expect(result.valid).toBe(false);
332
+ expect(result.errors).toHaveLength(1);
333
+ expect(result.errors[0].path).toBe('result');
334
+ });
335
+ });
336
+
337
+ function createMockContext(fields: string[]): CompilationContext {
338
+ return createBaseMockContext({
339
+ inputFields: fields,
340
+ parseExpression: (expr: string) => {
341
+ // Mock parser - just return a simple identifier
342
+ return { type: 'Identifier', name: expr };
343
+ },
344
+ parseUnaryExpression: (expr: string) => {
345
+ return { type: 'Identifier', name: expr };
346
+ },
347
+ getUniqueId: (prefix: string) => prefix,
348
+ loadDecision: () => {
349
+ throw new Error('Not implemented');
350
+ },
351
+ });
352
+ }
353
+
354
+ function createIdentifier(name: string): ExpressionAST {
355
+ return { type: 'Identifier', name };
356
+ }
357
+
358
+ function createBinaryOp(left: ExpressionAST, op: string, right: ExpressionAST): ExpressionAST {
359
+ return { type: 'BinaryOp', op: op as any, left, right };
360
+ }
361
+
362
+ function createMemberAccess(object: ExpressionAST, property: string): ExpressionAST {
363
+ return { type: 'MemberAccess', object, property };
364
+ }
365
+
366
+ describe('collectIdentifiers', () => {
367
+ it('should collect simple identifier', () => {
368
+ const ast = createIdentifier('age');
369
+ const identifiers = collectIdentifiers(ast);
370
+ expect(identifiers).toEqual(new Set(['age']));
371
+ });
372
+
373
+ it('should collect identifiers from binary expression', () => {
374
+ const ast = createBinaryOp(createIdentifier('age'), '+', createIdentifier('years'));
375
+ const identifiers = collectIdentifiers(ast);
376
+ expect(identifiers).toEqual(new Set(['age', 'years']));
377
+ });
378
+
379
+ it('should collect identifier from member access base', () => {
380
+ const ast = createMemberAccess(createIdentifier('customer'), 'age');
381
+ const identifiers = collectIdentifiers(ast);
382
+ expect(identifiers).toEqual(new Set(['customer']));
383
+ });
384
+
385
+ it('should collect all identifiers from nested expression', () => {
386
+ const ast = createBinaryOp(
387
+ createMemberAccess(createIdentifier('customer'), 'age'),
388
+ '>',
389
+ createBinaryOp(createIdentifier('minAge'), '+', createIdentifier('offset')),
390
+ );
391
+ const identifiers = collectIdentifiers(ast);
392
+ expect(identifiers).toEqual(new Set(['customer', 'minAge', 'offset']));
393
+ });
394
+ });
395
+
396
+ describe('validateInputReferences', () => {
397
+ it('should pass when all identifiers are declared', () => {
398
+ const context = createMockContext(['age', 'name', 'customer.age']);
399
+ const identifiers = new Set(['age', 'name']);
400
+ expect(() => validateInputReferences(identifiers, context)).not.toThrow();
401
+ });
402
+
403
+ it('should pass when identifier references object with nested fields', () => {
404
+ const context = createMockContext(['customer', 'customer.age', 'customer.name']);
405
+ const identifiers = new Set(['customer']);
406
+ expect(() => validateInputReferences(identifiers, context)).not.toThrow();
407
+ });
408
+
409
+ it('should pass when identifier references specific nested field', () => {
410
+ const context = createMockContext(['customer', 'customer.age', 'customer.name']);
411
+ const identifiers = new Set(['customer.age']);
412
+ expect(() => validateInputReferences(identifiers, context)).not.toThrow();
413
+ });
414
+
415
+ it('should throw error for single missing identifier', () => {
416
+ const context = createMockContext(['age', 'name']);
417
+ const identifiers = new Set(['age', 'undeclared']);
418
+ expect(() => validateInputReferences(identifiers, context)).toThrow(CompilationError);
419
+ try {
420
+ validateInputReferences(identifiers, context);
421
+ expect.fail('Should have thrown an error');
422
+ } catch (error) {
423
+ expect((error as CompilationError).message).toContain('undeclared');
424
+ expect((error as CompilationError).code).toBe('UNKNOWN_FIELD');
425
+ }
426
+ });
427
+
428
+ it('should throw error listing all missing identifiers', () => {
429
+ const context = createMockContext(['age', 'name']);
430
+ const identifiers = new Set(['undeclared1', 'age', 'undeclared2', 'name']);
431
+ expect(() => validateInputReferences(identifiers, context)).toThrow(CompilationError);
432
+ try {
433
+ validateInputReferences(identifiers, context);
434
+ expect.fail('Should have thrown an error');
435
+ } catch (error) {
436
+ const message = (error as CompilationError).message;
437
+ expect(message).toContain('undeclared1');
438
+ expect(message).toContain('undeclared2');
439
+ }
440
+ });
441
+
442
+ it('should skip special identifiers $ and #', () => {
443
+ const context = createMockContext(['age', 'name']);
444
+ const identifiers = new Set(['$', '#', 'age', 'name']);
445
+ expect(() => validateInputReferences(identifiers, context)).not.toThrow();
446
+ });
447
+ });
@@ -0,0 +1,143 @@
1
+ import { describe, it, expect, beforeEach } from 'vitest';
2
+ import { parser, lexer } from '../../../src/compiler/expression-parser';
3
+ import { cstToAst } from '../../../src/compiler/cst-visitor';
4
+
5
+ describe('Expression Parser - Logical AND Operator', () => {
6
+ beforeEach(() => {
7
+ // Reset parser state before each test
8
+ parser.reset();
9
+ });
10
+
11
+ it('should parse "true" as just a boolean literal', () => {
12
+ const lexingResult = lexer.tokenize('true');
13
+
14
+ if (lexingResult.errors.length > 0) {
15
+ throw new Error(`Lexer error: ${lexingResult.errors.map((e: any) => e.message).join(', ')}`);
16
+ }
17
+
18
+ parser.input = lexingResult.tokens;
19
+ const cst = parser.expression();
20
+
21
+ if (parser.errors.length > 0) {
22
+ throw new Error(`Parse error: ${parser.errors.map((e: any) => e.message).join(', ')}`);
23
+ }
24
+
25
+ const ast = cstToAst(cst);
26
+ expect(ast).toEqual({
27
+ type: 'BooleanLiteral',
28
+ value: true,
29
+ });
30
+ });
31
+
32
+ it('should parse "true and true"', () => {
33
+ const lexingResult = lexer.tokenize('true and true');
34
+
35
+ if (lexingResult.errors.length > 0) {
36
+ throw new Error(`Lexer error: ${lexingResult.errors.map((e: any) => e.message).join(', ')}`);
37
+ }
38
+
39
+ parser.input = lexingResult.tokens;
40
+ const cst = parser.expression();
41
+
42
+ if (parser.errors.length > 0) {
43
+ throw new Error(`Parse error: ${parser.errors.map((e: any) => e.message).join(', ')}`);
44
+ }
45
+
46
+ const ast = cstToAst(cst);
47
+ expect(ast).toEqual({
48
+ type: 'BinaryOp',
49
+ op: 'and',
50
+ left: { type: 'BooleanLiteral', value: true },
51
+ right: { type: 'BooleanLiteral', value: true },
52
+ });
53
+ });
54
+
55
+ it('should parse "true && true"', () => {
56
+ const lexingResult = lexer.tokenize('true && true');
57
+
58
+ if (lexingResult.errors.length > 0) {
59
+ throw new Error(`Lexer error: ${lexingResult.errors.map((e: any) => e.message).join(', ')}`);
60
+ }
61
+
62
+ parser.input = lexingResult.tokens;
63
+ const cst = parser.expression();
64
+
65
+ if (parser.errors.length > 0) {
66
+ throw new Error(`Parse error: ${parser.errors.map((e: any) => e.message).join(', ')}`);
67
+ }
68
+
69
+ const ast = cstToAst(cst);
70
+ expect(ast).toEqual({
71
+ type: 'BinaryOp',
72
+ op: 'and',
73
+ left: { type: 'BooleanLiteral', value: true },
74
+ right: { type: 'BooleanLiteral', value: true },
75
+ });
76
+ });
77
+
78
+ it('should parse "x > 5 and x < 10"', () => {
79
+ const lexingResult = lexer.tokenize('x > 5 and x < 10');
80
+
81
+ if (lexingResult.errors.length > 0) {
82
+ throw new Error(`Lexer error: ${lexingResult.errors.map((e: any) => e.message).join(', ')}`);
83
+ }
84
+
85
+ parser.input = lexingResult.tokens;
86
+ const cst = parser.expression();
87
+
88
+ if (parser.errors.length > 0) {
89
+ throw new Error(`Parse error: ${parser.errors.map((e: any) => e.message).join(', ')}`);
90
+ }
91
+
92
+ const ast = cstToAst(cst);
93
+ expect(ast).toEqual({
94
+ type: 'BinaryOp',
95
+ op: 'and',
96
+ left: {
97
+ type: 'BinaryOp',
98
+ op: '>',
99
+ left: { type: 'Identifier', name: 'x' },
100
+ right: { type: 'NumberLiteral', value: 5 },
101
+ },
102
+ right: {
103
+ type: 'BinaryOp',
104
+ op: '<',
105
+ left: { type: 'Identifier', name: 'x' },
106
+ right: { type: 'NumberLiteral', value: 10 },
107
+ },
108
+ });
109
+ });
110
+
111
+ it('should parse "x > 5 && x < 10"', () => {
112
+ const lexingResult = lexer.tokenize('x > 5 && x < 10');
113
+
114
+ if (lexingResult.errors.length > 0) {
115
+ throw new Error(`Lexer error: ${lexingResult.errors.map((e: any) => e.message).join(', ')}`);
116
+ }
117
+
118
+ parser.input = lexingResult.tokens;
119
+ const cst = parser.expression();
120
+
121
+ if (parser.errors.length > 0) {
122
+ throw new Error(`Parse error: ${parser.errors.map((e: any) => e.message).join(', ')}`);
123
+ }
124
+
125
+ const ast = cstToAst(cst);
126
+ expect(ast).toEqual({
127
+ type: 'BinaryOp',
128
+ op: 'and',
129
+ left: {
130
+ type: 'BinaryOp',
131
+ op: '>',
132
+ left: { type: 'Identifier', name: 'x' },
133
+ right: { type: 'NumberLiteral', value: 5 },
134
+ },
135
+ right: {
136
+ type: 'BinaryOp',
137
+ op: '<',
138
+ left: { type: 'Identifier', name: 'x' },
139
+ right: { type: 'NumberLiteral', value: 10 },
140
+ },
141
+ });
142
+ });
143
+ });
@@ -0,0 +1,107 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { parseStandardExpression } from '../../../src/compiler/unary-parser';
3
+
4
+ describe('Logical NOT Expression Parser', () => {
5
+ describe('parseStandardExpression', () => {
6
+ it('should parse "not true" expression', () => {
7
+ const ast = parseStandardExpression('not true');
8
+ expect(ast.type).toBe('UnaryOp');
9
+ if (ast.type === 'UnaryOp') {
10
+ expect(ast.op).toBe('not');
11
+ expect(ast.operand).toEqual({ type: 'BooleanLiteral', value: true });
12
+ }
13
+ });
14
+
15
+ it('should parse "not false" expression', () => {
16
+ const ast = parseStandardExpression('not false');
17
+ expect(ast.type).toBe('UnaryOp');
18
+ if (ast.type === 'UnaryOp') {
19
+ expect(ast.op).toBe('not');
20
+ expect(ast.operand).toEqual({ type: 'BooleanLiteral', value: false });
21
+ }
22
+ });
23
+
24
+ it('should parse "!true" expression', () => {
25
+ const ast = parseStandardExpression('!true');
26
+ expect(ast.type).toBe('UnaryOp');
27
+ if (ast.type === 'UnaryOp') {
28
+ expect(ast.op).toBe('not');
29
+ expect(ast.operand).toEqual({ type: 'BooleanLiteral', value: true });
30
+ }
31
+ });
32
+
33
+ it('should parse "!false" expression', () => {
34
+ const ast = parseStandardExpression('!false');
35
+ expect(ast.type).toBe('UnaryOp');
36
+ if (ast.type === 'UnaryOp') {
37
+ expect(ast.op).toBe('not');
38
+ expect(ast.operand).toEqual({ type: 'BooleanLiteral', value: false });
39
+ }
40
+ });
41
+
42
+ it('should parse "not (x > 5)" expression with comparison', () => {
43
+ const ast = parseStandardExpression('not (x > 5)');
44
+ expect(ast.type).toBe('UnaryOp');
45
+ if (ast.type === 'UnaryOp') {
46
+ expect(ast.op).toBe('not');
47
+ expect(ast.operand.type).toBe('BinaryOp');
48
+ }
49
+ });
50
+
51
+ it('should parse "!(x > 5)" expression with comparison', () => {
52
+ const ast = parseStandardExpression('!(x > 5)');
53
+ expect(ast.type).toBe('UnaryOp');
54
+ if (ast.type === 'UnaryOp') {
55
+ expect(ast.op).toBe('not');
56
+ expect(ast.operand.type).toBe('BinaryOp');
57
+ }
58
+ });
59
+
60
+ it('should parse double NOT "not not true"', () => {
61
+ const ast = parseStandardExpression('not not true');
62
+ expect(ast.type).toBe('UnaryOp');
63
+ if (ast.type === 'UnaryOp') {
64
+ expect(ast.op).toBe('not');
65
+ expect(ast.operand.type).toBe('UnaryOp');
66
+ if (ast.operand.type === 'UnaryOp') {
67
+ expect(ast.operand.op).toBe('not');
68
+ }
69
+ }
70
+ });
71
+
72
+ it('should parse NOT with AND "not a and b"', () => {
73
+ const ast = parseStandardExpression('not a and b');
74
+ // NOT should bind tighter than AND, so this should be (not a) and b
75
+ expect(ast.type).toBe('BinaryOp');
76
+ if (ast.type === 'BinaryOp') {
77
+ expect(ast.op).toBe('and');
78
+ expect(ast.left.type).toBe('UnaryOp');
79
+ if (ast.left.type === 'UnaryOp') {
80
+ expect(ast.left.op).toBe('not');
81
+ }
82
+ }
83
+ });
84
+
85
+ it('should parse NOT with parentheses "not (a or b)"', () => {
86
+ const ast = parseStandardExpression('not (a or b)');
87
+ expect(ast.type).toBe('UnaryOp');
88
+ if (ast.type === 'UnaryOp') {
89
+ expect(ast.op).toBe('not');
90
+ expect(ast.operand.type).toBe('BinaryOp');
91
+ if (ast.operand.type === 'BinaryOp') {
92
+ expect(ast.operand.op).toBe('or');
93
+ }
94
+ }
95
+ });
96
+
97
+ it('should handle type coercion with non-boolean operands', () => {
98
+ // In standard parsing mode, NOT should accept any expression
99
+ // and coerce to boolean
100
+ const ast = parseStandardExpression('not 0');
101
+ expect(ast.type).toBe('UnaryOp');
102
+ if (ast.type === 'UnaryOp') {
103
+ expect(ast.op).toBe('not');
104
+ }
105
+ });
106
+ });
107
+ });