@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,868 @@
1
+ // packages/factory/src/agent/__tests__/loop.test.ts
2
+ // Tests for CodingAgentSession and the core agentic loop.
3
+ // Story 48-7: Coding Agent Loop — Core Agentic Loop
4
+ import { describe, it, expect, vi } from 'vitest';
5
+ import { EventEmitter } from 'node:events';
6
+ import { createSession, CodingAgentSession, convertHistoryToMessages } from '../loop.js';
7
+ import { EventKind, SessionState } from '../types.js';
8
+ // ---------------------------------------------------------------------------
9
+ // Test helpers / shared mocks
10
+ // ---------------------------------------------------------------------------
11
+ function makeUsage(override = {}) {
12
+ return {
13
+ inputTokens: 10,
14
+ outputTokens: 5,
15
+ totalTokens: 15,
16
+ ...override,
17
+ };
18
+ }
19
+ function makeTextResponse(content = 'Done!') {
20
+ return {
21
+ content,
22
+ toolCalls: [],
23
+ usage: makeUsage(),
24
+ model: 'test-model',
25
+ stopReason: 'stop',
26
+ providerMetadata: {},
27
+ };
28
+ }
29
+ function makeToolCallResponse(toolName, callId = 'call1', args = {}) {
30
+ return {
31
+ content: '',
32
+ toolCalls: [{ id: callId, name: toolName, arguments: args }],
33
+ usage: makeUsage(),
34
+ model: 'test-model',
35
+ stopReason: 'tool_calls',
36
+ providerMetadata: {},
37
+ };
38
+ }
39
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
40
+ function makeToolDefinition(name, executor) {
41
+ return {
42
+ name,
43
+ description: `Mock ${name} tool`,
44
+ inputSchema: { type: 'object', properties: {}, required: [] },
45
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
46
+ executor: executor ?? vi.fn().mockResolvedValue(`${name} output`),
47
+ };
48
+ }
49
+ function makeProfile(tools = [], overrides = {}) {
50
+ return {
51
+ id: 'test',
52
+ model: 'test-model',
53
+ supports_streaming: false,
54
+ context_window_size: 100_000,
55
+ supports_parallel_tool_calls: true,
56
+ build_system_prompt: vi.fn().mockReturnValue('System prompt'),
57
+ tools: vi.fn().mockReturnValue(tools),
58
+ provider_options: vi.fn().mockReturnValue({}),
59
+ ...overrides,
60
+ };
61
+ }
62
+ function makeMockClient(completeFn) {
63
+ return { complete: completeFn };
64
+ }
65
+ function makeEnv() {
66
+ return {
67
+ workdir: '/tmp',
68
+ exec: vi.fn().mockResolvedValue({ stdout: '', stderr: '', exitCode: 0 }),
69
+ };
70
+ }
71
+ function makeConfig(overrides = {}) {
72
+ return {
73
+ max_turns: 0,
74
+ max_tool_rounds_per_input: 0,
75
+ tool_output_limits: new Map(),
76
+ ...overrides,
77
+ };
78
+ }
79
+ // Collect all events for specific kinds
80
+ function collectEvents(session, ...kinds) {
81
+ const events = [];
82
+ for (const kind of kinds) {
83
+ session.on(kind, e => events.push(e));
84
+ }
85
+ return events;
86
+ }
87
+ // Collect all events from a session
88
+ function collectAllEvents(session) {
89
+ const events = [];
90
+ for (const kind of Object.values(EventKind)) {
91
+ session.on(kind, e => events.push(e));
92
+ }
93
+ return events;
94
+ }
95
+ // ---------------------------------------------------------------------------
96
+ // Test Suite: createSession
97
+ // ---------------------------------------------------------------------------
98
+ describe('createSession', () => {
99
+ it('returns a CodingAgentSession instance in IDLE state with a UUID id', () => {
100
+ const session = createSession({
101
+ llmClient: makeMockClient(vi.fn()),
102
+ providerProfile: makeProfile(),
103
+ executionEnv: makeEnv(),
104
+ });
105
+ expect(session).toBeInstanceOf(CodingAgentSession);
106
+ expect(session.state).toBe(SessionState.IDLE);
107
+ expect(session.id).toMatch(/^[0-9a-f]{8}-[0-9a-f]{4}-4[0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/i);
108
+ });
109
+ it('emits SESSION_START event synchronously during creation', () => {
110
+ // Spy on EventEmitter.prototype.emit before construction
111
+ const emitSpy = vi.spyOn(EventEmitter.prototype, 'emit');
112
+ const session = createSession({
113
+ llmClient: makeMockClient(vi.fn()),
114
+ providerProfile: makeProfile(),
115
+ executionEnv: makeEnv(),
116
+ });
117
+ const startCall = emitSpy.mock.calls.find(args => args[0] === EventKind.SESSION_START);
118
+ expect(startCall).toBeDefined();
119
+ if (startCall) {
120
+ const event = startCall[1];
121
+ expect(event.kind).toBe(EventKind.SESSION_START);
122
+ expect(event.session_id).toBe(session.id);
123
+ }
124
+ emitSpy.mockRestore();
125
+ });
126
+ it('returns session with empty history', () => {
127
+ const session = createSession({
128
+ llmClient: makeMockClient(vi.fn()),
129
+ providerProfile: makeProfile(),
130
+ executionEnv: makeEnv(),
131
+ });
132
+ expect(session.history).toHaveLength(0);
133
+ });
134
+ });
135
+ // ---------------------------------------------------------------------------
136
+ // Test Suite: close()
137
+ // ---------------------------------------------------------------------------
138
+ describe('session.close()', () => {
139
+ it('transitions state to CLOSED and emits SESSION_END', () => {
140
+ const session = createSession({
141
+ llmClient: makeMockClient(vi.fn()),
142
+ providerProfile: makeProfile(),
143
+ executionEnv: makeEnv(),
144
+ });
145
+ const events = [];
146
+ session.on(EventKind.SESSION_END, e => events.push(e));
147
+ session.close();
148
+ expect(session.state).toBe(SessionState.CLOSED);
149
+ expect(events).toHaveLength(1);
150
+ expect(events[0].kind).toBe(EventKind.SESSION_END);
151
+ expect(events[0].session_id).toBe(session.id);
152
+ });
153
+ });
154
+ // ---------------------------------------------------------------------------
155
+ // Test Suite: processInput — natural completion
156
+ // ---------------------------------------------------------------------------
157
+ describe('processInput — natural completion (no tool calls)', () => {
158
+ it('emits USER_INPUT then ASSISTANT_TEXT_END then PROCESSING_END for text-only response', async () => {
159
+ const complete = vi.fn().mockResolvedValue(makeTextResponse('Hello, world!'));
160
+ const session = createSession({
161
+ llmClient: makeMockClient(complete),
162
+ providerProfile: makeProfile(),
163
+ executionEnv: makeEnv(),
164
+ });
165
+ const events = collectAllEvents(session);
166
+ await session.processInput('hi');
167
+ const kinds = events.map(e => e.kind);
168
+ expect(kinds).toContain(EventKind.USER_INPUT);
169
+ expect(kinds).toContain(EventKind.ASSISTANT_TEXT_END);
170
+ expect(kinds).toContain(EventKind.PROCESSING_END);
171
+ // Verify ordering
172
+ const uiIdx = kinds.indexOf(EventKind.USER_INPUT);
173
+ const ateIdx = kinds.indexOf(EventKind.ASSISTANT_TEXT_END);
174
+ const peIdx = kinds.indexOf(EventKind.PROCESSING_END);
175
+ expect(uiIdx).toBeLessThan(ateIdx);
176
+ expect(ateIdx).toBeLessThan(peIdx);
177
+ });
178
+ it('appends UserTurn and AssistantTurn to history', async () => {
179
+ const complete = vi.fn().mockResolvedValue(makeTextResponse('Response'));
180
+ const session = createSession({
181
+ llmClient: makeMockClient(complete),
182
+ providerProfile: makeProfile(),
183
+ executionEnv: makeEnv(),
184
+ });
185
+ await session.processInput('my question');
186
+ expect(session.history).toHaveLength(2);
187
+ expect(session.history[0].type).toBe('user');
188
+ expect(session.history[1].type).toBe('assistant');
189
+ });
190
+ it('returns to IDLE state after processing', async () => {
191
+ const complete = vi.fn().mockResolvedValue(makeTextResponse());
192
+ const session = createSession({
193
+ llmClient: makeMockClient(complete),
194
+ providerProfile: makeProfile(),
195
+ executionEnv: makeEnv(),
196
+ });
197
+ await session.processInput('test');
198
+ expect(session.state).toBe(SessionState.IDLE);
199
+ });
200
+ it('calls LLM with a request built from the session (model, systemPrompt, toolChoice)', async () => {
201
+ const complete = vi.fn().mockResolvedValue(makeTextResponse());
202
+ const profile = makeProfile();
203
+ const session = createSession({
204
+ llmClient: makeMockClient(complete),
205
+ providerProfile: profile,
206
+ executionEnv: makeEnv(),
207
+ });
208
+ await session.processInput('hello');
209
+ expect(complete).toHaveBeenCalledOnce();
210
+ const req = complete.mock.calls[0][0];
211
+ expect(req.model).toBe('test-model');
212
+ expect(req.systemPrompt).toBe('System prompt');
213
+ expect(req.toolChoice).toBe('auto');
214
+ });
215
+ });
216
+ // ---------------------------------------------------------------------------
217
+ // Test Suite: processInput — tool call loop
218
+ // ---------------------------------------------------------------------------
219
+ describe('processInput — one tool call then natural completion', () => {
220
+ it('emits TOOL_CALL_START, TOOL_CALL_END, final ASSISTANT_TEXT_END, PROCESSING_END', async () => {
221
+ const tool = makeToolDefinition('my_tool');
222
+ const complete = vi
223
+ .fn()
224
+ .mockResolvedValueOnce(makeToolCallResponse('my_tool'))
225
+ .mockResolvedValueOnce(makeTextResponse('All done'));
226
+ const session = createSession({
227
+ llmClient: makeMockClient(complete),
228
+ providerProfile: makeProfile([tool]),
229
+ executionEnv: makeEnv(),
230
+ });
231
+ const events = collectAllEvents(session);
232
+ await session.processInput('run my tool');
233
+ const kinds = events.map(e => e.kind);
234
+ expect(kinds).toContain(EventKind.TOOL_CALL_START);
235
+ expect(kinds).toContain(EventKind.TOOL_CALL_END);
236
+ expect(kinds).toContain(EventKind.PROCESSING_END);
237
+ // LLM was called twice (once returning tool call, once returning text)
238
+ expect(complete).toHaveBeenCalledTimes(2);
239
+ });
240
+ it('appends UserTurn, AssistantTurn, ToolResultsTurn, and final AssistantTurn to history', async () => {
241
+ const tool = makeToolDefinition('my_tool');
242
+ const complete = vi
243
+ .fn()
244
+ .mockResolvedValueOnce(makeToolCallResponse('my_tool'))
245
+ .mockResolvedValueOnce(makeTextResponse('All done'));
246
+ const session = createSession({
247
+ llmClient: makeMockClient(complete),
248
+ providerProfile: makeProfile([tool]),
249
+ executionEnv: makeEnv(),
250
+ });
251
+ await session.processInput('run my tool');
252
+ expect(session.history).toHaveLength(4);
253
+ expect(session.history[0].type).toBe('user');
254
+ expect(session.history[1].type).toBe('assistant');
255
+ expect(session.history[2].type).toBe('tool_results');
256
+ expect(session.history[3].type).toBe('assistant');
257
+ });
258
+ });
259
+ // ---------------------------------------------------------------------------
260
+ // Test Suite: Limit enforcement
261
+ // ---------------------------------------------------------------------------
262
+ describe('max_tool_rounds_per_input enforcement', () => {
263
+ it('stops after N rounds and emits TURN_LIMIT with correct data', async () => {
264
+ const tool = makeToolDefinition('looping_tool');
265
+ // LLM always returns a tool call (never natural completion)
266
+ const complete = vi.fn().mockResolvedValue(makeToolCallResponse('looping_tool'));
267
+ const session = createSession({
268
+ llmClient: makeMockClient(complete),
269
+ providerProfile: makeProfile([tool]),
270
+ executionEnv: makeEnv(),
271
+ config: makeConfig({ max_tool_rounds_per_input: 2 }),
272
+ });
273
+ const turnLimitEvents = [];
274
+ session.on(EventKind.TURN_LIMIT, e => turnLimitEvents.push(e));
275
+ const processingEndEvents = [];
276
+ session.on(EventKind.PROCESSING_END, e => processingEndEvents.push(e));
277
+ await session.processInput('loop me');
278
+ expect(turnLimitEvents).toHaveLength(1);
279
+ expect(turnLimitEvents[0].data.reason).toBe('max_tool_rounds_per_input');
280
+ expect(turnLimitEvents[0].data.round).toBe(2);
281
+ // PROCESSING_END emitted after TURN_LIMIT
282
+ expect(processingEndEvents).toHaveLength(1);
283
+ expect(session.state).toBe(SessionState.IDLE);
284
+ });
285
+ });
286
+ describe('max_turns enforcement', () => {
287
+ it('stops loop when total turn count reaches max_turns and emits TURN_LIMIT', async () => {
288
+ const tool = makeToolDefinition('looping_tool');
289
+ const complete = vi.fn().mockResolvedValue(makeToolCallResponse('looping_tool'));
290
+ // max_turns: 3 — after UserTurn(1) + AssistantTurn(2) + ToolResultsTurn(3) = 3 turns
291
+ // At next iteration start, history.length = 3 >= 3 → TURN_LIMIT
292
+ const session = createSession({
293
+ llmClient: makeMockClient(complete),
294
+ providerProfile: makeProfile([tool]),
295
+ executionEnv: makeEnv(),
296
+ config: makeConfig({ max_turns: 3 }),
297
+ });
298
+ const turnLimitEvents = [];
299
+ session.on(EventKind.TURN_LIMIT, e => turnLimitEvents.push(e));
300
+ await session.processInput('test');
301
+ expect(turnLimitEvents).toHaveLength(1);
302
+ expect(turnLimitEvents[0].data.reason).toBe('max_turns');
303
+ expect(session.state).toBe(SessionState.IDLE);
304
+ });
305
+ it('does not enforce limit when max_turns is 0 (unlimited)', async () => {
306
+ const complete = vi.fn().mockResolvedValue(makeTextResponse());
307
+ const session = createSession({
308
+ llmClient: makeMockClient(complete),
309
+ providerProfile: makeProfile(),
310
+ executionEnv: makeEnv(),
311
+ config: makeConfig({ max_turns: 0 }),
312
+ });
313
+ const turnLimitEvents = [];
314
+ session.on(EventKind.TURN_LIMIT, e => turnLimitEvents.push(e));
315
+ await session.processInput('test');
316
+ expect(turnLimitEvents).toHaveLength(0);
317
+ });
318
+ });
319
+ // ---------------------------------------------------------------------------
320
+ // Test Suite: Unknown tool handling
321
+ // ---------------------------------------------------------------------------
322
+ describe('unknown tool handling', () => {
323
+ it('returns is_error: true without throwing for unknown tool calls', async () => {
324
+ // Profile has NO tools registered
325
+ const complete = vi
326
+ .fn()
327
+ .mockResolvedValueOnce(makeToolCallResponse('nonexistent_tool', 'call99'))
328
+ .mockResolvedValueOnce(makeTextResponse());
329
+ const session = createSession({
330
+ llmClient: makeMockClient(complete),
331
+ providerProfile: makeProfile([]), // empty tools list
332
+ executionEnv: makeEnv(),
333
+ });
334
+ const toolEndEvents = [];
335
+ session.on(EventKind.TOOL_CALL_END, e => toolEndEvents.push(e));
336
+ // Should NOT throw
337
+ await expect(session.processInput('call unknown')).resolves.toBeUndefined();
338
+ expect(toolEndEvents).toHaveLength(1);
339
+ expect(toolEndEvents[0].data.is_error).toBe(true);
340
+ expect(toolEndEvents[0].data.output).toContain('Unknown tool: nonexistent_tool');
341
+ });
342
+ it('records the error in ToolResultsTurn history', async () => {
343
+ const complete = vi
344
+ .fn()
345
+ .mockResolvedValueOnce(makeToolCallResponse('ghost_tool', 'c1'))
346
+ .mockResolvedValueOnce(makeTextResponse());
347
+ const session = createSession({
348
+ llmClient: makeMockClient(complete),
349
+ providerProfile: makeProfile([]),
350
+ executionEnv: makeEnv(),
351
+ });
352
+ await session.processInput('call ghost');
353
+ const toolResultsTurn = session.history.find(t => t.type === 'tool_results');
354
+ expect(toolResultsTurn).toBeDefined();
355
+ if (toolResultsTurn?.type === 'tool_results') {
356
+ expect(toolResultsTurn.results[0].is_error).toBe(true);
357
+ expect(toolResultsTurn.results[0].content).toContain('Unknown tool: ghost_tool');
358
+ }
359
+ else {
360
+ throw new Error('Expected tool_results turn');
361
+ }
362
+ });
363
+ });
364
+ // ---------------------------------------------------------------------------
365
+ // Test Suite: TOOL_CALL_END carries full untruncated output
366
+ // ---------------------------------------------------------------------------
367
+ describe('TOOL_CALL_END full output in event', () => {
368
+ it('event carries full untruncated output when truncation is applied to LLM content', async () => {
369
+ // 'shell' has a 30,000 char default limit
370
+ const largeOutput = 'X'.repeat(35_000);
371
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
372
+ const shellTool = makeToolDefinition('shell', vi.fn().mockResolvedValue(largeOutput));
373
+ const complete = vi
374
+ .fn()
375
+ .mockResolvedValueOnce(makeToolCallResponse('shell', 'c1'))
376
+ .mockResolvedValueOnce(makeTextResponse());
377
+ const session = createSession({
378
+ llmClient: makeMockClient(complete),
379
+ providerProfile: makeProfile([shellTool]),
380
+ executionEnv: makeEnv(),
381
+ });
382
+ const toolEndEvents = [];
383
+ session.on(EventKind.TOOL_CALL_END, e => toolEndEvents.push(e));
384
+ await session.processInput('run shell');
385
+ // Event output = full untruncated
386
+ expect(toolEndEvents).toHaveLength(1);
387
+ expect(toolEndEvents[0].data.output).toBe(largeOutput);
388
+ expect(toolEndEvents[0].data.output.length).toBe(35_000);
389
+ // History ToolResultsTurn content should be truncated
390
+ const toolResultsTurn = session.history.find(t => t.type === 'tool_results');
391
+ expect(toolResultsTurn).toBeDefined();
392
+ if (toolResultsTurn?.type === 'tool_results') {
393
+ expect(toolResultsTurn.results[0].content).toContain('characters truncated from middle.');
394
+ expect(toolResultsTurn.results[0].content.length).toBeLessThan(35_000);
395
+ }
396
+ });
397
+ });
398
+ // ---------------------------------------------------------------------------
399
+ // Test Suite: Parallel vs sequential tool dispatch
400
+ // ---------------------------------------------------------------------------
401
+ describe('parallel tool call dispatch', () => {
402
+ it('dispatches multiple tool calls concurrently when supports_parallel_tool_calls=true', async () => {
403
+ let activeCount = 0;
404
+ let maxActiveCount = 0;
405
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
406
+ const makeParallelExecutor = () => vi.fn().mockImplementation(async () => {
407
+ activeCount++;
408
+ maxActiveCount = Math.max(maxActiveCount, activeCount);
409
+ await new Promise(r => setTimeout(r, 15));
410
+ activeCount--;
411
+ return 'output';
412
+ });
413
+ const tool1 = makeToolDefinition('tool1', makeParallelExecutor());
414
+ const tool2 = makeToolDefinition('tool2', makeParallelExecutor());
415
+ const complete = vi
416
+ .fn()
417
+ .mockResolvedValueOnce({
418
+ content: '',
419
+ toolCalls: [
420
+ { id: 'c1', name: 'tool1', arguments: {} },
421
+ { id: 'c2', name: 'tool2', arguments: {} },
422
+ ],
423
+ usage: makeUsage(),
424
+ model: 'test-model',
425
+ stopReason: 'tool_calls',
426
+ providerMetadata: {},
427
+ })
428
+ .mockResolvedValueOnce(makeTextResponse());
429
+ const profile = makeProfile([tool1, tool2], { supports_parallel_tool_calls: true });
430
+ const session = createSession({
431
+ llmClient: makeMockClient(complete),
432
+ providerProfile: profile,
433
+ executionEnv: makeEnv(),
434
+ });
435
+ await session.processInput('run tools in parallel');
436
+ // Parallel: both tools run simultaneously → maxActiveCount = 2
437
+ expect(maxActiveCount).toBe(2);
438
+ });
439
+ it('dispatches tool calls sequentially when supports_parallel_tool_calls=false', async () => {
440
+ const callOrder = [];
441
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
442
+ const makeSeqExecutor = (name) => vi.fn().mockImplementation(async () => {
443
+ callOrder.push(`start_${name}`);
444
+ await new Promise(r => setTimeout(r, 10));
445
+ callOrder.push(`end_${name}`);
446
+ return 'output';
447
+ });
448
+ const tool1 = makeToolDefinition('tool1', makeSeqExecutor('tool1'));
449
+ const tool2 = makeToolDefinition('tool2', makeSeqExecutor('tool2'));
450
+ const complete = vi
451
+ .fn()
452
+ .mockResolvedValueOnce({
453
+ content: '',
454
+ toolCalls: [
455
+ { id: 'c1', name: 'tool1', arguments: {} },
456
+ { id: 'c2', name: 'tool2', arguments: {} },
457
+ ],
458
+ usage: makeUsage(),
459
+ model: 'test-model',
460
+ stopReason: 'tool_calls',
461
+ providerMetadata: {},
462
+ })
463
+ .mockResolvedValueOnce(makeTextResponse());
464
+ const profile = makeProfile([tool1, tool2], { supports_parallel_tool_calls: false });
465
+ const session = createSession({
466
+ llmClient: makeMockClient(complete),
467
+ providerProfile: profile,
468
+ executionEnv: makeEnv(),
469
+ });
470
+ await session.processInput('run tools sequentially');
471
+ // Sequential: tool1 must complete before tool2 starts
472
+ expect(callOrder).toEqual(['start_tool1', 'end_tool1', 'start_tool2', 'end_tool2']);
473
+ });
474
+ });
475
+ // ---------------------------------------------------------------------------
476
+ // Test Suite: convertHistoryToMessages
477
+ // ---------------------------------------------------------------------------
478
+ describe('convertHistoryToMessages', () => {
479
+ it('converts UserTurn to user role message with text content', () => {
480
+ const history = [{ type: 'user', content: 'hello', timestamp: new Date() }];
481
+ const messages = convertHistoryToMessages(history);
482
+ expect(messages).toHaveLength(1);
483
+ expect(messages[0].role).toBe('user');
484
+ expect(messages[0].content[0]).toMatchObject({ kind: 'text', text: 'hello' });
485
+ });
486
+ it('converts SteeringTurn to user role message', () => {
487
+ const history = [{ type: 'steering', content: 'steer this', timestamp: new Date() }];
488
+ const messages = convertHistoryToMessages(history);
489
+ expect(messages).toHaveLength(1);
490
+ expect(messages[0].role).toBe('user');
491
+ expect(messages[0].content[0]).toMatchObject({ kind: 'text', text: 'steer this' });
492
+ });
493
+ it('skips SystemTurn (system prompt passed separately)', () => {
494
+ const history = [{ type: 'system', content: 'system msg', timestamp: new Date() }];
495
+ const messages = convertHistoryToMessages(history);
496
+ expect(messages).toHaveLength(0);
497
+ });
498
+ it('converts ToolResultsTurn to user role with tool_result content parts', () => {
499
+ const history = [{
500
+ type: 'tool_results',
501
+ results: [{ tool_call_id: 'c1', content: 'result', is_error: false }],
502
+ timestamp: new Date(),
503
+ }];
504
+ const messages = convertHistoryToMessages(history);
505
+ expect(messages).toHaveLength(1);
506
+ expect(messages[0].role).toBe('user');
507
+ expect(messages[0].content[0]).toMatchObject({
508
+ kind: 'tool_result',
509
+ toolResult: { toolCallId: 'c1', content: 'result', isError: false },
510
+ });
511
+ });
512
+ it('converts AssistantTurn with tool calls to assistant role with tool_call parts', () => {
513
+ const history = [{
514
+ type: 'assistant',
515
+ content: 'Using tool',
516
+ tool_calls: [{ id: 'tc1', name: 'my_tool', arguments: { x: 1 } }],
517
+ reasoning: null,
518
+ usage: makeUsage(),
519
+ response_id: null,
520
+ timestamp: new Date(),
521
+ }];
522
+ const messages = convertHistoryToMessages(history);
523
+ expect(messages).toHaveLength(1);
524
+ expect(messages[0].role).toBe('assistant');
525
+ const parts = messages[0].content;
526
+ expect(parts.some(p => p.kind === 'text')).toBe(true);
527
+ expect(parts.some(p => p.kind === 'tool_call')).toBe(true);
528
+ });
529
+ });
530
+ // ---------------------------------------------------------------------------
531
+ // Test Suite: error handling
532
+ // ---------------------------------------------------------------------------
533
+ describe('error handling', () => {
534
+ it('emits ERROR event and sets state to CLOSED when LLM throws', async () => {
535
+ const complete = vi.fn().mockRejectedValue(new Error('LLM unavailable'));
536
+ const session = createSession({
537
+ llmClient: makeMockClient(complete),
538
+ providerProfile: makeProfile(),
539
+ executionEnv: makeEnv(),
540
+ });
541
+ const errorEvents = [];
542
+ session.on(EventKind.ERROR, e => errorEvents.push(e));
543
+ await expect(session.processInput('test')).rejects.toThrow('LLM unavailable');
544
+ expect(errorEvents).toHaveLength(1);
545
+ expect(errorEvents[0].data.message).toBe('LLM unavailable');
546
+ expect(session.state).toBe(SessionState.CLOSED);
547
+ });
548
+ });
549
+ // ---------------------------------------------------------------------------
550
+ // Test Suite: steer() — steering injection (Story 48-8)
551
+ // ---------------------------------------------------------------------------
552
+ describe('steer() — steering injection', () => {
553
+ it('message queued via steer() appears as SteeringTurn before next LLM call', async () => {
554
+ const tool = makeToolDefinition('my_tool');
555
+ const complete = vi
556
+ .fn()
557
+ .mockResolvedValueOnce(makeToolCallResponse('my_tool'))
558
+ .mockResolvedValueOnce(makeTextResponse('Done'));
559
+ const session = createSession({
560
+ llmClient: makeMockClient(complete),
561
+ providerProfile: makeProfile([tool]),
562
+ executionEnv: makeEnv(),
563
+ });
564
+ // Queue a steering message before processing starts
565
+ session.steer('please use a different approach');
566
+ await session.processInput('do something');
567
+ // SteeringTurn should appear in history
568
+ const steeringTurns = session.history.filter(t => t.type === 'steering');
569
+ expect(steeringTurns.length).toBeGreaterThanOrEqual(1);
570
+ const turn = steeringTurns[0];
571
+ expect(turn?.type).toBe('steering');
572
+ if (turn?.type === 'steering') {
573
+ expect(turn.content).toBe('please use a different approach');
574
+ }
575
+ });
576
+ it('STEERING_INJECTED event is emitted with correct content', async () => {
577
+ const complete = vi.fn().mockResolvedValue(makeTextResponse('Done'));
578
+ const session = createSession({
579
+ llmClient: makeMockClient(complete),
580
+ providerProfile: makeProfile(),
581
+ executionEnv: makeEnv(),
582
+ });
583
+ const steeringEvents = [];
584
+ session.on(EventKind.STEERING_INJECTED, e => steeringEvents.push(e));
585
+ session.steer('redirect now');
586
+ await session.processInput('hello');
587
+ expect(steeringEvents).toHaveLength(1);
588
+ expect(steeringEvents[0].data.content).toBe('redirect now');
589
+ });
590
+ it('steering message is in history before the next LLM request (correct history position)', async () => {
591
+ const tool = makeToolDefinition('my_tool');
592
+ // Capture what history looks like at each LLM call
593
+ const historySnapshotsAtLLMCall = [];
594
+ const complete = vi.fn().mockImplementation(() => {
595
+ // Capture the types of turns in history at call time
596
+ historySnapshotsAtLLMCall.push(session.history.map(t => t.type));
597
+ if (complete.mock.calls.length === 1) {
598
+ return Promise.resolve(makeToolCallResponse('my_tool'));
599
+ }
600
+ return Promise.resolve(makeTextResponse('Done'));
601
+ });
602
+ const session = createSession({
603
+ llmClient: makeMockClient(complete),
604
+ providerProfile: makeProfile([tool]),
605
+ executionEnv: makeEnv(),
606
+ });
607
+ // Steer before the second LLM call (while the tool is executing)
608
+ // We'll queue it before processInput; it will be drained before the first LLM call
609
+ session.steer('steered!');
610
+ await session.processInput('use tool');
611
+ // First LLM call should see the steering turn in history
612
+ expect(historySnapshotsAtLLMCall[0]).toContain('steering');
613
+ });
614
+ it('steer() called while IDLE: message is buffered and injected on next processInput call', async () => {
615
+ const complete = vi.fn().mockResolvedValue(makeTextResponse('OK'));
616
+ const session = createSession({
617
+ llmClient: makeMockClient(complete),
618
+ providerProfile: makeProfile(),
619
+ executionEnv: makeEnv(),
620
+ });
621
+ // Session is IDLE; queue steering
622
+ session.steer('buffered-steer');
623
+ expect(session._steeringQueue).toHaveLength(1);
624
+ const steeringEvents = [];
625
+ session.on(EventKind.STEERING_INJECTED, e => steeringEvents.push(e));
626
+ await session.processInput('now process');
627
+ // Steering should have been drained
628
+ expect(steeringEvents).toHaveLength(1);
629
+ expect(steeringEvents[0].data.content).toBe('buffered-steer');
630
+ expect(session._steeringQueue).toHaveLength(0);
631
+ });
632
+ it('multiple steer() messages are drained FIFO', async () => {
633
+ const complete = vi.fn().mockResolvedValue(makeTextResponse('OK'));
634
+ const session = createSession({
635
+ llmClient: makeMockClient(complete),
636
+ providerProfile: makeProfile(),
637
+ executionEnv: makeEnv(),
638
+ });
639
+ session.steer('first');
640
+ session.steer('second');
641
+ session.steer('third');
642
+ const steeringEvents = [];
643
+ session.on(EventKind.STEERING_INJECTED, e => steeringEvents.push(e));
644
+ await session.processInput('go');
645
+ expect(steeringEvents).toHaveLength(3);
646
+ expect(steeringEvents[0].data.content).toBe('first');
647
+ expect(steeringEvents[1].data.content).toBe('second');
648
+ expect(steeringEvents[2].data.content).toBe('third');
649
+ });
650
+ });
651
+ // ---------------------------------------------------------------------------
652
+ // Test Suite: _drainSteering (Story 48-8)
653
+ // ---------------------------------------------------------------------------
654
+ describe('_drainSteering()', () => {
655
+ it('dequeues all messages in FIFO order, appends SteeringTurns, emits STEERING_INJECTED', () => {
656
+ const complete = vi.fn();
657
+ const session = createSession({
658
+ llmClient: makeMockClient(complete),
659
+ providerProfile: makeProfile(),
660
+ executionEnv: makeEnv(),
661
+ });
662
+ const steeringEvents = [];
663
+ session.on(EventKind.STEERING_INJECTED, e => steeringEvents.push(e));
664
+ session._steeringQueue.push('msg1', 'msg2', 'msg3');
665
+ session._drainSteering();
666
+ // All messages dequeued
667
+ expect(session._steeringQueue).toHaveLength(0);
668
+ // SteeringTurns appended to history
669
+ expect(session.history).toHaveLength(3);
670
+ expect(session.history.every(t => t.type === 'steering')).toBe(true);
671
+ expect(session.history[0].content).toBe('msg1');
672
+ expect(session.history[1].content).toBe('msg2');
673
+ expect(session.history[2].content).toBe('msg3');
674
+ // Events emitted per message
675
+ expect(steeringEvents).toHaveLength(3);
676
+ expect(steeringEvents[0].data.content).toBe('msg1');
677
+ expect(steeringEvents[1].data.content).toBe('msg2');
678
+ expect(steeringEvents[2].data.content).toBe('msg3');
679
+ });
680
+ it('does nothing when steering queue is empty', () => {
681
+ const session = createSession({
682
+ llmClient: makeMockClient(vi.fn()),
683
+ providerProfile: makeProfile(),
684
+ executionEnv: makeEnv(),
685
+ });
686
+ const steeringEvents = [];
687
+ session.on(EventKind.STEERING_INJECTED, e => steeringEvents.push(e));
688
+ session._drainSteering();
689
+ expect(session.history).toHaveLength(0);
690
+ expect(steeringEvents).toHaveLength(0);
691
+ });
692
+ it('SteeringTurn has a timestamp (Date instance)', () => {
693
+ const session = createSession({
694
+ llmClient: makeMockClient(vi.fn()),
695
+ providerProfile: makeProfile(),
696
+ executionEnv: makeEnv(),
697
+ });
698
+ session._steeringQueue.push('msg');
699
+ session._drainSteering();
700
+ const turn = session.history[0];
701
+ expect(turn?.type).toBe('steering');
702
+ if (turn?.type === 'steering') {
703
+ expect(turn.timestamp).toBeInstanceOf(Date);
704
+ }
705
+ });
706
+ });
707
+ // ---------------------------------------------------------------------------
708
+ // Test Suite: follow_up() — follow-up queue (Story 48-8)
709
+ // ---------------------------------------------------------------------------
710
+ describe('follow_up() — follow-up queue', () => {
711
+ it('queued follow-up triggers a new processing cycle after natural completion', async () => {
712
+ const complete = vi
713
+ .fn()
714
+ .mockResolvedValueOnce(makeTextResponse('First response'))
715
+ .mockResolvedValueOnce(makeTextResponse('Second response'));
716
+ const session = createSession({
717
+ llmClient: makeMockClient(complete),
718
+ providerProfile: makeProfile(),
719
+ executionEnv: makeEnv(),
720
+ });
721
+ session.follow_up('follow-up message');
722
+ const userInputEvents = [];
723
+ session.on(EventKind.USER_INPUT, e => userInputEvents.push(e));
724
+ const processingEndEvents = [];
725
+ session.on(EventKind.PROCESSING_END, e => processingEndEvents.push(e));
726
+ await session.processInput('initial message');
727
+ // LLM should have been called twice
728
+ expect(complete).toHaveBeenCalledTimes(2);
729
+ // USER_INPUT emitted twice (once per processInput call)
730
+ expect(userInputEvents).toHaveLength(2);
731
+ expect(userInputEvents[0].data.content).toBe('initial message');
732
+ expect(userInputEvents[1].data.content).toBe('follow-up message');
733
+ // PROCESSING_END emitted exactly once (after follow-up is fully exhausted)
734
+ expect(processingEndEvents).toHaveLength(1);
735
+ });
736
+ it('PROCESSING_END is NOT emitted until all follow-ups are exhausted', async () => {
737
+ const complete = vi
738
+ .fn()
739
+ .mockResolvedValueOnce(makeTextResponse('R1'))
740
+ .mockResolvedValueOnce(makeTextResponse('R2'))
741
+ .mockResolvedValueOnce(makeTextResponse('R3'));
742
+ const session = createSession({
743
+ llmClient: makeMockClient(complete),
744
+ providerProfile: makeProfile(),
745
+ executionEnv: makeEnv(),
746
+ });
747
+ session.follow_up('follow-up-1');
748
+ session.follow_up('follow-up-2');
749
+ const processingEndEvents = [];
750
+ session.on(EventKind.PROCESSING_END, e => processingEndEvents.push(e));
751
+ await session.processInput('initial');
752
+ expect(complete).toHaveBeenCalledTimes(3);
753
+ expect(processingEndEvents).toHaveLength(1);
754
+ });
755
+ it('follow-up messages are processed FIFO', async () => {
756
+ const complete = vi.fn()
757
+ .mockResolvedValueOnce(makeTextResponse('R1'))
758
+ .mockResolvedValueOnce(makeTextResponse('R2'))
759
+ .mockResolvedValueOnce(makeTextResponse('R3'));
760
+ const session = createSession({
761
+ llmClient: makeMockClient(complete),
762
+ providerProfile: makeProfile(),
763
+ executionEnv: makeEnv(),
764
+ });
765
+ session.follow_up('fu-first');
766
+ session.follow_up('fu-second');
767
+ const inputEvents = [];
768
+ session.on(EventKind.USER_INPUT, e => inputEvents.push(e));
769
+ await session.processInput('initial');
770
+ expect(inputEvents).toHaveLength(3);
771
+ expect(inputEvents[0].data.content).toBe('initial');
772
+ expect(inputEvents[1].data.content).toBe('fu-first');
773
+ expect(inputEvents[2].data.content).toBe('fu-second');
774
+ });
775
+ });
776
+ // ---------------------------------------------------------------------------
777
+ // Test Suite: Loop detection integration (Story 48-8)
778
+ // ---------------------------------------------------------------------------
779
+ describe('loop detection integration', () => {
780
+ it('LOOP_DETECTION event is emitted when the same tool is called 10+ times', async () => {
781
+ const tool = makeToolDefinition('looping_tool');
782
+ // LLM returns same tool call many times then completes
783
+ const complete = vi.fn();
784
+ // Return 'looping_tool' call 10 times, then text
785
+ for (let i = 0; i < 10; i++) {
786
+ complete.mockResolvedValueOnce(makeToolCallResponse('looping_tool', `call${i}`, {}));
787
+ }
788
+ complete.mockResolvedValueOnce(makeTextResponse('Done'));
789
+ const session = createSession({
790
+ llmClient: makeMockClient(complete),
791
+ providerProfile: makeProfile([tool]),
792
+ executionEnv: makeEnv(),
793
+ config: makeConfig({ enable_loop_detection: true, loop_detection_window: 10 }),
794
+ });
795
+ const loopEvents = [];
796
+ session.on(EventKind.LOOP_DETECTION, e => loopEvents.push(e));
797
+ await session.processInput('run loop');
798
+ expect(loopEvents).toHaveLength(1);
799
+ expect(loopEvents[0].data.message).toContain('Loop detected');
800
+ expect(loopEvents[0].data.message).toContain('10');
801
+ expect(loopEvents[0].data.message).toContain('repeating pattern');
802
+ });
803
+ it('LOOP_DETECTION injects SteeringTurn directly into history', async () => {
804
+ const tool = makeToolDefinition('looping_tool');
805
+ const complete = vi.fn();
806
+ for (let i = 0; i < 10; i++) {
807
+ complete.mockResolvedValueOnce(makeToolCallResponse('looping_tool', `call${i}`, {}));
808
+ }
809
+ complete.mockResolvedValueOnce(makeTextResponse('Done'));
810
+ const session = createSession({
811
+ llmClient: makeMockClient(complete),
812
+ providerProfile: makeProfile([tool]),
813
+ executionEnv: makeEnv(),
814
+ config: makeConfig({ enable_loop_detection: true, loop_detection_window: 10 }),
815
+ });
816
+ await session.processInput('run');
817
+ const steeringTurns = session.history.filter(t => t.type === 'steering');
818
+ const loopWarning = steeringTurns.find(t => t.type === 'steering' && t.content.includes('Loop detected'));
819
+ expect(loopWarning).toBeDefined();
820
+ if (loopWarning?.type === 'steering') {
821
+ expect(loopWarning.content).toContain('repeating pattern');
822
+ }
823
+ });
824
+ it('enable_loop_detection: false suppresses LOOP_DETECTION event even with repeating pattern', async () => {
825
+ const tool = makeToolDefinition('looping_tool');
826
+ const complete = vi.fn();
827
+ for (let i = 0; i < 10; i++) {
828
+ complete.mockResolvedValueOnce(makeToolCallResponse('looping_tool', `call${i}`, {}));
829
+ }
830
+ complete.mockResolvedValueOnce(makeTextResponse('Done'));
831
+ const session = createSession({
832
+ llmClient: makeMockClient(complete),
833
+ providerProfile: makeProfile([tool]),
834
+ executionEnv: makeEnv(),
835
+ config: makeConfig({
836
+ enable_loop_detection: false,
837
+ loop_detection_window: 10,
838
+ max_tool_rounds_per_input: 12, // allow enough rounds
839
+ }),
840
+ });
841
+ const loopEvents = [];
842
+ session.on(EventKind.LOOP_DETECTION, e => loopEvents.push(e));
843
+ await session.processInput('run no detection');
844
+ expect(loopEvents).toHaveLength(0);
845
+ });
846
+ it('loop detection warning message has correct format', async () => {
847
+ const tool = makeToolDefinition('repeat_tool');
848
+ const windowSize = 6;
849
+ const complete = vi.fn();
850
+ for (let i = 0; i < windowSize; i++) {
851
+ complete.mockResolvedValueOnce(makeToolCallResponse('repeat_tool', `c${i}`, {}));
852
+ }
853
+ complete.mockResolvedValueOnce(makeTextResponse('Done'));
854
+ const session = createSession({
855
+ llmClient: makeMockClient(complete),
856
+ providerProfile: makeProfile([tool]),
857
+ executionEnv: makeEnv(),
858
+ config: makeConfig({ enable_loop_detection: true, loop_detection_window: windowSize }),
859
+ });
860
+ const loopEvents = [];
861
+ session.on(EventKind.LOOP_DETECTION, e => loopEvents.push(e));
862
+ await session.processInput('run');
863
+ expect(loopEvents).toHaveLength(1);
864
+ const expectedMsg = `Loop detected: the last ${windowSize} tool calls follow a repeating pattern. Try a different approach.`;
865
+ expect(loopEvents[0].data.message).toBe(expectedMsg);
866
+ });
867
+ });
868
+ //# sourceMappingURL=loop.test.js.map