@substrate-ai/factory 0.19.54

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 (885) hide show
  1. package/dist/__tests__/config.test.d.ts +11 -0
  2. package/dist/__tests__/config.test.d.ts.map +1 -0
  3. package/dist/__tests__/config.test.js +215 -0
  4. package/dist/__tests__/config.test.js.map +1 -0
  5. package/dist/__tests__/factory-run-command.test.d.ts +12 -0
  6. package/dist/__tests__/factory-run-command.test.d.ts.map +1 -0
  7. package/dist/__tests__/factory-run-command.test.js +454 -0
  8. package/dist/__tests__/factory-run-command.test.js.map +1 -0
  9. package/dist/__tests__/factory-validate-command.test.d.ts +15 -0
  10. package/dist/__tests__/factory-validate-command.test.d.ts.map +1 -0
  11. package/dist/__tests__/factory-validate-command.test.js +339 -0
  12. package/dist/__tests__/factory-validate-command.test.js.map +1 -0
  13. package/dist/__tests__/fixtures/advanced-cross-project-validation.dot.d.ts +72 -0
  14. package/dist/__tests__/fixtures/advanced-cross-project-validation.dot.d.ts.map +1 -0
  15. package/dist/__tests__/fixtures/advanced-cross-project-validation.dot.js +121 -0
  16. package/dist/__tests__/fixtures/advanced-cross-project-validation.dot.js.map +1 -0
  17. package/dist/__tests__/fixtures/llm-edge-routing.dot.d.ts +28 -0
  18. package/dist/__tests__/fixtures/llm-edge-routing.dot.d.ts.map +1 -0
  19. package/dist/__tests__/fixtures/llm-edge-routing.dot.js +55 -0
  20. package/dist/__tests__/fixtures/llm-edge-routing.dot.js.map +1 -0
  21. package/dist/__tests__/fixtures/manager-loop.dot.d.ts +34 -0
  22. package/dist/__tests__/fixtures/manager-loop.dot.d.ts.map +1 -0
  23. package/dist/__tests__/fixtures/manager-loop.dot.js +61 -0
  24. package/dist/__tests__/fixtures/manager-loop.dot.js.map +1 -0
  25. package/dist/__tests__/fixtures/parallel-fan-out-fan-in.dot.d.ts +42 -0
  26. package/dist/__tests__/fixtures/parallel-fan-out-fan-in.dot.d.ts.map +1 -0
  27. package/dist/__tests__/fixtures/parallel-fan-out-fan-in.dot.js +118 -0
  28. package/dist/__tests__/fixtures/parallel-fan-out-fan-in.dot.js.map +1 -0
  29. package/dist/__tests__/fixtures/subgraph-parent.dot.d.ts +35 -0
  30. package/dist/__tests__/fixtures/subgraph-parent.dot.d.ts.map +1 -0
  31. package/dist/__tests__/fixtures/subgraph-parent.dot.js +69 -0
  32. package/dist/__tests__/fixtures/subgraph-parent.dot.js.map +1 -0
  33. package/dist/__tests__/integration/advanced-graph-events.test.d.ts +19 -0
  34. package/dist/__tests__/integration/advanced-graph-events.test.d.ts.map +1 -0
  35. package/dist/__tests__/integration/advanced-graph-events.test.js +288 -0
  36. package/dist/__tests__/integration/advanced-graph-events.test.js.map +1 -0
  37. package/dist/__tests__/integration/checkpoint-resume.test.d.ts +10 -0
  38. package/dist/__tests__/integration/checkpoint-resume.test.d.ts.map +1 -0
  39. package/dist/__tests__/integration/checkpoint-resume.test.js +125 -0
  40. package/dist/__tests__/integration/checkpoint-resume.test.js.map +1 -0
  41. package/dist/__tests__/integration/conditional-pipeline.test.d.ts +10 -0
  42. package/dist/__tests__/integration/conditional-pipeline.test.d.ts.map +1 -0
  43. package/dist/__tests__/integration/conditional-pipeline.test.js +106 -0
  44. package/dist/__tests__/integration/conditional-pipeline.test.js.map +1 -0
  45. package/dist/__tests__/integration/convergence-validation.test.d.ts +14 -0
  46. package/dist/__tests__/integration/convergence-validation.test.d.ts.map +1 -0
  47. package/dist/__tests__/integration/convergence-validation.test.js +449 -0
  48. package/dist/__tests__/integration/convergence-validation.test.js.map +1 -0
  49. package/dist/__tests__/integration/epic44-coverage-gate.test.d.ts +12 -0
  50. package/dist/__tests__/integration/epic44-coverage-gate.test.d.ts.map +1 -0
  51. package/dist/__tests__/integration/epic44-coverage-gate.test.js +58 -0
  52. package/dist/__tests__/integration/epic44-coverage-gate.test.js.map +1 -0
  53. package/dist/__tests__/integration/epic45-coverage-gate.test.d.ts +11 -0
  54. package/dist/__tests__/integration/epic45-coverage-gate.test.d.ts.map +1 -0
  55. package/dist/__tests__/integration/epic45-coverage-gate.test.js +64 -0
  56. package/dist/__tests__/integration/epic45-coverage-gate.test.js.map +1 -0
  57. package/dist/__tests__/integration/epic46-scenario-primary-executor.test.d.ts +2 -0
  58. package/dist/__tests__/integration/epic46-scenario-primary-executor.test.d.ts.map +1 -0
  59. package/dist/__tests__/integration/epic46-scenario-primary-executor.test.js +285 -0
  60. package/dist/__tests__/integration/epic46-scenario-primary-executor.test.js.map +1 -0
  61. package/dist/__tests__/integration/events.test.d.ts +8 -0
  62. package/dist/__tests__/integration/events.test.d.ts.map +1 -0
  63. package/dist/__tests__/integration/events.test.js +194 -0
  64. package/dist/__tests__/integration/events.test.js.map +1 -0
  65. package/dist/__tests__/integration/graphs.d.ts +59 -0
  66. package/dist/__tests__/integration/graphs.d.ts.map +1 -0
  67. package/dist/__tests__/integration/graphs.js +164 -0
  68. package/dist/__tests__/integration/graphs.js.map +1 -0
  69. package/dist/__tests__/integration/helpers.d.ts +127 -0
  70. package/dist/__tests__/integration/helpers.d.ts.map +1 -0
  71. package/dist/__tests__/integration/helpers.js +167 -0
  72. package/dist/__tests__/integration/helpers.js.map +1 -0
  73. package/dist/__tests__/integration/integrity.test.d.ts +8 -0
  74. package/dist/__tests__/integration/integrity.test.d.ts.map +1 -0
  75. package/dist/__tests__/integration/integrity.test.js +198 -0
  76. package/dist/__tests__/integration/integrity.test.js.map +1 -0
  77. package/dist/__tests__/integration/llm-edge-routing.test.d.ts +21 -0
  78. package/dist/__tests__/integration/llm-edge-routing.test.d.ts.map +1 -0
  79. package/dist/__tests__/integration/llm-edge-routing.test.js +341 -0
  80. package/dist/__tests__/integration/llm-edge-routing.test.js.map +1 -0
  81. package/dist/__tests__/integration/manager-loop.test.d.ts +24 -0
  82. package/dist/__tests__/integration/manager-loop.test.d.ts.map +1 -0
  83. package/dist/__tests__/integration/manager-loop.test.js +276 -0
  84. package/dist/__tests__/integration/manager-loop.test.js.map +1 -0
  85. package/dist/__tests__/integration/multi-type-graph.test.d.ts +10 -0
  86. package/dist/__tests__/integration/multi-type-graph.test.d.ts.map +1 -0
  87. package/dist/__tests__/integration/multi-type-graph.test.js +100 -0
  88. package/dist/__tests__/integration/multi-type-graph.test.js.map +1 -0
  89. package/dist/__tests__/integration/parallel-fan-out-fan-in.test.d.ts +22 -0
  90. package/dist/__tests__/integration/parallel-fan-out-fan-in.test.d.ts.map +1 -0
  91. package/dist/__tests__/integration/parallel-fan-out-fan-in.test.js +515 -0
  92. package/dist/__tests__/integration/parallel-fan-out-fan-in.test.js.map +1 -0
  93. package/dist/__tests__/integration/persistence.test.d.ts +8 -0
  94. package/dist/__tests__/integration/persistence.test.d.ts.map +1 -0
  95. package/dist/__tests__/integration/persistence.test.js +129 -0
  96. package/dist/__tests__/integration/persistence.test.js.map +1 -0
  97. package/dist/__tests__/integration/pipeline-templates-integration.test.d.ts +16 -0
  98. package/dist/__tests__/integration/pipeline-templates-integration.test.d.ts.map +1 -0
  99. package/dist/__tests__/integration/pipeline-templates-integration.test.js +171 -0
  100. package/dist/__tests__/integration/pipeline-templates-integration.test.js.map +1 -0
  101. package/dist/__tests__/integration/scenario-pipeline.test.d.ts +11 -0
  102. package/dist/__tests__/integration/scenario-pipeline.test.d.ts.map +1 -0
  103. package/dist/__tests__/integration/scenario-pipeline.test.js +243 -0
  104. package/dist/__tests__/integration/scenario-pipeline.test.js.map +1 -0
  105. package/dist/__tests__/integration/stylesheet-application.test.d.ts +12 -0
  106. package/dist/__tests__/integration/stylesheet-application.test.d.ts.map +1 -0
  107. package/dist/__tests__/integration/stylesheet-application.test.js +119 -0
  108. package/dist/__tests__/integration/stylesheet-application.test.js.map +1 -0
  109. package/dist/__tests__/integration/subgraph-execution.test.d.ts +24 -0
  110. package/dist/__tests__/integration/subgraph-execution.test.d.ts.map +1 -0
  111. package/dist/__tests__/integration/subgraph-execution.test.js +291 -0
  112. package/dist/__tests__/integration/subgraph-execution.test.js.map +1 -0
  113. package/dist/__tests__/integration/validation-errors.test.d.ts +8 -0
  114. package/dist/__tests__/integration/validation-errors.test.d.ts.map +1 -0
  115. package/dist/__tests__/integration/validation-errors.test.js +150 -0
  116. package/dist/__tests__/integration/validation-errors.test.js.map +1 -0
  117. package/dist/agent/__tests__/loop-detection.test.d.ts +2 -0
  118. package/dist/agent/__tests__/loop-detection.test.d.ts.map +1 -0
  119. package/dist/agent/__tests__/loop-detection.test.js +236 -0
  120. package/dist/agent/__tests__/loop-detection.test.js.map +1 -0
  121. package/dist/agent/__tests__/loop.test.d.ts +2 -0
  122. package/dist/agent/__tests__/loop.test.d.ts.map +1 -0
  123. package/dist/agent/__tests__/loop.test.js +868 -0
  124. package/dist/agent/__tests__/loop.test.js.map +1 -0
  125. package/dist/agent/__tests__/truncation.test.d.ts +2 -0
  126. package/dist/agent/__tests__/truncation.test.d.ts.map +1 -0
  127. package/dist/agent/__tests__/truncation.test.js +276 -0
  128. package/dist/agent/__tests__/truncation.test.js.map +1 -0
  129. package/dist/agent/index.d.ts +11 -0
  130. package/dist/agent/index.d.ts.map +1 -0
  131. package/dist/agent/index.js +13 -0
  132. package/dist/agent/index.js.map +1 -0
  133. package/dist/agent/loop-detection.d.ts +21 -0
  134. package/dist/agent/loop-detection.d.ts.map +1 -0
  135. package/dist/agent/loop-detection.js +61 -0
  136. package/dist/agent/loop-detection.js.map +1 -0
  137. package/dist/agent/loop.d.ts +88 -0
  138. package/dist/agent/loop.d.ts.map +1 -0
  139. package/dist/agent/loop.js +411 -0
  140. package/dist/agent/loop.js.map +1 -0
  141. package/dist/agent/tools/__tests__/anthropic-tools.test.d.ts +6 -0
  142. package/dist/agent/tools/__tests__/anthropic-tools.test.d.ts.map +1 -0
  143. package/dist/agent/tools/__tests__/anthropic-tools.test.js +49 -0
  144. package/dist/agent/tools/__tests__/anthropic-tools.test.js.map +1 -0
  145. package/dist/agent/tools/__tests__/environment.test.d.ts +6 -0
  146. package/dist/agent/tools/__tests__/environment.test.d.ts.map +1 -0
  147. package/dist/agent/tools/__tests__/environment.test.js +33 -0
  148. package/dist/agent/tools/__tests__/environment.test.js.map +1 -0
  149. package/dist/agent/tools/__tests__/gemini-tools.test.d.ts +6 -0
  150. package/dist/agent/tools/__tests__/gemini-tools.test.d.ts.map +1 -0
  151. package/dist/agent/tools/__tests__/gemini-tools.test.js +98 -0
  152. package/dist/agent/tools/__tests__/gemini-tools.test.js.map +1 -0
  153. package/dist/agent/tools/__tests__/openai-tools.test.d.ts +6 -0
  154. package/dist/agent/tools/__tests__/openai-tools.test.d.ts.map +1 -0
  155. package/dist/agent/tools/__tests__/openai-tools.test.js +53 -0
  156. package/dist/agent/tools/__tests__/openai-tools.test.js.map +1 -0
  157. package/dist/agent/tools/__tests__/patch.test.d.ts +6 -0
  158. package/dist/agent/tools/__tests__/patch.test.d.ts.map +1 -0
  159. package/dist/agent/tools/__tests__/patch.test.js +116 -0
  160. package/dist/agent/tools/__tests__/patch.test.js.map +1 -0
  161. package/dist/agent/tools/__tests__/profiles.test.d.ts +6 -0
  162. package/dist/agent/tools/__tests__/profiles.test.d.ts.map +1 -0
  163. package/dist/agent/tools/__tests__/profiles.test.js +125 -0
  164. package/dist/agent/tools/__tests__/profiles.test.js.map +1 -0
  165. package/dist/agent/tools/__tests__/registry.test.d.ts +6 -0
  166. package/dist/agent/tools/__tests__/registry.test.d.ts.map +1 -0
  167. package/dist/agent/tools/__tests__/registry.test.js +94 -0
  168. package/dist/agent/tools/__tests__/registry.test.js.map +1 -0
  169. package/dist/agent/tools/__tests__/shared.test.d.ts +6 -0
  170. package/dist/agent/tools/__tests__/shared.test.d.ts.map +1 -0
  171. package/dist/agent/tools/__tests__/shared.test.js +131 -0
  172. package/dist/agent/tools/__tests__/shared.test.js.map +1 -0
  173. package/dist/agent/tools/anthropic-tools.d.ts +15 -0
  174. package/dist/agent/tools/anthropic-tools.d.ts.map +1 -0
  175. package/dist/agent/tools/anthropic-tools.js +45 -0
  176. package/dist/agent/tools/anthropic-tools.js.map +1 -0
  177. package/dist/agent/tools/environment.d.ts +11 -0
  178. package/dist/agent/tools/environment.d.ts.map +1 -0
  179. package/dist/agent/tools/environment.js +35 -0
  180. package/dist/agent/tools/environment.js.map +1 -0
  181. package/dist/agent/tools/gemini-tools.d.ts +29 -0
  182. package/dist/agent/tools/gemini-tools.d.ts.map +1 -0
  183. package/dist/agent/tools/gemini-tools.js +112 -0
  184. package/dist/agent/tools/gemini-tools.js.map +1 -0
  185. package/dist/agent/tools/index.d.ts +13 -0
  186. package/dist/agent/tools/index.d.ts.map +1 -0
  187. package/dist/agent/tools/index.js +13 -0
  188. package/dist/agent/tools/index.js.map +1 -0
  189. package/dist/agent/tools/openai-tools.d.ts +18 -0
  190. package/dist/agent/tools/openai-tools.d.ts.map +1 -0
  191. package/dist/agent/tools/openai-tools.js +208 -0
  192. package/dist/agent/tools/openai-tools.js.map +1 -0
  193. package/dist/agent/tools/profiles.d.ts +66 -0
  194. package/dist/agent/tools/profiles.d.ts.map +1 -0
  195. package/dist/agent/tools/profiles.js +121 -0
  196. package/dist/agent/tools/profiles.js.map +1 -0
  197. package/dist/agent/tools/registry.d.ts +16 -0
  198. package/dist/agent/tools/registry.d.ts.map +1 -0
  199. package/dist/agent/tools/registry.js +55 -0
  200. package/dist/agent/tools/registry.js.map +1 -0
  201. package/dist/agent/tools/shared.d.ts +10 -0
  202. package/dist/agent/tools/shared.d.ts.map +1 -0
  203. package/dist/agent/tools/shared.js +207 -0
  204. package/dist/agent/tools/shared.js.map +1 -0
  205. package/dist/agent/tools/types.d.ts +55 -0
  206. package/dist/agent/tools/types.d.ts.map +1 -0
  207. package/dist/agent/tools/types.js +14 -0
  208. package/dist/agent/tools/types.js.map +1 -0
  209. package/dist/agent/truncation.d.ts +29 -0
  210. package/dist/agent/truncation.d.ts.map +1 -0
  211. package/dist/agent/truncation.js +76 -0
  212. package/dist/agent/truncation.js.map +1 -0
  213. package/dist/agent/types.d.ts +91 -0
  214. package/dist/agent/types.d.ts.map +1 -0
  215. package/dist/agent/types.js +45 -0
  216. package/dist/agent/types.js.map +1 -0
  217. package/dist/backend/__tests__/direct-backend.test.d.ts +14 -0
  218. package/dist/backend/__tests__/direct-backend.test.d.ts.map +1 -0
  219. package/dist/backend/__tests__/direct-backend.test.js +393 -0
  220. package/dist/backend/__tests__/direct-backend.test.js.map +1 -0
  221. package/dist/backend/__tests__/direct-bootstrap.test.d.ts +7 -0
  222. package/dist/backend/__tests__/direct-bootstrap.test.d.ts.map +1 -0
  223. package/dist/backend/__tests__/direct-bootstrap.test.js +177 -0
  224. package/dist/backend/__tests__/direct-bootstrap.test.js.map +1 -0
  225. package/dist/backend/__tests__/mock-backend.test.d.ts +7 -0
  226. package/dist/backend/__tests__/mock-backend.test.d.ts.map +1 -0
  227. package/dist/backend/__tests__/mock-backend.test.js +273 -0
  228. package/dist/backend/__tests__/mock-backend.test.js.map +1 -0
  229. package/dist/backend/__tests__/parity.test.d.ts +17 -0
  230. package/dist/backend/__tests__/parity.test.d.ts.map +1 -0
  231. package/dist/backend/__tests__/parity.test.js +411 -0
  232. package/dist/backend/__tests__/parity.test.js.map +1 -0
  233. package/dist/backend/direct-backend.d.ts +27 -0
  234. package/dist/backend/direct-backend.d.ts.map +1 -0
  235. package/dist/backend/direct-backend.js +76 -0
  236. package/dist/backend/direct-backend.js.map +1 -0
  237. package/dist/backend/direct-bootstrap.d.ts +29 -0
  238. package/dist/backend/direct-bootstrap.d.ts.map +1 -0
  239. package/dist/backend/direct-bootstrap.js +109 -0
  240. package/dist/backend/direct-bootstrap.js.map +1 -0
  241. package/dist/backend/index.d.ts +11 -0
  242. package/dist/backend/index.d.ts.map +1 -0
  243. package/dist/backend/index.js +8 -0
  244. package/dist/backend/index.js.map +1 -0
  245. package/dist/backend/mock-backend.d.ts +49 -0
  246. package/dist/backend/mock-backend.d.ts.map +1 -0
  247. package/dist/backend/mock-backend.js +87 -0
  248. package/dist/backend/mock-backend.js.map +1 -0
  249. package/dist/backend/types.d.ts +74 -0
  250. package/dist/backend/types.d.ts.map +1 -0
  251. package/dist/backend/types.js +12 -0
  252. package/dist/backend/types.js.map +1 -0
  253. package/dist/config.d.ts +196 -0
  254. package/dist/config.d.ts.map +1 -0
  255. package/dist/config.js +140 -0
  256. package/dist/config.js.map +1 -0
  257. package/dist/context/__tests__/auto-summarizer.test.d.ts +14 -0
  258. package/dist/context/__tests__/auto-summarizer.test.d.ts.map +1 -0
  259. package/dist/context/__tests__/auto-summarizer.test.js +189 -0
  260. package/dist/context/__tests__/auto-summarizer.test.js.map +1 -0
  261. package/dist/context/__tests__/context-cli-command.test.d.ts +7 -0
  262. package/dist/context/__tests__/context-cli-command.test.d.ts.map +1 -0
  263. package/dist/context/__tests__/context-cli-command.test.js +331 -0
  264. package/dist/context/__tests__/context-cli-command.test.js.map +1 -0
  265. package/dist/context/__tests__/pyramid-summary-integration.test.d.ts +2 -0
  266. package/dist/context/__tests__/pyramid-summary-integration.test.d.ts.map +1 -0
  267. package/dist/context/__tests__/pyramid-summary-integration.test.js +533 -0
  268. package/dist/context/__tests__/pyramid-summary-integration.test.js.map +1 -0
  269. package/dist/context/__tests__/summarizer.test.d.ts +2 -0
  270. package/dist/context/__tests__/summarizer.test.d.ts.map +1 -0
  271. package/dist/context/__tests__/summarizer.test.js +189 -0
  272. package/dist/context/__tests__/summarizer.test.js.map +1 -0
  273. package/dist/context/__tests__/summary-cache.test.d.ts +2 -0
  274. package/dist/context/__tests__/summary-cache.test.d.ts.map +1 -0
  275. package/dist/context/__tests__/summary-cache.test.js +214 -0
  276. package/dist/context/__tests__/summary-cache.test.js.map +1 -0
  277. package/dist/context/__tests__/summary-metrics.test.d.ts +2 -0
  278. package/dist/context/__tests__/summary-metrics.test.d.ts.map +1 -0
  279. package/dist/context/__tests__/summary-metrics.test.js +172 -0
  280. package/dist/context/__tests__/summary-metrics.test.js.map +1 -0
  281. package/dist/context/__tests__/summary-types.test.d.ts +2 -0
  282. package/dist/context/__tests__/summary-types.test.d.ts.map +1 -0
  283. package/dist/context/__tests__/summary-types.test.js +130 -0
  284. package/dist/context/__tests__/summary-types.test.js.map +1 -0
  285. package/dist/context/auto-summarizer.d.ts +120 -0
  286. package/dist/context/auto-summarizer.d.ts.map +1 -0
  287. package/dist/context/auto-summarizer.js +91 -0
  288. package/dist/context/auto-summarizer.js.map +1 -0
  289. package/dist/context/cli-command.d.ts +109 -0
  290. package/dist/context/cli-command.d.ts.map +1 -0
  291. package/dist/context/cli-command.js +382 -0
  292. package/dist/context/cli-command.js.map +1 -0
  293. package/dist/context/index.d.ts +8 -0
  294. package/dist/context/index.d.ts.map +1 -0
  295. package/dist/context/index.js +8 -0
  296. package/dist/context/index.js.map +1 -0
  297. package/dist/context/summarizer.d.ts +12 -0
  298. package/dist/context/summarizer.d.ts.map +1 -0
  299. package/dist/context/summarizer.js +105 -0
  300. package/dist/context/summarizer.js.map +1 -0
  301. package/dist/context/summary-cache.d.ts +41 -0
  302. package/dist/context/summary-cache.d.ts.map +1 -0
  303. package/dist/context/summary-cache.js +80 -0
  304. package/dist/context/summary-cache.js.map +1 -0
  305. package/dist/context/summary-engine.d.ts +46 -0
  306. package/dist/context/summary-engine.d.ts.map +1 -0
  307. package/dist/context/summary-engine.js +2 -0
  308. package/dist/context/summary-engine.js.map +1 -0
  309. package/dist/context/summary-metrics.d.ts +75 -0
  310. package/dist/context/summary-metrics.d.ts.map +1 -0
  311. package/dist/context/summary-metrics.js +160 -0
  312. package/dist/context/summary-metrics.js.map +1 -0
  313. package/dist/context/summary-types.d.ts +103 -0
  314. package/dist/context/summary-types.d.ts.map +1 -0
  315. package/dist/context/summary-types.js +39 -0
  316. package/dist/context/summary-types.js.map +1 -0
  317. package/dist/convergence/__tests__/budget.test.d.ts +6 -0
  318. package/dist/convergence/__tests__/budget.test.d.ts.map +1 -0
  319. package/dist/convergence/__tests__/budget.test.js +187 -0
  320. package/dist/convergence/__tests__/budget.test.js.map +1 -0
  321. package/dist/convergence/__tests__/controller.test.d.ts +9 -0
  322. package/dist/convergence/__tests__/controller.test.d.ts.map +1 -0
  323. package/dist/convergence/__tests__/controller.test.js +585 -0
  324. package/dist/convergence/__tests__/controller.test.js.map +1 -0
  325. package/dist/convergence/__tests__/dual-signal.test.d.ts +14 -0
  326. package/dist/convergence/__tests__/dual-signal.test.d.ts.map +1 -0
  327. package/dist/convergence/__tests__/dual-signal.test.js +123 -0
  328. package/dist/convergence/__tests__/dual-signal.test.js.map +1 -0
  329. package/dist/convergence/__tests__/epic46-integration.test.d.ts +15 -0
  330. package/dist/convergence/__tests__/epic46-integration.test.d.ts.map +1 -0
  331. package/dist/convergence/__tests__/epic46-integration.test.js +522 -0
  332. package/dist/convergence/__tests__/epic46-integration.test.js.map +1 -0
  333. package/dist/convergence/__tests__/plateau.test.d.ts +6 -0
  334. package/dist/convergence/__tests__/plateau.test.d.ts.map +1 -0
  335. package/dist/convergence/__tests__/plateau.test.js +163 -0
  336. package/dist/convergence/__tests__/plateau.test.js.map +1 -0
  337. package/dist/convergence/__tests__/remediation.test.d.ts +11 -0
  338. package/dist/convergence/__tests__/remediation.test.d.ts.map +1 -0
  339. package/dist/convergence/__tests__/remediation.test.js +209 -0
  340. package/dist/convergence/__tests__/remediation.test.js.map +1 -0
  341. package/dist/convergence/__tests__/scenario-primary.test.d.ts +13 -0
  342. package/dist/convergence/__tests__/scenario-primary.test.d.ts.map +1 -0
  343. package/dist/convergence/__tests__/scenario-primary.test.js +183 -0
  344. package/dist/convergence/__tests__/scenario-primary.test.js.map +1 -0
  345. package/dist/convergence/budget.d.ts +181 -0
  346. package/dist/convergence/budget.d.ts.map +1 -0
  347. package/dist/convergence/budget.js +218 -0
  348. package/dist/convergence/budget.js.map +1 -0
  349. package/dist/convergence/controller.d.ts +133 -0
  350. package/dist/convergence/controller.d.ts.map +1 -0
  351. package/dist/convergence/controller.js +102 -0
  352. package/dist/convergence/controller.js.map +1 -0
  353. package/dist/convergence/dual-signal.d.ts +73 -0
  354. package/dist/convergence/dual-signal.d.ts.map +1 -0
  355. package/dist/convergence/dual-signal.js +78 -0
  356. package/dist/convergence/dual-signal.js.map +1 -0
  357. package/dist/convergence/index.d.ts +17 -0
  358. package/dist/convergence/index.d.ts.map +1 -0
  359. package/dist/convergence/index.js +11 -0
  360. package/dist/convergence/index.js.map +1 -0
  361. package/dist/convergence/plateau.d.ts +99 -0
  362. package/dist/convergence/plateau.d.ts.map +1 -0
  363. package/dist/convergence/plateau.js +83 -0
  364. package/dist/convergence/plateau.js.map +1 -0
  365. package/dist/convergence/remediation.d.ts +105 -0
  366. package/dist/convergence/remediation.d.ts.map +1 -0
  367. package/dist/convergence/remediation.js +117 -0
  368. package/dist/convergence/remediation.js.map +1 -0
  369. package/dist/events.d.ts +331 -0
  370. package/dist/events.d.ts.map +1 -0
  371. package/dist/events.js +13 -0
  372. package/dist/events.js.map +1 -0
  373. package/dist/factory-command.d.ts +40 -0
  374. package/dist/factory-command.d.ts.map +1 -0
  375. package/dist/factory-command.js +626 -0
  376. package/dist/factory-command.js.map +1 -0
  377. package/dist/factory-command.test.d.ts +8 -0
  378. package/dist/factory-command.test.d.ts.map +1 -0
  379. package/dist/factory-command.test.js +304 -0
  380. package/dist/factory-command.test.js.map +1 -0
  381. package/dist/graph/__tests__/attractor-compliance.test.d.ts +10 -0
  382. package/dist/graph/__tests__/attractor-compliance.test.d.ts.map +1 -0
  383. package/dist/graph/__tests__/attractor-compliance.test.js +766 -0
  384. package/dist/graph/__tests__/attractor-compliance.test.js.map +1 -0
  385. package/dist/graph/__tests__/checkpoint.test.d.ts +8 -0
  386. package/dist/graph/__tests__/checkpoint.test.d.ts.map +1 -0
  387. package/dist/graph/__tests__/checkpoint.test.js +329 -0
  388. package/dist/graph/__tests__/checkpoint.test.js.map +1 -0
  389. package/dist/graph/__tests__/condition-parser.test.d.ts +14 -0
  390. package/dist/graph/__tests__/condition-parser.test.d.ts.map +1 -0
  391. package/dist/graph/__tests__/condition-parser.test.js +406 -0
  392. package/dist/graph/__tests__/condition-parser.test.js.map +1 -0
  393. package/dist/graph/__tests__/context.test.d.ts +14 -0
  394. package/dist/graph/__tests__/context.test.d.ts.map +1 -0
  395. package/dist/graph/__tests__/context.test.js +276 -0
  396. package/dist/graph/__tests__/context.test.js.map +1 -0
  397. package/dist/graph/__tests__/edge-selector-events.test.d.ts +11 -0
  398. package/dist/graph/__tests__/edge-selector-events.test.d.ts.map +1 -0
  399. package/dist/graph/__tests__/edge-selector-events.test.js +184 -0
  400. package/dist/graph/__tests__/edge-selector-events.test.js.map +1 -0
  401. package/dist/graph/__tests__/edge-selector.test.d.ts +6 -0
  402. package/dist/graph/__tests__/edge-selector.test.d.ts.map +1 -0
  403. package/dist/graph/__tests__/edge-selector.test.js +452 -0
  404. package/dist/graph/__tests__/edge-selector.test.js.map +1 -0
  405. package/dist/graph/__tests__/executor-convergence.test.d.ts +12 -0
  406. package/dist/graph/__tests__/executor-convergence.test.d.ts.map +1 -0
  407. package/dist/graph/__tests__/executor-convergence.test.js +432 -0
  408. package/dist/graph/__tests__/executor-convergence.test.js.map +1 -0
  409. package/dist/graph/__tests__/executor-fidelity.test.d.ts +13 -0
  410. package/dist/graph/__tests__/executor-fidelity.test.d.ts.map +1 -0
  411. package/dist/graph/__tests__/executor-fidelity.test.js +335 -0
  412. package/dist/graph/__tests__/executor-fidelity.test.js.map +1 -0
  413. package/dist/graph/__tests__/executor.test.d.ts +14 -0
  414. package/dist/graph/__tests__/executor.test.d.ts.map +1 -0
  415. package/dist/graph/__tests__/executor.test.js +901 -0
  416. package/dist/graph/__tests__/executor.test.js.map +1 -0
  417. package/dist/graph/__tests__/fidelity.test.d.ts +8 -0
  418. package/dist/graph/__tests__/fidelity.test.d.ts.map +1 -0
  419. package/dist/graph/__tests__/fidelity.test.js +135 -0
  420. package/dist/graph/__tests__/fidelity.test.js.map +1 -0
  421. package/dist/graph/__tests__/llm-evaluator.test.d.ts +7 -0
  422. package/dist/graph/__tests__/llm-evaluator.test.d.ts.map +1 -0
  423. package/dist/graph/__tests__/llm-evaluator.test.js +106 -0
  424. package/dist/graph/__tests__/llm-evaluator.test.js.map +1 -0
  425. package/dist/graph/__tests__/parser-chaining.test.d.ts +13 -0
  426. package/dist/graph/__tests__/parser-chaining.test.d.ts.map +1 -0
  427. package/dist/graph/__tests__/parser-chaining.test.js +215 -0
  428. package/dist/graph/__tests__/parser-chaining.test.js.map +1 -0
  429. package/dist/graph/__tests__/parser.test.d.ts +22 -0
  430. package/dist/graph/__tests__/parser.test.d.ts.map +1 -0
  431. package/dist/graph/__tests__/parser.test.js +452 -0
  432. package/dist/graph/__tests__/parser.test.js.map +1 -0
  433. package/dist/graph/__tests__/run-state.test.d.ts +13 -0
  434. package/dist/graph/__tests__/run-state.test.d.ts.map +1 -0
  435. package/dist/graph/__tests__/run-state.test.js +189 -0
  436. package/dist/graph/__tests__/run-state.test.js.map +1 -0
  437. package/dist/graph/__tests__/transformer.test.d.ts +16 -0
  438. package/dist/graph/__tests__/transformer.test.d.ts.map +1 -0
  439. package/dist/graph/__tests__/transformer.test.js +350 -0
  440. package/dist/graph/__tests__/transformer.test.js.map +1 -0
  441. package/dist/graph/__tests__/validator-errors.test.d.ts +15 -0
  442. package/dist/graph/__tests__/validator-errors.test.d.ts.map +1 -0
  443. package/dist/graph/__tests__/validator-errors.test.js +572 -0
  444. package/dist/graph/__tests__/validator-errors.test.js.map +1 -0
  445. package/dist/graph/__tests__/validator-warnings.test.d.ts +15 -0
  446. package/dist/graph/__tests__/validator-warnings.test.d.ts.map +1 -0
  447. package/dist/graph/__tests__/validator-warnings.test.js +363 -0
  448. package/dist/graph/__tests__/validator-warnings.test.js.map +1 -0
  449. package/dist/graph/checkpoint.d.ts +61 -0
  450. package/dist/graph/checkpoint.d.ts.map +1 -0
  451. package/dist/graph/checkpoint.js +80 -0
  452. package/dist/graph/checkpoint.js.map +1 -0
  453. package/dist/graph/condition-parser.d.ts +55 -0
  454. package/dist/graph/condition-parser.d.ts.map +1 -0
  455. package/dist/graph/condition-parser.js +213 -0
  456. package/dist/graph/condition-parser.js.map +1 -0
  457. package/dist/graph/context.d.ts +53 -0
  458. package/dist/graph/context.d.ts.map +1 -0
  459. package/dist/graph/context.js +90 -0
  460. package/dist/graph/context.js.map +1 -0
  461. package/dist/graph/edge-selector.d.ts +74 -0
  462. package/dist/graph/edge-selector.d.ts.map +1 -0
  463. package/dist/graph/edge-selector.js +178 -0
  464. package/dist/graph/edge-selector.js.map +1 -0
  465. package/dist/graph/executor.d.ts +133 -0
  466. package/dist/graph/executor.d.ts.map +1 -0
  467. package/dist/graph/executor.js +787 -0
  468. package/dist/graph/executor.js.map +1 -0
  469. package/dist/graph/fidelity.d.ts +27 -0
  470. package/dist/graph/fidelity.d.ts.map +1 -0
  471. package/dist/graph/fidelity.js +38 -0
  472. package/dist/graph/fidelity.js.map +1 -0
  473. package/dist/graph/index.d.ts +13 -0
  474. package/dist/graph/index.d.ts.map +1 -0
  475. package/dist/graph/index.js +9 -0
  476. package/dist/graph/index.js.map +1 -0
  477. package/dist/graph/llm-evaluator.d.ts +61 -0
  478. package/dist/graph/llm-evaluator.d.ts.map +1 -0
  479. package/dist/graph/llm-evaluator.js +106 -0
  480. package/dist/graph/llm-evaluator.js.map +1 -0
  481. package/dist/graph/parser.d.ts +28 -0
  482. package/dist/graph/parser.d.ts.map +1 -0
  483. package/dist/graph/parser.js +370 -0
  484. package/dist/graph/parser.js.map +1 -0
  485. package/dist/graph/rules/error-rules.d.ts +19 -0
  486. package/dist/graph/rules/error-rules.d.ts.map +1 -0
  487. package/dist/graph/rules/error-rules.js +264 -0
  488. package/dist/graph/rules/error-rules.js.map +1 -0
  489. package/dist/graph/rules/warning-rules.d.ts +7 -0
  490. package/dist/graph/rules/warning-rules.d.ts.map +1 -0
  491. package/dist/graph/rules/warning-rules.js +165 -0
  492. package/dist/graph/rules/warning-rules.js.map +1 -0
  493. package/dist/graph/run-state.d.ts +62 -0
  494. package/dist/graph/run-state.d.ts.map +1 -0
  495. package/dist/graph/run-state.js +79 -0
  496. package/dist/graph/run-state.js.map +1 -0
  497. package/dist/graph/transformer.d.ts +31 -0
  498. package/dist/graph/transformer.d.ts.map +1 -0
  499. package/dist/graph/transformer.js +64 -0
  500. package/dist/graph/transformer.js.map +1 -0
  501. package/dist/graph/types.d.ts +320 -0
  502. package/dist/graph/types.d.ts.map +1 -0
  503. package/dist/graph/types.js +6 -0
  504. package/dist/graph/types.js.map +1 -0
  505. package/dist/graph/validator.d.ts +16 -0
  506. package/dist/graph/validator.d.ts.map +1 -0
  507. package/dist/graph/validator.js +40 -0
  508. package/dist/graph/validator.js.map +1 -0
  509. package/dist/handlers/__tests__/codergen-handler.test.d.ts +14 -0
  510. package/dist/handlers/__tests__/codergen-handler.test.d.ts.map +1 -0
  511. package/dist/handlers/__tests__/codergen-handler.test.js +442 -0
  512. package/dist/handlers/__tests__/codergen-handler.test.js.map +1 -0
  513. package/dist/handlers/__tests__/fan-in.test.d.ts +14 -0
  514. package/dist/handlers/__tests__/fan-in.test.d.ts.map +1 -0
  515. package/dist/handlers/__tests__/fan-in.test.js +399 -0
  516. package/dist/handlers/__tests__/fan-in.test.js.map +1 -0
  517. package/dist/handlers/__tests__/join-policy.test.d.ts +9 -0
  518. package/dist/handlers/__tests__/join-policy.test.d.ts.map +1 -0
  519. package/dist/handlers/__tests__/join-policy.test.js +201 -0
  520. package/dist/handlers/__tests__/join-policy.test.js.map +1 -0
  521. package/dist/handlers/__tests__/manager-loop.test.d.ts +14 -0
  522. package/dist/handlers/__tests__/manager-loop.test.d.ts.map +1 -0
  523. package/dist/handlers/__tests__/manager-loop.test.js +322 -0
  524. package/dist/handlers/__tests__/manager-loop.test.js.map +1 -0
  525. package/dist/handlers/__tests__/parallel-events.test.d.ts +12 -0
  526. package/dist/handlers/__tests__/parallel-events.test.d.ts.map +1 -0
  527. package/dist/handlers/__tests__/parallel-events.test.js +252 -0
  528. package/dist/handlers/__tests__/parallel-events.test.js.map +1 -0
  529. package/dist/handlers/__tests__/parallel-handler.test.d.ts +14 -0
  530. package/dist/handlers/__tests__/parallel-handler.test.d.ts.map +1 -0
  531. package/dist/handlers/__tests__/parallel-handler.test.js +337 -0
  532. package/dist/handlers/__tests__/parallel-handler.test.js.map +1 -0
  533. package/dist/handlers/__tests__/parallel-join.test.d.ts +9 -0
  534. package/dist/handlers/__tests__/parallel-join.test.d.ts.map +1 -0
  535. package/dist/handlers/__tests__/parallel-join.test.js +267 -0
  536. package/dist/handlers/__tests__/parallel-join.test.js.map +1 -0
  537. package/dist/handlers/__tests__/registry.test.d.ts +14 -0
  538. package/dist/handlers/__tests__/registry.test.d.ts.map +1 -0
  539. package/dist/handlers/__tests__/registry.test.js +315 -0
  540. package/dist/handlers/__tests__/registry.test.js.map +1 -0
  541. package/dist/handlers/__tests__/subgraph-events.test.d.ts +10 -0
  542. package/dist/handlers/__tests__/subgraph-events.test.d.ts.map +1 -0
  543. package/dist/handlers/__tests__/subgraph-events.test.js +189 -0
  544. package/dist/handlers/__tests__/subgraph-events.test.js.map +1 -0
  545. package/dist/handlers/__tests__/subgraph-inheritance.test.d.ts +14 -0
  546. package/dist/handlers/__tests__/subgraph-inheritance.test.d.ts.map +1 -0
  547. package/dist/handlers/__tests__/subgraph-inheritance.test.js +267 -0
  548. package/dist/handlers/__tests__/subgraph-inheritance.test.js.map +1 -0
  549. package/dist/handlers/__tests__/subgraph.test.d.ts +14 -0
  550. package/dist/handlers/__tests__/subgraph.test.d.ts.map +1 -0
  551. package/dist/handlers/__tests__/subgraph.test.js +369 -0
  552. package/dist/handlers/__tests__/subgraph.test.js.map +1 -0
  553. package/dist/handlers/__tests__/tool-handler.test.d.ts +11 -0
  554. package/dist/handlers/__tests__/tool-handler.test.d.ts.map +1 -0
  555. package/dist/handlers/__tests__/tool-handler.test.js +184 -0
  556. package/dist/handlers/__tests__/tool-handler.test.js.map +1 -0
  557. package/dist/handlers/__tests__/tool-scenario.test.d.ts +12 -0
  558. package/dist/handlers/__tests__/tool-scenario.test.d.ts.map +1 -0
  559. package/dist/handlers/__tests__/tool-scenario.test.js +222 -0
  560. package/dist/handlers/__tests__/tool-scenario.test.js.map +1 -0
  561. package/dist/handlers/__tests__/wait-human-handler.test.d.ts +11 -0
  562. package/dist/handlers/__tests__/wait-human-handler.test.d.ts.map +1 -0
  563. package/dist/handlers/__tests__/wait-human-handler.test.js +251 -0
  564. package/dist/handlers/__tests__/wait-human-handler.test.js.map +1 -0
  565. package/dist/handlers/codergen-handler.d.ts +83 -0
  566. package/dist/handlers/codergen-handler.d.ts.map +1 -0
  567. package/dist/handlers/codergen-handler.js +152 -0
  568. package/dist/handlers/codergen-handler.js.map +1 -0
  569. package/dist/handlers/conditional.d.ts +15 -0
  570. package/dist/handlers/conditional.d.ts.map +1 -0
  571. package/dist/handlers/conditional.js +16 -0
  572. package/dist/handlers/conditional.js.map +1 -0
  573. package/dist/handlers/exit.d.ts +10 -0
  574. package/dist/handlers/exit.d.ts.map +1 -0
  575. package/dist/handlers/exit.js +11 -0
  576. package/dist/handlers/exit.js.map +1 -0
  577. package/dist/handlers/fan-in.d.ts +74 -0
  578. package/dist/handlers/fan-in.d.ts.map +1 -0
  579. package/dist/handlers/fan-in.js +191 -0
  580. package/dist/handlers/fan-in.js.map +1 -0
  581. package/dist/handlers/index.d.ts +29 -0
  582. package/dist/handlers/index.d.ts.map +1 -0
  583. package/dist/handlers/index.js +22 -0
  584. package/dist/handlers/index.js.map +1 -0
  585. package/dist/handlers/join-policy.d.ts +117 -0
  586. package/dist/handlers/join-policy.d.ts.map +1 -0
  587. package/dist/handlers/join-policy.js +129 -0
  588. package/dist/handlers/join-policy.js.map +1 -0
  589. package/dist/handlers/manager-loop.d.ts +54 -0
  590. package/dist/handlers/manager-loop.d.ts.map +1 -0
  591. package/dist/handlers/manager-loop.js +177 -0
  592. package/dist/handlers/manager-loop.js.map +1 -0
  593. package/dist/handlers/parallel.d.ts +31 -0
  594. package/dist/handlers/parallel.d.ts.map +1 -0
  595. package/dist/handlers/parallel.js +362 -0
  596. package/dist/handlers/parallel.js.map +1 -0
  597. package/dist/handlers/registry.d.ts +75 -0
  598. package/dist/handlers/registry.d.ts.map +1 -0
  599. package/dist/handlers/registry.js +132 -0
  600. package/dist/handlers/registry.js.map +1 -0
  601. package/dist/handlers/start.d.ts +10 -0
  602. package/dist/handlers/start.d.ts.map +1 -0
  603. package/dist/handlers/start.js +11 -0
  604. package/dist/handlers/start.js.map +1 -0
  605. package/dist/handlers/subgraph.d.ts +46 -0
  606. package/dist/handlers/subgraph.d.ts.map +1 -0
  607. package/dist/handlers/subgraph.js +178 -0
  608. package/dist/handlers/subgraph.js.map +1 -0
  609. package/dist/handlers/tool.d.ts +36 -0
  610. package/dist/handlers/tool.d.ts.map +1 -0
  611. package/dist/handlers/tool.js +99 -0
  612. package/dist/handlers/tool.js.map +1 -0
  613. package/dist/handlers/types.d.ts +98 -0
  614. package/dist/handlers/types.d.ts.map +1 -0
  615. package/dist/handlers/types.js +6 -0
  616. package/dist/handlers/types.js.map +1 -0
  617. package/dist/handlers/wait-human.d.ts +65 -0
  618. package/dist/handlers/wait-human.d.ts.map +1 -0
  619. package/dist/handlers/wait-human.js +124 -0
  620. package/dist/handlers/wait-human.js.map +1 -0
  621. package/dist/index.d.ts +24 -0
  622. package/dist/index.d.ts.map +1 -0
  623. package/dist/index.js +33 -0
  624. package/dist/index.js.map +1 -0
  625. package/dist/llm/__tests__/client.test.d.ts +2 -0
  626. package/dist/llm/__tests__/client.test.d.ts.map +1 -0
  627. package/dist/llm/__tests__/client.test.js +198 -0
  628. package/dist/llm/__tests__/client.test.js.map +1 -0
  629. package/dist/llm/__tests__/types.test.d.ts +2 -0
  630. package/dist/llm/__tests__/types.test.d.ts.map +1 -0
  631. package/dist/llm/__tests__/types.test.js +289 -0
  632. package/dist/llm/__tests__/types.test.js.map +1 -0
  633. package/dist/llm/client.d.ts +19 -0
  634. package/dist/llm/client.d.ts.map +1 -0
  635. package/dist/llm/client.js +50 -0
  636. package/dist/llm/client.js.map +1 -0
  637. package/dist/llm/index.d.ts +6 -0
  638. package/dist/llm/index.d.ts.map +1 -0
  639. package/dist/llm/index.js +6 -0
  640. package/dist/llm/index.js.map +1 -0
  641. package/dist/llm/middleware/__tests__/cost-tracking.test.d.ts +2 -0
  642. package/dist/llm/middleware/__tests__/cost-tracking.test.d.ts.map +1 -0
  643. package/dist/llm/middleware/__tests__/cost-tracking.test.js +73 -0
  644. package/dist/llm/middleware/__tests__/cost-tracking.test.js.map +1 -0
  645. package/dist/llm/middleware/__tests__/logging.test.d.ts +2 -0
  646. package/dist/llm/middleware/__tests__/logging.test.d.ts.map +1 -0
  647. package/dist/llm/middleware/__tests__/logging.test.js +127 -0
  648. package/dist/llm/middleware/__tests__/logging.test.js.map +1 -0
  649. package/dist/llm/middleware/__tests__/retry.test.d.ts +2 -0
  650. package/dist/llm/middleware/__tests__/retry.test.d.ts.map +1 -0
  651. package/dist/llm/middleware/__tests__/retry.test.js +126 -0
  652. package/dist/llm/middleware/__tests__/retry.test.js.map +1 -0
  653. package/dist/llm/middleware/cost-tracking.d.ts +8 -0
  654. package/dist/llm/middleware/cost-tracking.d.ts.map +1 -0
  655. package/dist/llm/middleware/cost-tracking.js +34 -0
  656. package/dist/llm/middleware/cost-tracking.js.map +1 -0
  657. package/dist/llm/middleware/index.d.ts +5 -0
  658. package/dist/llm/middleware/index.d.ts.map +1 -0
  659. package/dist/llm/middleware/index.js +6 -0
  660. package/dist/llm/middleware/index.js.map +1 -0
  661. package/dist/llm/middleware/logging.d.ts +6 -0
  662. package/dist/llm/middleware/logging.d.ts.map +1 -0
  663. package/dist/llm/middleware/logging.js +54 -0
  664. package/dist/llm/middleware/logging.js.map +1 -0
  665. package/dist/llm/middleware/retry.d.ts +14 -0
  666. package/dist/llm/middleware/retry.d.ts.map +1 -0
  667. package/dist/llm/middleware/retry.js +40 -0
  668. package/dist/llm/middleware/retry.js.map +1 -0
  669. package/dist/llm/middleware/types.d.ts +23 -0
  670. package/dist/llm/middleware/types.d.ts.map +1 -0
  671. package/dist/llm/middleware/types.js +24 -0
  672. package/dist/llm/middleware/types.js.map +1 -0
  673. package/dist/llm/model-registry.d.ts +14 -0
  674. package/dist/llm/model-registry.d.ts.map +1 -0
  675. package/dist/llm/model-registry.js +32 -0
  676. package/dist/llm/model-registry.js.map +1 -0
  677. package/dist/llm/providers/__tests__/anthropic.test.d.ts +2 -0
  678. package/dist/llm/providers/__tests__/anthropic.test.d.ts.map +1 -0
  679. package/dist/llm/providers/__tests__/anthropic.test.js +412 -0
  680. package/dist/llm/providers/__tests__/anthropic.test.js.map +1 -0
  681. package/dist/llm/providers/__tests__/gemini.test.d.ts +2 -0
  682. package/dist/llm/providers/__tests__/gemini.test.d.ts.map +1 -0
  683. package/dist/llm/providers/__tests__/gemini.test.js +591 -0
  684. package/dist/llm/providers/__tests__/gemini.test.js.map +1 -0
  685. package/dist/llm/providers/__tests__/openai.test.d.ts +2 -0
  686. package/dist/llm/providers/__tests__/openai.test.d.ts.map +1 -0
  687. package/dist/llm/providers/__tests__/openai.test.js +546 -0
  688. package/dist/llm/providers/__tests__/openai.test.js.map +1 -0
  689. package/dist/llm/providers/anthropic.d.ts +25 -0
  690. package/dist/llm/providers/anthropic.d.ts.map +1 -0
  691. package/dist/llm/providers/anthropic.js +315 -0
  692. package/dist/llm/providers/anthropic.js.map +1 -0
  693. package/dist/llm/providers/gemini.d.ts +28 -0
  694. package/dist/llm/providers/gemini.d.ts.map +1 -0
  695. package/dist/llm/providers/gemini.js +429 -0
  696. package/dist/llm/providers/gemini.js.map +1 -0
  697. package/dist/llm/providers/index.d.ts +7 -0
  698. package/dist/llm/providers/index.d.ts.map +1 -0
  699. package/dist/llm/providers/index.js +4 -0
  700. package/dist/llm/providers/index.js.map +1 -0
  701. package/dist/llm/providers/openai.d.ts +28 -0
  702. package/dist/llm/providers/openai.d.ts.map +1 -0
  703. package/dist/llm/providers/openai.js +426 -0
  704. package/dist/llm/providers/openai.js.map +1 -0
  705. package/dist/llm/types.d.ts +127 -0
  706. package/dist/llm/types.d.ts.map +1 -0
  707. package/dist/llm/types.js +21 -0
  708. package/dist/llm/types.js.map +1 -0
  709. package/dist/persistence/__tests__/factory-queries.test.d.ts +9 -0
  710. package/dist/persistence/__tests__/factory-queries.test.d.ts.map +1 -0
  711. package/dist/persistence/__tests__/factory-queries.test.js +372 -0
  712. package/dist/persistence/__tests__/factory-queries.test.js.map +1 -0
  713. package/dist/persistence/__tests__/factory-schema.test.d.ts +6 -0
  714. package/dist/persistence/__tests__/factory-schema.test.d.ts.map +1 -0
  715. package/dist/persistence/__tests__/factory-schema.test.js +105 -0
  716. package/dist/persistence/__tests__/factory-schema.test.js.map +1 -0
  717. package/dist/persistence/factory-queries.d.ts +204 -0
  718. package/dist/persistence/factory-queries.d.ts.map +1 -0
  719. package/dist/persistence/factory-queries.js +186 -0
  720. package/dist/persistence/factory-queries.js.map +1 -0
  721. package/dist/persistence/factory-schema.d.ts +16 -0
  722. package/dist/persistence/factory-schema.d.ts.map +1 -0
  723. package/dist/persistence/factory-schema.js +89 -0
  724. package/dist/persistence/factory-schema.js.map +1 -0
  725. package/dist/scenarios/__tests__/cli-command-list.test.d.ts +7 -0
  726. package/dist/scenarios/__tests__/cli-command-list.test.d.ts.map +1 -0
  727. package/dist/scenarios/__tests__/cli-command-list.test.js +237 -0
  728. package/dist/scenarios/__tests__/cli-command-list.test.js.map +1 -0
  729. package/dist/scenarios/__tests__/cli-command.test.d.ts +11 -0
  730. package/dist/scenarios/__tests__/cli-command.test.d.ts.map +1 -0
  731. package/dist/scenarios/__tests__/cli-command.test.js +275 -0
  732. package/dist/scenarios/__tests__/cli-command.test.js.map +1 -0
  733. package/dist/scenarios/__tests__/integrity-pipeline.test.d.ts +15 -0
  734. package/dist/scenarios/__tests__/integrity-pipeline.test.d.ts.map +1 -0
  735. package/dist/scenarios/__tests__/integrity-pipeline.test.js +318 -0
  736. package/dist/scenarios/__tests__/integrity-pipeline.test.js.map +1 -0
  737. package/dist/scenarios/__tests__/runner-twins.test.d.ts +13 -0
  738. package/dist/scenarios/__tests__/runner-twins.test.d.ts.map +1 -0
  739. package/dist/scenarios/__tests__/runner-twins.test.js +205 -0
  740. package/dist/scenarios/__tests__/runner-twins.test.js.map +1 -0
  741. package/dist/scenarios/__tests__/scorer.test.d.ts +11 -0
  742. package/dist/scenarios/__tests__/scorer.test.d.ts.map +1 -0
  743. package/dist/scenarios/__tests__/scorer.test.js +225 -0
  744. package/dist/scenarios/__tests__/scorer.test.js.map +1 -0
  745. package/dist/scenarios/__tests__/scoring-integration.test.d.ts +8 -0
  746. package/dist/scenarios/__tests__/scoring-integration.test.d.ts.map +1 -0
  747. package/dist/scenarios/__tests__/scoring-integration.test.js +178 -0
  748. package/dist/scenarios/__tests__/scoring-integration.test.js.map +1 -0
  749. package/dist/scenarios/__tests__/store.test.d.ts +5 -0
  750. package/dist/scenarios/__tests__/store.test.d.ts.map +1 -0
  751. package/dist/scenarios/__tests__/store.test.js +169 -0
  752. package/dist/scenarios/__tests__/store.test.js.map +1 -0
  753. package/dist/scenarios/cli-command.d.ts +20 -0
  754. package/dist/scenarios/cli-command.d.ts.map +1 -0
  755. package/dist/scenarios/cli-command.js +66 -0
  756. package/dist/scenarios/cli-command.js.map +1 -0
  757. package/dist/scenarios/index.d.ts +11 -0
  758. package/dist/scenarios/index.d.ts.map +1 -0
  759. package/dist/scenarios/index.js +11 -0
  760. package/dist/scenarios/index.js.map +1 -0
  761. package/dist/scenarios/runner.d.ts +64 -0
  762. package/dist/scenarios/runner.d.ts.map +1 -0
  763. package/dist/scenarios/runner.js +205 -0
  764. package/dist/scenarios/runner.js.map +1 -0
  765. package/dist/scenarios/scorer.d.ts +69 -0
  766. package/dist/scenarios/scorer.d.ts.map +1 -0
  767. package/dist/scenarios/scorer.js +66 -0
  768. package/dist/scenarios/scorer.js.map +1 -0
  769. package/dist/scenarios/store.d.ts +33 -0
  770. package/dist/scenarios/store.d.ts.map +1 -0
  771. package/dist/scenarios/store.js +92 -0
  772. package/dist/scenarios/store.js.map +1 -0
  773. package/dist/scenarios/types.d.ts +40 -0
  774. package/dist/scenarios/types.d.ts.map +1 -0
  775. package/dist/scenarios/types.js +5 -0
  776. package/dist/scenarios/types.js.map +1 -0
  777. package/dist/stylesheet/__tests__/stylesheet.test.d.ts +17 -0
  778. package/dist/stylesheet/__tests__/stylesheet.test.d.ts.map +1 -0
  779. package/dist/stylesheet/__tests__/stylesheet.test.js +368 -0
  780. package/dist/stylesheet/__tests__/stylesheet.test.js.map +1 -0
  781. package/dist/stylesheet/parser.d.ts +44 -0
  782. package/dist/stylesheet/parser.d.ts.map +1 -0
  783. package/dist/stylesheet/parser.js +190 -0
  784. package/dist/stylesheet/parser.js.map +1 -0
  785. package/dist/stylesheet/resolver.d.ts +38 -0
  786. package/dist/stylesheet/resolver.d.ts.map +1 -0
  787. package/dist/stylesheet/resolver.js +96 -0
  788. package/dist/stylesheet/resolver.js.map +1 -0
  789. package/dist/templates/__tests__/templates.test.d.ts +7 -0
  790. package/dist/templates/__tests__/templates.test.d.ts.map +1 -0
  791. package/dist/templates/__tests__/templates.test.js +92 -0
  792. package/dist/templates/__tests__/templates.test.js.map +1 -0
  793. package/dist/templates/index.d.ts +38 -0
  794. package/dist/templates/index.d.ts.map +1 -0
  795. package/dist/templates/index.js +153 -0
  796. package/dist/templates/index.js.map +1 -0
  797. package/dist/twins/__tests__/docker-compose.test.d.ts +13 -0
  798. package/dist/twins/__tests__/docker-compose.test.d.ts.map +1 -0
  799. package/dist/twins/__tests__/docker-compose.test.js +247 -0
  800. package/dist/twins/__tests__/docker-compose.test.js.map +1 -0
  801. package/dist/twins/__tests__/health-monitor.test.d.ts +19 -0
  802. package/dist/twins/__tests__/health-monitor.test.d.ts.map +1 -0
  803. package/dist/twins/__tests__/health-monitor.test.js +301 -0
  804. package/dist/twins/__tests__/health-monitor.test.js.map +1 -0
  805. package/dist/twins/__tests__/integration/e2e.test.d.ts +18 -0
  806. package/dist/twins/__tests__/integration/e2e.test.d.ts.map +1 -0
  807. package/dist/twins/__tests__/integration/e2e.test.js +146 -0
  808. package/dist/twins/__tests__/integration/e2e.test.js.map +1 -0
  809. package/dist/twins/__tests__/integration/health-monitor-integration.test.d.ts +16 -0
  810. package/dist/twins/__tests__/integration/health-monitor-integration.test.d.ts.map +1 -0
  811. package/dist/twins/__tests__/integration/health-monitor-integration.test.js +183 -0
  812. package/dist/twins/__tests__/integration/health-monitor-integration.test.js.map +1 -0
  813. package/dist/twins/__tests__/integration/helpers.d.ts +32 -0
  814. package/dist/twins/__tests__/integration/helpers.d.ts.map +1 -0
  815. package/dist/twins/__tests__/integration/helpers.js +67 -0
  816. package/dist/twins/__tests__/integration/helpers.js.map +1 -0
  817. package/dist/twins/__tests__/integration/lifecycle.test.d.ts +14 -0
  818. package/dist/twins/__tests__/integration/lifecycle.test.d.ts.map +1 -0
  819. package/dist/twins/__tests__/integration/lifecycle.test.js +127 -0
  820. package/dist/twins/__tests__/integration/lifecycle.test.js.map +1 -0
  821. package/dist/twins/__tests__/integration/persistence-integration.test.d.ts +14 -0
  822. package/dist/twins/__tests__/integration/persistence-integration.test.d.ts.map +1 -0
  823. package/dist/twins/__tests__/integration/persistence-integration.test.js +132 -0
  824. package/dist/twins/__tests__/integration/persistence-integration.test.js.map +1 -0
  825. package/dist/twins/__tests__/persistence.test.d.ts +10 -0
  826. package/dist/twins/__tests__/persistence.test.d.ts.map +1 -0
  827. package/dist/twins/__tests__/persistence.test.js +300 -0
  828. package/dist/twins/__tests__/persistence.test.js.map +1 -0
  829. package/dist/twins/__tests__/registry.test.d.ts +7 -0
  830. package/dist/twins/__tests__/registry.test.d.ts.map +1 -0
  831. package/dist/twins/__tests__/registry.test.js +282 -0
  832. package/dist/twins/__tests__/registry.test.js.map +1 -0
  833. package/dist/twins/__tests__/run-state.test.d.ts +9 -0
  834. package/dist/twins/__tests__/run-state.test.d.ts.map +1 -0
  835. package/dist/twins/__tests__/run-state.test.js +112 -0
  836. package/dist/twins/__tests__/run-state.test.js.map +1 -0
  837. package/dist/twins/__tests__/templates-cli.test.d.ts +10 -0
  838. package/dist/twins/__tests__/templates-cli.test.d.ts.map +1 -0
  839. package/dist/twins/__tests__/templates-cli.test.js +187 -0
  840. package/dist/twins/__tests__/templates-cli.test.js.map +1 -0
  841. package/dist/twins/__tests__/templates.test.d.ts +7 -0
  842. package/dist/twins/__tests__/templates.test.d.ts.map +1 -0
  843. package/dist/twins/__tests__/templates.test.js +87 -0
  844. package/dist/twins/__tests__/templates.test.js.map +1 -0
  845. package/dist/twins/__tests__/twins-cli.test.d.ts +11 -0
  846. package/dist/twins/__tests__/twins-cli.test.d.ts.map +1 -0
  847. package/dist/twins/__tests__/twins-cli.test.js +365 -0
  848. package/dist/twins/__tests__/twins-cli.test.js.map +1 -0
  849. package/dist/twins/docker-compose.d.ts +48 -0
  850. package/dist/twins/docker-compose.d.ts.map +1 -0
  851. package/dist/twins/docker-compose.js +172 -0
  852. package/dist/twins/docker-compose.js.map +1 -0
  853. package/dist/twins/health-monitor.d.ts +36 -0
  854. package/dist/twins/health-monitor.d.ts.map +1 -0
  855. package/dist/twins/health-monitor.js +114 -0
  856. package/dist/twins/health-monitor.js.map +1 -0
  857. package/dist/twins/index.d.ts +22 -0
  858. package/dist/twins/index.d.ts.map +1 -0
  859. package/dist/twins/index.js +19 -0
  860. package/dist/twins/index.js.map +1 -0
  861. package/dist/twins/persistence.d.ts +100 -0
  862. package/dist/twins/persistence.d.ts.map +1 -0
  863. package/dist/twins/persistence.js +120 -0
  864. package/dist/twins/persistence.js.map +1 -0
  865. package/dist/twins/registry.d.ts +40 -0
  866. package/dist/twins/registry.d.ts.map +1 -0
  867. package/dist/twins/registry.js +175 -0
  868. package/dist/twins/registry.js.map +1 -0
  869. package/dist/twins/run-state.d.ts +47 -0
  870. package/dist/twins/run-state.d.ts.map +1 -0
  871. package/dist/twins/run-state.js +70 -0
  872. package/dist/twins/run-state.js.map +1 -0
  873. package/dist/twins/schema.d.ts +40 -0
  874. package/dist/twins/schema.d.ts.map +1 -0
  875. package/dist/twins/schema.js +46 -0
  876. package/dist/twins/schema.js.map +1 -0
  877. package/dist/twins/templates.d.ts +27 -0
  878. package/dist/twins/templates.d.ts.map +1 -0
  879. package/dist/twins/templates.js +63 -0
  880. package/dist/twins/templates.js.map +1 -0
  881. package/dist/twins/types.d.ts +55 -0
  882. package/dist/twins/types.d.ts.map +1 -0
  883. package/dist/twins/types.js +26 -0
  884. package/dist/twins/types.js.map +1 -0
  885. package/package.json +28 -0
@@ -0,0 +1,766 @@
1
+ /**
2
+ * Spec compliance tests for the Attractor graph engine.
3
+ *
4
+ * Replays pseudocode examples from the Attractor spec through
5
+ * selectEdge(), evaluateGates(), and checkpoint resume APIs.
6
+ *
7
+ * Story 42-17.
8
+ */
9
+ // AttractorBench structural conformance parity — Phase A mandatory
10
+ // See: https://github.com/strongdm/attractorbench
11
+ // Behavioral conformance tests are advisory in Phase A; required by Phase B exit.
12
+ import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
13
+ import os from 'node:os';
14
+ import path from 'node:path';
15
+ import { mkdir, rm } from 'node:fs/promises';
16
+ import crypto from 'node:crypto';
17
+ import { selectEdge } from '../edge-selector.js';
18
+ import { createConvergenceController } from '../../convergence/controller.js';
19
+ import { CheckpointManager } from '../checkpoint.js';
20
+ import { createGraphExecutor } from '../executor.js';
21
+ import { createValidator } from '../validator.js';
22
+ import { parseGraph } from '../parser.js';
23
+ import { GraphContext } from '../context.js';
24
+ // ---------------------------------------------------------------------------
25
+ // DOT fixture for resume/fidelity tests (spec Section 5.3)
26
+ // ---------------------------------------------------------------------------
27
+ const COMPLIANCE_RESUME_DOT = `
28
+ digraph compliance_resume {
29
+ graph [goal="Resume compliance test"]
30
+ start [shape=Mdiamond]
31
+ node1 [type=codergen, prompt="Step 1"]
32
+ node2 [type=codergen, prompt="Step 2"]
33
+ exit [shape=Msquare]
34
+
35
+ start -> node1
36
+ node1 -> node2
37
+ node2 -> exit
38
+ }
39
+ `;
40
+ const COMPLIANCE_FIDELITY_RESUME_DOT = `
41
+ digraph compliance_fidelity_resume {
42
+ graph [goal="Fidelity degradation test"]
43
+ start [shape=Mdiamond]
44
+ node1 [type=codergen, prompt="Step 1", fidelity=full]
45
+ node2 [type=codergen, prompt="Step 2"]
46
+ exit [shape=Msquare]
47
+
48
+ start -> node1
49
+ node1 -> node2
50
+ node2 -> exit
51
+ }
52
+ `;
53
+ // ---------------------------------------------------------------------------
54
+ // Shared helper factories
55
+ // ---------------------------------------------------------------------------
56
+ /** A node with all required fields at sensible defaults. */
57
+ const MINIMAL_NODE = {
58
+ id: '',
59
+ label: '',
60
+ shape: 'box',
61
+ type: 'codergen',
62
+ prompt: '',
63
+ maxRetries: 0,
64
+ goalGate: false,
65
+ retryTarget: '',
66
+ fallbackRetryTarget: '',
67
+ fidelity: '',
68
+ threadId: '',
69
+ class: '',
70
+ timeout: 0,
71
+ llmModel: '',
72
+ llmProvider: '',
73
+ reasoningEffort: '',
74
+ autoStatus: false,
75
+ allowPartial: false,
76
+ toolCommand: '',
77
+ backend: '',
78
+ };
79
+ function makeNode(id, overrides) {
80
+ return { ...MINIMAL_NODE, id, ...overrides };
81
+ }
82
+ function makeEdge(fromNode, toNode, overrides) {
83
+ return {
84
+ fromNode,
85
+ toNode,
86
+ label: '',
87
+ condition: '',
88
+ weight: 0,
89
+ fidelity: '',
90
+ threadId: '',
91
+ loopRestart: false,
92
+ ...overrides,
93
+ };
94
+ }
95
+ /**
96
+ * Build a minimal Graph conforming to the Graph interface.
97
+ * startNodeId / exitNodeId control which nodes serve as start/exit.
98
+ */
99
+ function makeGraph(nodes, edges, startNodeId, exitNodeId) {
100
+ const nodeMap = new Map(nodes.map((n) => [n.id, n]));
101
+ const resolveStart = () => {
102
+ if (startNodeId)
103
+ return nodeMap.get(startNodeId);
104
+ for (const n of nodeMap.values()) {
105
+ if (n.shape === 'Mdiamond')
106
+ return n;
107
+ }
108
+ throw new Error('No start node');
109
+ };
110
+ const resolveExit = () => {
111
+ if (exitNodeId)
112
+ return nodeMap.get(exitNodeId);
113
+ for (const n of nodeMap.values()) {
114
+ if (n.shape === 'Msquare')
115
+ return n;
116
+ }
117
+ throw new Error('No exit node');
118
+ };
119
+ return {
120
+ id: '',
121
+ goal: '',
122
+ label: '',
123
+ modelStylesheet: '',
124
+ defaultMaxRetries: 0,
125
+ retryTarget: '',
126
+ fallbackRetryTarget: '',
127
+ defaultFidelity: '',
128
+ nodes: nodeMap,
129
+ edges,
130
+ outgoingEdges: (nodeId) => edges.filter((e) => e.fromNode === nodeId),
131
+ startNode: resolveStart,
132
+ exitNode: resolveExit,
133
+ };
134
+ }
135
+ /** Build a Graph with only the minimal start/exit structure for lint-rule tests. */
136
+ function makeLintGraph(extraNodes, extraEdges, overrides) {
137
+ const baseNodes = [
138
+ makeNode('start', { shape: 'Mdiamond', type: '' }),
139
+ makeNode('exit', { shape: 'Msquare', type: '' }),
140
+ ...extraNodes.map((n) => makeNode(n.id, n)),
141
+ ];
142
+ const baseEdges = extraEdges.map((e) => makeEdge(e.fromNode, e.toNode, {
143
+ condition: e.condition ?? '',
144
+ loopRestart: e.loopRestart ?? false,
145
+ }));
146
+ const nodeMap = new Map(baseNodes.map((n) => [n.id, n]));
147
+ return {
148
+ id: 'test',
149
+ goal: '',
150
+ label: '',
151
+ modelStylesheet: overrides?.modelStylesheet ?? '',
152
+ defaultMaxRetries: 0,
153
+ retryTarget: overrides?.retryTarget ?? '',
154
+ fallbackRetryTarget: overrides?.fallbackRetryTarget ?? '',
155
+ defaultFidelity: '',
156
+ nodes: nodeMap,
157
+ edges: baseEdges,
158
+ outgoingEdges: (nodeId) => baseEdges.filter((e) => e.fromNode === nodeId),
159
+ startNode: () => nodeMap.get('start'),
160
+ exitNode: () => nodeMap.get('exit'),
161
+ };
162
+ }
163
+ /** Build a mock IHandlerRegistry that resolves every node via the given handler factory. */
164
+ function makeRegistry(handlerFactory, statusOverride) {
165
+ return {
166
+ register: vi.fn(),
167
+ registerShape: vi.fn(),
168
+ setDefault: vi.fn(),
169
+ resolve: vi.fn().mockImplementation(() => async (node, ctx) => {
170
+ handlerFactory(node, ctx);
171
+ return { status: statusOverride ?? 'SUCCESS' };
172
+ }),
173
+ };
174
+ }
175
+ /** Build a mock TypedEventBus that captures all emitted events. */
176
+ function makeEventBus() {
177
+ const nodeStartedIds = [];
178
+ const bus = {
179
+ emit: vi.fn().mockImplementation((event, payload) => {
180
+ if (event === 'graph:node-started') {
181
+ nodeStartedIds.push(payload.nodeId);
182
+ }
183
+ }),
184
+ on: vi.fn(),
185
+ off: vi.fn(),
186
+ };
187
+ return { bus, nodeStartedIds: () => [...nodeStartedIds] };
188
+ }
189
+ // ---------------------------------------------------------------------------
190
+ // Task 2: Edge selection compliance — spec Section 3.3
191
+ // ---------------------------------------------------------------------------
192
+ describe('selectEdge compliance — spec Section 3.3', () => {
193
+ // Step 1: Condition-matched edges beat weight
194
+ it('AC1 — Step 1: condition match returns conditional edge over high-weight unconditional', async () => {
195
+ const node = makeNode('origin');
196
+ const condEdge = makeEdge('origin', 'dest-conditional', { condition: 'outcome=success', weight: 0 });
197
+ const unconEdge = makeEdge('origin', 'dest-unconditional', { weight: 10 });
198
+ const graph = makeGraph([node], [condEdge, unconEdge]);
199
+ const ctx = new GraphContext({ outcome: 'success' });
200
+ const outcome = { status: 'SUCCESS' };
201
+ const selected = await selectEdge(node, outcome, ctx, graph);
202
+ expect(selected).toBe(condEdge);
203
+ });
204
+ it('AC1 variant — Step 1: no condition match falls through to Step 4 (weight)', async () => {
205
+ const node = makeNode('origin');
206
+ const condEdge = makeEdge('origin', 'dest-conditional', { condition: 'outcome=success', weight: 0 });
207
+ const unconEdge = makeEdge('origin', 'dest-unconditional', { weight: 10 });
208
+ const graph = makeGraph([node], [condEdge, unconEdge]);
209
+ const ctx = new GraphContext({ outcome: 'failure' });
210
+ const outcome = { status: 'SUCCESS' };
211
+ const selected = await selectEdge(node, outcome, ctx, graph);
212
+ expect(selected).toBe(unconEdge);
213
+ });
214
+ // Step 2: Preferred label with accelerator prefix stripping
215
+ it('AC2 — Step 2: [Y] prefix stripped, label matches preferredLabel "yes"', async () => {
216
+ const node = makeNode('origin');
217
+ const yesEdge = makeEdge('origin', 'yes-target', { label: '[Y] Yes' });
218
+ const noEdge = makeEdge('origin', 'no-target', { label: 'No' });
219
+ const graph = makeGraph([node], [yesEdge, noEdge]);
220
+ const ctx = new GraphContext();
221
+ const outcome = { status: 'SUCCESS', preferredLabel: 'yes' };
222
+ const selected = await selectEdge(node, outcome, ctx, graph);
223
+ expect(selected).toBe(yesEdge);
224
+ });
225
+ it('AC2 variant — Step 2: Y) prefix stripped, label matches preferredLabel "confirm"', async () => {
226
+ const node = makeNode('origin');
227
+ const confirmEdge = makeEdge('origin', 'confirm-target', { label: 'Y) Confirm' });
228
+ const otherEdge = makeEdge('origin', 'other-target', { label: 'Cancel' });
229
+ const graph = makeGraph([node], [confirmEdge, otherEdge]);
230
+ const ctx = new GraphContext();
231
+ const outcome = { status: 'SUCCESS', preferredLabel: 'confirm' };
232
+ const selected = await selectEdge(node, outcome, ctx, graph);
233
+ expect(selected).toBe(confirmEdge);
234
+ });
235
+ // Step 3: Suggested next IDs
236
+ it('Step 3: suggestedNextIds matches edge target', async () => {
237
+ const node = makeNode('origin');
238
+ const edgeA = makeEdge('origin', 'a');
239
+ const edgeB = makeEdge('origin', 'b');
240
+ const edgeC = makeEdge('origin', 'c');
241
+ const graph = makeGraph([node], [edgeA, edgeB, edgeC]);
242
+ const ctx = new GraphContext();
243
+ const outcome = { status: 'SUCCESS', suggestedNextIds: ['b'] };
244
+ const selected = await selectEdge(node, outcome, ctx, graph);
245
+ expect(selected).toBe(edgeB);
246
+ });
247
+ // Steps 4 & 5: Weight with lexical tiebreak
248
+ it('AC3 — Steps 4&5: weight 5 tie between charlie and alpha → lexically-first alpha wins', async () => {
249
+ const node = makeNode('origin');
250
+ const charlieEdge = makeEdge('origin', 'charlie', { weight: 5 });
251
+ const alphaEdge = makeEdge('origin', 'alpha', { weight: 5 });
252
+ const bravoEdge = makeEdge('origin', 'bravo', { weight: 3 });
253
+ const graph = makeGraph([node], [charlieEdge, alphaEdge, bravoEdge]);
254
+ const ctx = new GraphContext();
255
+ const outcome = { status: 'SUCCESS' };
256
+ const selected = await selectEdge(node, outcome, ctx, graph);
257
+ expect(selected).toBe(alphaEdge);
258
+ expect(selected?.toNode).toBe('alpha');
259
+ });
260
+ // Empty edges
261
+ it('empty edges: no outgoing edges → returns null', async () => {
262
+ const node = makeNode('origin');
263
+ const graph = makeGraph([node], []);
264
+ const ctx = new GraphContext();
265
+ const outcome = { status: 'SUCCESS' };
266
+ const selected = await selectEdge(node, outcome, ctx, graph);
267
+ expect(selected).toBeNull();
268
+ });
269
+ });
270
+ // ---------------------------------------------------------------------------
271
+ // Task 3: Goal gate compliance — spec Section 3.4
272
+ // ---------------------------------------------------------------------------
273
+ describe('evaluateGates compliance — spec Section 3.4', () => {
274
+ it('SUCCESS satisfies a goalGate=true node', () => {
275
+ const controller = createConvergenceController();
276
+ controller.recordOutcome('nodeA', 'SUCCESS');
277
+ const graph = makeGraph([makeNode('nodeA', { goalGate: true })], []);
278
+ const result = controller.evaluateGates(graph);
279
+ expect(result).toEqual({ satisfied: true, failingNodes: [] });
280
+ });
281
+ it('AC4 — PARTIAL_SUCCESS satisfies a goalGate=true node (spec Section 3.4)', () => {
282
+ const controller = createConvergenceController();
283
+ controller.recordOutcome('nodeA', 'PARTIAL_SUCCESS');
284
+ const graph = makeGraph([makeNode('nodeA', { goalGate: true })], []);
285
+ const result = controller.evaluateGates(graph);
286
+ expect(result).toEqual({ satisfied: true, failingNodes: [] });
287
+ });
288
+ it('AC4 — FAILURE does not satisfy goalGate=true node, appears in failingNodes', () => {
289
+ const controller = createConvergenceController();
290
+ controller.recordOutcome('nodeA', 'FAILURE');
291
+ const graph = makeGraph([makeNode('nodeA', { goalGate: true })], []);
292
+ const result = controller.evaluateGates(graph);
293
+ expect(result.satisfied).toBe(false);
294
+ expect(result.failingNodes).toContain('nodeA');
295
+ });
296
+ it('AC4 — Unrecorded outcome (never executed) fails gate', () => {
297
+ const controller = createConvergenceController();
298
+ // nodeA has no recorded outcome
299
+ const graph = makeGraph([makeNode('nodeA', { goalGate: true })], []);
300
+ const result = controller.evaluateGates(graph);
301
+ expect(result.satisfied).toBe(false);
302
+ expect(result.failingNodes).toContain('nodeA');
303
+ });
304
+ it('AC4 — PARTIAL_SUCCESS satisfies; FAILURE and unrecorded both fail', () => {
305
+ const controller = createConvergenceController();
306
+ controller.recordOutcome('partial', 'PARTIAL_SUCCESS');
307
+ controller.recordOutcome('failing', 'FAILURE');
308
+ // 'unrecorded' has no recorded outcome
309
+ const graph = makeGraph([
310
+ makeNode('partial', { goalGate: true }),
311
+ makeNode('failing', { goalGate: true }),
312
+ makeNode('unrecorded', { goalGate: true }),
313
+ ], []);
314
+ const result = controller.evaluateGates(graph);
315
+ expect(result.satisfied).toBe(false);
316
+ expect(result.failingNodes).toContain('failing');
317
+ expect(result.failingNodes).toContain('unrecorded');
318
+ expect(result.failingNodes).not.toContain('partial');
319
+ });
320
+ it('mixed SUCCESS + PARTIAL_SUCCESS: both satisfied → { satisfied: true, failingNodes: [] }', () => {
321
+ const controller = createConvergenceController();
322
+ controller.recordOutcome('nodeA', 'SUCCESS');
323
+ controller.recordOutcome('nodeB', 'PARTIAL_SUCCESS');
324
+ const graph = makeGraph([makeNode('nodeA', { goalGate: true }), makeNode('nodeB', { goalGate: true })], []);
325
+ const result = controller.evaluateGates(graph);
326
+ expect(result).toEqual({ satisfied: true, failingNodes: [] });
327
+ });
328
+ it('mixed SUCCESS + FAILURE: only failing node in failingNodes', () => {
329
+ const controller = createConvergenceController();
330
+ controller.recordOutcome('nodeA', 'SUCCESS');
331
+ controller.recordOutcome('nodeB', 'FAILURE');
332
+ const graph = makeGraph([makeNode('nodeA', { goalGate: true }), makeNode('nodeB', { goalGate: true })], []);
333
+ const result = controller.evaluateGates(graph);
334
+ expect(result.satisfied).toBe(false);
335
+ expect(result.failingNodes).toEqual(['nodeB']);
336
+ expect(result.failingNodes).not.toContain('nodeA');
337
+ });
338
+ it('no goalGate=true nodes: vacuously satisfied', () => {
339
+ const controller = createConvergenceController();
340
+ const graph = makeGraph([makeNode('nodeA', { goalGate: false }), makeNode('nodeB', { goalGate: false })], []);
341
+ const result = controller.evaluateGates(graph);
342
+ expect(result).toEqual({ satisfied: true, failingNodes: [] });
343
+ });
344
+ });
345
+ // ---------------------------------------------------------------------------
346
+ // Task 4: Checkpoint resume compliance — spec Section 5.3
347
+ // ---------------------------------------------------------------------------
348
+ describe('checkpoint resume compliance — spec Section 5.3', () => {
349
+ let tmpDir;
350
+ beforeEach(async () => {
351
+ tmpDir = path.join(os.tmpdir(), `attractor-compliance-${crypto.randomUUID()}`);
352
+ await mkdir(tmpDir, { recursive: true });
353
+ });
354
+ afterEach(async () => {
355
+ await rm(tmpDir, { recursive: true, force: true });
356
+ });
357
+ it('AC5 — completed nodes are skipped and context is restored on resume', async () => {
358
+ // Write seed checkpoint using CheckpointManager.save()
359
+ const checkpointManager = new CheckpointManager();
360
+ const seedContext = new GraphContext({ result: 'ok', attempt: '3' });
361
+ await checkpointManager.save(tmpDir, {
362
+ currentNode: 'node1',
363
+ completedNodes: ['start', 'node1'],
364
+ nodeRetries: {},
365
+ context: seedContext,
366
+ });
367
+ // Parse and validate the compliance resume graph
368
+ const graph = parseGraph(COMPLIANCE_RESUME_DOT);
369
+ const validator = createValidator();
370
+ const validationErrors = validator.validate(graph).filter((d) => d.severity === 'error');
371
+ expect(validationErrors).toHaveLength(0);
372
+ // Track graph:node-started events and captured contexts per node
373
+ const { bus, nodeStartedIds } = makeEventBus();
374
+ const capturedContextByNode = {};
375
+ const registry = makeRegistry((node, ctx) => {
376
+ capturedContextByNode[node.id] = ctx;
377
+ });
378
+ const executor = createGraphExecutor();
379
+ const config = {
380
+ runId: 'compliance-resume-test',
381
+ logsRoot: tmpDir,
382
+ handlerRegistry: registry,
383
+ eventBus: bus,
384
+ checkpointPath: path.join(tmpDir, 'checkpoint.json'),
385
+ };
386
+ const finalOutcome = await executor.run(graph, config);
387
+ // graph:node-started must NOT include 'start' or 'node1' (skipped)
388
+ expect(nodeStartedIds()).not.toContain('start');
389
+ expect(nodeStartedIds()).not.toContain('node1');
390
+ // graph:node-started MUST include 'node2' and 'exit' is never dispatched (exit check)
391
+ expect(nodeStartedIds()).toContain('node2');
392
+ // Context is restored: node2 handler receives context seeded from checkpoint
393
+ expect(capturedContextByNode['node2']).toBeDefined();
394
+ expect(capturedContextByNode['node2'].getString('result')).toBe('ok');
395
+ expect(capturedContextByNode['node2'].getString('attempt')).toBe('3');
396
+ // Final outcome is SUCCESS
397
+ expect(finalOutcome.status).toBe('SUCCESS');
398
+ });
399
+ });
400
+ // ---------------------------------------------------------------------------
401
+ // Task 5: Fidelity degradation on resume — spec Section 5.3 step 6
402
+ // ---------------------------------------------------------------------------
403
+ describe('fidelity degradation on resume — spec Section 5.3 step 6', () => {
404
+ let tmpDir;
405
+ beforeEach(async () => {
406
+ tmpDir = path.join(os.tmpdir(), `attractor-fidelity-${crypto.randomUUID()}`);
407
+ await mkdir(tmpDir, { recursive: true });
408
+ });
409
+ afterEach(async () => {
410
+ await rm(tmpDir, { recursive: true, force: true });
411
+ });
412
+ it('first resumed node gets fidelity="summary:high" when last completed node had fidelity="full"', async () => {
413
+ // Write seed checkpoint: node1 (which has fidelity=full in the DOT) is the last completed node
414
+ const checkpointManager = new CheckpointManager();
415
+ const seedContext = new GraphContext({ result: 'ok' });
416
+ await checkpointManager.save(tmpDir, {
417
+ currentNode: 'node1',
418
+ completedNodes: ['start', 'node1'],
419
+ nodeRetries: {},
420
+ context: seedContext,
421
+ });
422
+ // Parse graph where node1 has fidelity=full
423
+ const graph = parseGraph(COMPLIANCE_FIDELITY_RESUME_DOT);
424
+ // Capture the node argument passed to each handler dispatch
425
+ const capturedNodeByNodeId = {};
426
+ const registry = makeRegistry((node) => {
427
+ capturedNodeByNodeId[node.id] = node;
428
+ });
429
+ const { bus } = makeEventBus();
430
+ const config = {
431
+ runId: 'compliance-fidelity-test',
432
+ logsRoot: tmpDir,
433
+ handlerRegistry: registry,
434
+ eventBus: bus,
435
+ checkpointPath: path.join(tmpDir, 'checkpoint.json'),
436
+ };
437
+ await createGraphExecutor().run(graph, config);
438
+ // node2 is the first resumed node; its fidelity should be degraded to 'summary:high'
439
+ expect(capturedNodeByNodeId['node2']).toBeDefined();
440
+ expect(capturedNodeByNodeId['node2'].fidelity).toBe('summary:high');
441
+ });
442
+ it('subsequent resumed nodes use their configured fidelity (not the degraded value)', async () => {
443
+ // Extend the graph by adding node3 after node2
444
+ const EXTENDED_DOT = `
445
+ digraph compliance_fidelity_extended {
446
+ graph [goal="Extended fidelity test"]
447
+ start [shape=Mdiamond]
448
+ node1 [type=codergen, prompt="Step 1", fidelity=full]
449
+ node2 [type=codergen, prompt="Step 2", fidelity=medium]
450
+ node3 [type=codergen, prompt="Step 3", fidelity=high]
451
+ exit [shape=Msquare]
452
+
453
+ start -> node1
454
+ node1 -> node2
455
+ node2 -> node3
456
+ node3 -> exit
457
+ }
458
+ `;
459
+ const checkpointManager = new CheckpointManager();
460
+ const seedCtx = new GraphContext();
461
+ await checkpointManager.save(tmpDir, {
462
+ currentNode: 'node1',
463
+ completedNodes: ['start', 'node1'],
464
+ nodeRetries: {},
465
+ context: seedCtx,
466
+ });
467
+ const graph = parseGraph(EXTENDED_DOT);
468
+ const capturedNodeByNodeId = {};
469
+ const registry = makeRegistry((node) => {
470
+ capturedNodeByNodeId[node.id] = node;
471
+ });
472
+ const { bus } = makeEventBus();
473
+ const config = {
474
+ runId: 'compliance-fidelity-extended-test',
475
+ logsRoot: tmpDir,
476
+ handlerRegistry: registry,
477
+ eventBus: bus,
478
+ checkpointPath: path.join(tmpDir, 'checkpoint.json'),
479
+ };
480
+ await createGraphExecutor().run(graph, config);
481
+ // node2 is first resumed → degraded to 'summary:high'
482
+ expect(capturedNodeByNodeId['node2'].fidelity).toBe('summary:high');
483
+ // node3 is subsequent → uses its configured fidelity='high'
484
+ expect(capturedNodeByNodeId['node3'].fidelity).toBe('high');
485
+ });
486
+ });
487
+ // ---------------------------------------------------------------------------
488
+ // Task 6: Structural conformance — AttractorBench parity
489
+ // ---------------------------------------------------------------------------
490
+ describe('structural conformance — AttractorBench parity', () => {
491
+ // AttractorBench structural conformance parity — Phase A mandatory
492
+ // See: https://github.com/strongdm/attractorbench
493
+ // Behavioral conformance tests are advisory in Phase A; required by Phase B exit.
494
+ // -------------------------------------------------------------------------
495
+ // Attribute coverage — nodes (all 17 spec-defined + toolCommand from 42-11 + backend from 48-10)
496
+ // -------------------------------------------------------------------------
497
+ it('AC6 — all node attributes extracted with correct values from DOT', () => {
498
+ const dot = `
499
+ digraph compliance_node_attrs {
500
+ graph [goal="Attribute coverage"]
501
+ start [shape=Mdiamond]
502
+ target [
503
+ shape=box,
504
+ type=codergen,
505
+ label="Target Label",
506
+ prompt="Target prompt",
507
+ max_retries=3,
508
+ goal_gate=true,
509
+ retry_target="start",
510
+ fallback_retry_target="start",
511
+ fidelity=high,
512
+ thread_id="thread-1",
513
+ class="myclass",
514
+ timeout=30,
515
+ llm_model="gpt-4o",
516
+ llm_provider=openai,
517
+ reasoning_effort=high,
518
+ auto_status=false,
519
+ allow_partial=true,
520
+ tool_command="echo hello",
521
+ backend=direct
522
+ ]
523
+ exit [shape=Msquare]
524
+ start -> target
525
+ target -> exit
526
+ }
527
+ `;
528
+ const graph = parseGraph(dot);
529
+ const node = graph.nodes.get('target');
530
+ expect(node).toBeDefined();
531
+ expect(node.id).toBe('target');
532
+ expect(node.label).toBe('Target Label');
533
+ expect(node.shape).toBe('box');
534
+ expect(node.type).toBe('codergen');
535
+ expect(node.prompt).toBe('Target prompt');
536
+ expect(node.maxRetries).toBe(3);
537
+ expect(node.goalGate).toBe(true);
538
+ expect(node.retryTarget).toBe('start');
539
+ expect(node.fallbackRetryTarget).toBe('start');
540
+ expect(node.fidelity).toBe('high');
541
+ expect(node.threadId).toBe('thread-1');
542
+ expect(node.class).toBe('myclass');
543
+ expect(node.timeout).toBe(30);
544
+ expect(node.llmModel).toBe('gpt-4o');
545
+ expect(node.llmProvider).toBe('openai');
546
+ expect(node.reasoningEffort).toBe('high');
547
+ expect(node.autoStatus).toBe(false);
548
+ expect(node.allowPartial).toBe(true);
549
+ expect(node.toolCommand).toBe('echo hello');
550
+ expect(node.backend).toBe('direct');
551
+ });
552
+ // -------------------------------------------------------------------------
553
+ // Attribute coverage — edges (all 6 edge attributes)
554
+ // -------------------------------------------------------------------------
555
+ it('AC6 — all 6 edge attributes extracted with correct values from DOT', () => {
556
+ const dot = `
557
+ digraph compliance_edge_attrs {
558
+ graph [goal="Edge attribute coverage"]
559
+ start [shape=Mdiamond]
560
+ nodeA [type=codergen, prompt="a"]
561
+ exit [shape=Msquare]
562
+ start -> nodeA [
563
+ label="go",
564
+ condition="outcome=success",
565
+ weight=5,
566
+ fidelity=high,
567
+ thread_id="t2",
568
+ loop_restart=true
569
+ ]
570
+ nodeA -> exit
571
+ }
572
+ `;
573
+ const graph = parseGraph(dot);
574
+ const edge = graph.edges.find((e) => e.fromNode === 'start' && e.toNode === 'nodeA');
575
+ expect(edge).toBeDefined();
576
+ expect(edge.label).toBe('go');
577
+ expect(edge.condition).toBe('outcome=success');
578
+ expect(edge.weight).toBe(5);
579
+ expect(edge.fidelity).toBe('high');
580
+ expect(edge.threadId).toBe('t2');
581
+ expect(edge.loopRestart).toBe(true);
582
+ });
583
+ // -------------------------------------------------------------------------
584
+ // Lint rule coverage — 8 error-level rules (stories 42-4)
585
+ // -------------------------------------------------------------------------
586
+ it('AC6 — lint rule "start_node" fires (severity=error) when no start node exists', () => {
587
+ // Graph with no Mdiamond node and no id=start/Start
588
+ const graph = makeLintGraph([{ id: 'work', type: 'codergen', prompt: 'work' }], [{ fromNode: 'work', toNode: 'exit' }]);
589
+ // Override the nodes map to remove start (the makeLintGraph adds start by default — use manual build)
590
+ const nodeMap = new Map();
591
+ nodeMap.set('work', makeNode('work', { type: 'codergen', prompt: 'work' }));
592
+ nodeMap.set('exit', makeNode('exit', { shape: 'Msquare', type: '' }));
593
+ const edges = [makeEdge('work', 'exit')];
594
+ const g = {
595
+ id: 'test', goal: '', label: '', modelStylesheet: '',
596
+ defaultMaxRetries: 0, retryTarget: '', fallbackRetryTarget: '', defaultFidelity: '',
597
+ nodes: nodeMap,
598
+ edges,
599
+ outgoingEdges: (id) => edges.filter((e) => e.fromNode === id),
600
+ startNode: () => { throw new Error('no start'); },
601
+ exitNode: () => nodeMap.get('exit'),
602
+ };
603
+ const diags = createValidator().validate(g);
604
+ const rule = diags.find((d) => d.ruleId === 'start_node');
605
+ expect(rule).toBeDefined();
606
+ expect(rule.severity).toBe('error');
607
+ });
608
+ it('AC6 — lint rule "terminal_node" fires (severity=error) when no exit node exists', () => {
609
+ const nodeMap = new Map();
610
+ nodeMap.set('start', makeNode('start', { shape: 'Mdiamond', type: '' }));
611
+ nodeMap.set('work', makeNode('work', { type: 'codergen', prompt: 'work' }));
612
+ const edges = [makeEdge('start', 'work')];
613
+ const g = {
614
+ id: 'test', goal: '', label: '', modelStylesheet: '',
615
+ defaultMaxRetries: 0, retryTarget: '', fallbackRetryTarget: '', defaultFidelity: '',
616
+ nodes: nodeMap,
617
+ edges,
618
+ outgoingEdges: (id) => edges.filter((e) => e.fromNode === id),
619
+ startNode: () => nodeMap.get('start'),
620
+ exitNode: () => { throw new Error('no exit'); },
621
+ };
622
+ const diags = createValidator().validate(g);
623
+ const rule = diags.find((d) => d.ruleId === 'terminal_node');
624
+ expect(rule).toBeDefined();
625
+ expect(rule.severity).toBe('error');
626
+ });
627
+ it('AC6 — lint rule "start_no_incoming" fires (severity=error) when edge targets the start node', () => {
628
+ const graph = makeLintGraph([{ id: 'work', type: 'codergen', prompt: 'work' }], [
629
+ { fromNode: 'start', toNode: 'work' },
630
+ { fromNode: 'work', toNode: 'exit' },
631
+ { fromNode: 'work', toNode: 'start' }, // incoming edge to start
632
+ ]);
633
+ const diags = createValidator().validate(graph);
634
+ const rule = diags.find((d) => d.ruleId === 'start_no_incoming');
635
+ expect(rule).toBeDefined();
636
+ expect(rule.severity).toBe('error');
637
+ });
638
+ it('AC6 — lint rule "exit_no_outgoing" fires (severity=error) when exit node has an outgoing edge', () => {
639
+ const graph = makeLintGraph([{ id: 'work', type: 'codergen', prompt: 'work' }], [
640
+ { fromNode: 'start', toNode: 'work' },
641
+ { fromNode: 'work', toNode: 'exit' },
642
+ { fromNode: 'exit', toNode: 'work' }, // outgoing edge from exit
643
+ ]);
644
+ const diags = createValidator().validate(graph);
645
+ const rule = diags.find((d) => d.ruleId === 'exit_no_outgoing');
646
+ expect(rule).toBeDefined();
647
+ expect(rule.severity).toBe('error');
648
+ });
649
+ it('AC6 — lint rule "edge_target_exists" fires (severity=error) when edge targets non-existent node', () => {
650
+ // Build graph directly — DOT auto-creates referenced nodes, so we use in-memory construction
651
+ const nodeMap = new Map();
652
+ nodeMap.set('start', makeNode('start', { shape: 'Mdiamond', type: '' }));
653
+ nodeMap.set('work', makeNode('work', { type: 'codergen', prompt: 'work' }));
654
+ nodeMap.set('exit', makeNode('exit', { shape: 'Msquare', type: '' }));
655
+ const edges = [
656
+ makeEdge('start', 'work'),
657
+ makeEdge('work', 'exit'),
658
+ makeEdge('work', 'ghost_node'), // ghost_node does NOT exist in nodeMap
659
+ ];
660
+ const g = {
661
+ id: 'test', goal: '', label: '', modelStylesheet: '',
662
+ defaultMaxRetries: 0, retryTarget: '', fallbackRetryTarget: '', defaultFidelity: '',
663
+ nodes: nodeMap,
664
+ edges,
665
+ outgoingEdges: (id) => edges.filter((e) => e.fromNode === id),
666
+ startNode: () => nodeMap.get('start'),
667
+ exitNode: () => nodeMap.get('exit'),
668
+ };
669
+ const diags = createValidator().validate(g);
670
+ const rule = diags.find((d) => d.ruleId === 'edge_target_exists');
671
+ expect(rule).toBeDefined();
672
+ expect(rule.severity).toBe('error');
673
+ });
674
+ it('AC6 — lint rule "reachability" fires (severity=error) when a node is unreachable', () => {
675
+ const graph = makeLintGraph([
676
+ { id: 'work', type: 'codergen', prompt: 'work' },
677
+ { id: 'unreachable', type: 'codergen', prompt: 'unreachable' },
678
+ ], [
679
+ { fromNode: 'start', toNode: 'work' },
680
+ { fromNode: 'work', toNode: 'exit' },
681
+ // 'unreachable' has no incoming edges from start
682
+ ]);
683
+ const diags = createValidator().validate(graph);
684
+ const rule = diags.find((d) => d.ruleId === 'reachability');
685
+ expect(rule).toBeDefined();
686
+ expect(rule.severity).toBe('error');
687
+ });
688
+ it('AC6 — lint rule "condition_syntax" fires (severity=error) for invalid condition expression', () => {
689
+ const graph = makeLintGraph([{ id: 'work', type: 'codergen', prompt: 'work' }], [
690
+ { fromNode: 'start', toNode: 'work' },
691
+ { fromNode: 'work', toNode: 'exit', condition: 'a==b' }, // double-equals is invalid
692
+ ]);
693
+ const diags = createValidator().validate(graph);
694
+ const rule = diags.find((d) => d.ruleId === 'condition_syntax');
695
+ expect(rule).toBeDefined();
696
+ expect(rule.severity).toBe('error');
697
+ });
698
+ it('AC6 — lint rule "stylesheet_syntax" fires (severity=error) for invalid stylesheet content', () => {
699
+ // Build graph with invalid inline stylesheet
700
+ const graph = makeLintGraph([{ id: 'work', type: 'codergen', prompt: 'work' }], [{ fromNode: 'start', toNode: 'work' }, { fromNode: 'work', toNode: 'exit' }], { modelStylesheet: '!!! invalid stylesheet content !!!' });
701
+ const diags = createValidator().validate(graph);
702
+ const rule = diags.find((d) => d.ruleId === 'stylesheet_syntax');
703
+ expect(rule).toBeDefined();
704
+ expect(rule.severity).toBe('error');
705
+ });
706
+ // -------------------------------------------------------------------------
707
+ // Lint rule coverage — 5 warning-level rules (stories 42-5)
708
+ // -------------------------------------------------------------------------
709
+ it('AC6 — lint rule "type_known" fires (severity=warning) for unknown node type', () => {
710
+ const graph = makeLintGraph([{ id: 'work', type: 'wizard_type', prompt: 'work' }], // wizard_type is unknown
711
+ [{ fromNode: 'start', toNode: 'work' }, { fromNode: 'work', toNode: 'exit' }]);
712
+ const diags = createValidator().validate(graph);
713
+ const rule = diags.find((d) => d.ruleId === 'type_known');
714
+ expect(rule).toBeDefined();
715
+ expect(rule.severity).toBe('warning');
716
+ });
717
+ it('AC6 — lint rule "fidelity_valid" fires (severity=warning) for invalid fidelity value', () => {
718
+ const graph = makeLintGraph([{ id: 'work', type: 'codergen', prompt: 'work', fidelity: 'ultra_high' }], // invalid fidelity
719
+ [{ fromNode: 'start', toNode: 'work' }, { fromNode: 'work', toNode: 'exit' }]);
720
+ const diags = createValidator().validate(graph);
721
+ const rule = diags.find((d) => d.ruleId === 'fidelity_valid');
722
+ expect(rule).toBeDefined();
723
+ expect(rule.severity).toBe('warning');
724
+ });
725
+ it('AC6 — lint rule "retry_target_exists" fires (severity=warning) for missing retryTarget node', () => {
726
+ const graph = makeLintGraph([{ id: 'work', type: 'codergen', prompt: 'work', retryTarget: 'nonexistent' }], [{ fromNode: 'start', toNode: 'work' }, { fromNode: 'work', toNode: 'exit' }]);
727
+ const diags = createValidator().validate(graph);
728
+ const rule = diags.find((d) => d.ruleId === 'retry_target_exists');
729
+ expect(rule).toBeDefined();
730
+ expect(rule.severity).toBe('warning');
731
+ });
732
+ it('AC6 — lint rule "goal_gate_has_retry" fires (severity=warning) when goalGate=true node has no retryTarget', () => {
733
+ const graph = makeLintGraph([{ id: 'work', type: 'codergen', prompt: 'work', goalGate: true, retryTarget: '' }], [{ fromNode: 'start', toNode: 'work' }, { fromNode: 'work', toNode: 'exit' }]);
734
+ const diags = createValidator().validate(graph);
735
+ const rule = diags.find((d) => d.ruleId === 'goal_gate_has_retry');
736
+ expect(rule).toBeDefined();
737
+ expect(rule.severity).toBe('warning');
738
+ });
739
+ it('AC6 — lint rule "prompt_on_llm_nodes" fires (severity=warning) for codergen node with no prompt or label', () => {
740
+ const graph = makeLintGraph([{ id: 'work', type: 'codergen', prompt: '', label: '' }], // no prompt AND no label
741
+ [{ fromNode: 'start', toNode: 'work' }, { fromNode: 'work', toNode: 'exit' }]);
742
+ const diags = createValidator().validate(graph);
743
+ const rule = diags.find((d) => d.ruleId === 'prompt_on_llm_nodes');
744
+ expect(rule).toBeDefined();
745
+ expect(rule.severity).toBe('warning');
746
+ });
747
+ // -------------------------------------------------------------------------
748
+ // Edge selection determinism — identical inputs always produce same output
749
+ // -------------------------------------------------------------------------
750
+ it('AC6 — selectEdge is deterministic: identical inputs always produce same edge', async () => {
751
+ const node = makeNode('origin');
752
+ const edgeA = makeEdge('origin', 'alpha', { weight: 5 });
753
+ const edgeB = makeEdge('origin', 'bravo', { weight: 5 });
754
+ const edgeC = makeEdge('origin', 'charlie', { weight: 3 });
755
+ const graph = makeGraph([node], [edgeA, edgeB, edgeC]);
756
+ const ctx = new GraphContext({ x: '1' });
757
+ const outcome = { status: 'SUCCESS' };
758
+ const first = await selectEdge(node, outcome, ctx, graph);
759
+ const second = await selectEdge(node, outcome, ctx, graph);
760
+ const third = await selectEdge(node, outcome, ctx, graph);
761
+ expect(first?.toNode).toBe(second?.toNode);
762
+ expect(second?.toNode).toBe(third?.toNode);
763
+ expect(first?.toNode).toBe('alpha'); // lexically first among weight-5 ties
764
+ });
765
+ });
766
+ //# sourceMappingURL=attractor-compliance.test.js.map