@roj-ai/sdk 0.0.2

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 (763) hide show
  1. package/dist/bootstrap.d.ts +453 -0
  2. package/dist/bootstrap.d.ts.map +1 -0
  3. package/dist/builtin-events.d.ts +16 -0
  4. package/dist/builtin-events.d.ts.map +1 -0
  5. package/dist/bun-platform/fs.d.ts +9 -0
  6. package/dist/bun-platform/fs.d.ts.map +1 -0
  7. package/dist/bun-platform/index.d.ts +12 -0
  8. package/dist/bun-platform/index.d.ts.map +1 -0
  9. package/dist/bun-platform/process.d.ts +6 -0
  10. package/dist/bun-platform/process.d.ts.map +1 -0
  11. package/dist/config.d.ts +38 -0
  12. package/dist/config.d.ts.map +1 -0
  13. package/dist/config.test.d.ts +2 -0
  14. package/dist/config.test.d.ts.map +1 -0
  15. package/dist/core/agent-loop.integration.test.d.ts +2 -0
  16. package/dist/core/agent-loop.integration.test.d.ts.map +1 -0
  17. package/dist/core/agents/agent-config.test.d.ts +2 -0
  18. package/dist/core/agents/agent-config.test.d.ts.map +1 -0
  19. package/dist/core/agents/agent-roles.d.ts +33 -0
  20. package/dist/core/agents/agent-roles.d.ts.map +1 -0
  21. package/dist/core/agents/agent-shutdown.test.d.ts +6 -0
  22. package/dist/core/agents/agent-shutdown.test.d.ts.map +1 -0
  23. package/dist/core/agents/agent.d.ts +234 -0
  24. package/dist/core/agents/agent.d.ts.map +1 -0
  25. package/dist/core/agents/agent.test.d.ts +2 -0
  26. package/dist/core/agents/agent.test.d.ts.map +1 -0
  27. package/dist/core/agents/communicator.d.ts +3 -0
  28. package/dist/core/agents/communicator.d.ts.map +1 -0
  29. package/dist/core/agents/config.d.ts +84 -0
  30. package/dist/core/agents/config.d.ts.map +1 -0
  31. package/dist/core/agents/context.d.ts +16 -0
  32. package/dist/core/agents/context.d.ts.map +1 -0
  33. package/dist/core/agents/debounce.d.ts +52 -0
  34. package/dist/core/agents/debounce.d.ts.map +1 -0
  35. package/dist/core/agents/handler-events.test.d.ts +5 -0
  36. package/dist/core/agents/handler-events.test.d.ts.map +1 -0
  37. package/dist/core/agents/index.d.ts +2 -0
  38. package/dist/core/agents/index.d.ts.map +1 -0
  39. package/dist/core/agents/response-sanitizer.d.ts +25 -0
  40. package/dist/core/agents/response-sanitizer.d.ts.map +1 -0
  41. package/dist/core/agents/response-sanitizer.test.d.ts +2 -0
  42. package/dist/core/agents/response-sanitizer.test.d.ts.map +1 -0
  43. package/dist/core/agents/retry.d.ts +40 -0
  44. package/dist/core/agents/retry.d.ts.map +1 -0
  45. package/dist/core/agents/schema.d.ts +57 -0
  46. package/dist/core/agents/schema.d.ts.map +1 -0
  47. package/dist/core/agents/state.d.ts +255 -0
  48. package/dist/core/agents/state.d.ts.map +1 -0
  49. package/dist/core/context/state.d.ts +28 -0
  50. package/dist/core/context/state.d.ts.map +1 -0
  51. package/dist/core/errors.d.ts +29 -0
  52. package/dist/core/errors.d.ts.map +1 -0
  53. package/dist/core/event-sourcing.integration.test.d.ts +2 -0
  54. package/dist/core/event-sourcing.integration.test.d.ts.map +1 -0
  55. package/dist/core/events/base-event-store.d.ts +55 -0
  56. package/dist/core/events/base-event-store.d.ts.map +1 -0
  57. package/dist/core/events/event-store.d.ts +108 -0
  58. package/dist/core/events/event-store.d.ts.map +1 -0
  59. package/dist/core/events/file.d.ts +50 -0
  60. package/dist/core/events/file.d.ts.map +1 -0
  61. package/dist/core/events/file.test.d.ts +2 -0
  62. package/dist/core/events/file.test.d.ts.map +1 -0
  63. package/dist/core/events/index.d.ts +4 -0
  64. package/dist/core/events/index.d.ts.map +1 -0
  65. package/dist/core/events/memory.d.ts +43 -0
  66. package/dist/core/events/memory.d.ts.map +1 -0
  67. package/dist/core/events/memory.test.d.ts +2 -0
  68. package/dist/core/events/memory.test.d.ts.map +1 -0
  69. package/dist/core/events/metadata-utils.d.ts +27 -0
  70. package/dist/core/events/metadata-utils.d.ts.map +1 -0
  71. package/dist/core/events/test-helpers.d.ts +28 -0
  72. package/dist/core/events/test-helpers.d.ts.map +1 -0
  73. package/dist/core/events/types.d.ts +46 -0
  74. package/dist/core/events/types.d.ts.map +1 -0
  75. package/dist/core/file-store/file-store.d.ts +56 -0
  76. package/dist/core/file-store/file-store.d.ts.map +1 -0
  77. package/dist/core/file-store/types.d.ts +50 -0
  78. package/dist/core/file-store/types.d.ts.map +1 -0
  79. package/dist/core/image/image-processor.d.ts +15 -0
  80. package/dist/core/image/image-processor.d.ts.map +1 -0
  81. package/dist/core/image/image-processor.test.d.ts +2 -0
  82. package/dist/core/image/image-processor.test.d.ts.map +1 -0
  83. package/dist/core/image/index.d.ts +5 -0
  84. package/dist/core/image/index.d.ts.map +1 -0
  85. package/dist/core/image/noop-resizer.d.ts +5 -0
  86. package/dist/core/image/noop-resizer.d.ts.map +1 -0
  87. package/dist/core/image/types.d.ts +20 -0
  88. package/dist/core/image/types.d.ts.map +1 -0
  89. package/dist/core/image/vips-resizer.d.ts +21 -0
  90. package/dist/core/image/vips-resizer.d.ts.map +1 -0
  91. package/dist/core/image/vips-resizer.test.d.ts +2 -0
  92. package/dist/core/image/vips-resizer.test.d.ts.map +1 -0
  93. package/dist/core/llm/anthropic.d.ts +51 -0
  94. package/dist/core/llm/anthropic.d.ts.map +1 -0
  95. package/dist/core/llm/anthropic.test.d.ts +2 -0
  96. package/dist/core/llm/anthropic.test.d.ts.map +1 -0
  97. package/dist/core/llm/cache-breakpoints.d.ts +17 -0
  98. package/dist/core/llm/cache-breakpoints.d.ts.map +1 -0
  99. package/dist/core/llm/cache-live.test.d.ts +16 -0
  100. package/dist/core/llm/cache-live.test.d.ts.map +1 -0
  101. package/dist/core/llm/index.d.ts +16 -0
  102. package/dist/core/llm/index.d.ts.map +1 -0
  103. package/dist/core/llm/llm-log-types.d.ts +148 -0
  104. package/dist/core/llm/llm-log-types.d.ts.map +1 -0
  105. package/dist/core/llm/logger.d.ts +74 -0
  106. package/dist/core/llm/logger.d.ts.map +1 -0
  107. package/dist/core/llm/logger.test.d.ts +7 -0
  108. package/dist/core/llm/logger.test.d.ts.map +1 -0
  109. package/dist/core/llm/logging-provider.d.ts +20 -0
  110. package/dist/core/llm/logging-provider.d.ts.map +1 -0
  111. package/dist/core/llm/middleware.d.ts +79 -0
  112. package/dist/core/llm/middleware.d.ts.map +1 -0
  113. package/dist/core/llm/mock.d.ts +79 -0
  114. package/dist/core/llm/mock.d.ts.map +1 -0
  115. package/dist/core/llm/mock.test.d.ts +2 -0
  116. package/dist/core/llm/mock.test.d.ts.map +1 -0
  117. package/dist/core/llm/openrouter-mapping.test.d.ts +2 -0
  118. package/dist/core/llm/openrouter-mapping.test.d.ts.map +1 -0
  119. package/dist/core/llm/openrouter.d.ts +37 -0
  120. package/dist/core/llm/openrouter.d.ts.map +1 -0
  121. package/dist/core/llm/openrouter.test.d.ts +2 -0
  122. package/dist/core/llm/openrouter.test.d.ts.map +1 -0
  123. package/dist/core/llm/provider-integration.test.d.ts +12 -0
  124. package/dist/core/llm/provider-integration.test.d.ts.map +1 -0
  125. package/dist/core/llm/provider.d.ts +175 -0
  126. package/dist/core/llm/provider.d.ts.map +1 -0
  127. package/dist/core/llm/routing-provider.d.ts +31 -0
  128. package/dist/core/llm/routing-provider.d.ts.map +1 -0
  129. package/dist/core/llm/routing-provider.test.d.ts +2 -0
  130. package/dist/core/llm/routing-provider.test.d.ts.map +1 -0
  131. package/dist/core/llm/schema.d.ts +24 -0
  132. package/dist/core/llm/schema.d.ts.map +1 -0
  133. package/dist/core/llm/snapshot-fetch.d.ts +21 -0
  134. package/dist/core/llm/snapshot-fetch.d.ts.map +1 -0
  135. package/dist/core/llm/snapshot-middleware.d.ts +71 -0
  136. package/dist/core/llm/snapshot-middleware.d.ts.map +1 -0
  137. package/dist/core/llm/snapshot-middleware.test.d.ts +2 -0
  138. package/dist/core/llm/snapshot-middleware.test.d.ts.map +1 -0
  139. package/dist/core/llm/state.d.ts +73 -0
  140. package/dist/core/llm/state.d.ts.map +1 -0
  141. package/dist/core/llm/tokens.d.ts +36 -0
  142. package/dist/core/llm/tokens.d.ts.map +1 -0
  143. package/dist/core/multi-agent.integration.test.d.ts +2 -0
  144. package/dist/core/multi-agent.integration.test.d.ts.map +1 -0
  145. package/dist/core/plugin-hooks.integration.test.d.ts +2 -0
  146. package/dist/core/plugin-hooks.integration.test.d.ts.map +1 -0
  147. package/dist/core/plugins/hook-types.d.ts +55 -0
  148. package/dist/core/plugins/hook-types.d.ts.map +1 -0
  149. package/dist/core/plugins/index.d.ts +23 -0
  150. package/dist/core/plugins/index.d.ts.map +1 -0
  151. package/dist/core/plugins/plugin-builder.d.ts +474 -0
  152. package/dist/core/plugins/plugin-builder.d.ts.map +1 -0
  153. package/dist/core/preset/config.d.ts +55 -0
  154. package/dist/core/preset/config.d.ts.map +1 -0
  155. package/dist/core/preset/index.d.ts +8 -0
  156. package/dist/core/preset/index.d.ts.map +1 -0
  157. package/dist/core/preset/preset-builder.d.ts +44 -0
  158. package/dist/core/preset/preset-builder.d.ts.map +1 -0
  159. package/dist/core/session-lifecycle.integration.test.d.ts +2 -0
  160. package/dist/core/session-lifecycle.integration.test.d.ts.map +1 -0
  161. package/dist/core/sessions/apply-event.d.ts +19 -0
  162. package/dist/core/sessions/apply-event.d.ts.map +1 -0
  163. package/dist/core/sessions/context.d.ts +34 -0
  164. package/dist/core/sessions/context.d.ts.map +1 -0
  165. package/dist/core/sessions/fork-utils.d.ts +20 -0
  166. package/dist/core/sessions/fork-utils.d.ts.map +1 -0
  167. package/dist/core/sessions/fork-utils.test.d.ts +2 -0
  168. package/dist/core/sessions/fork-utils.test.d.ts.map +1 -0
  169. package/dist/core/sessions/reducer.d.ts +50 -0
  170. package/dist/core/sessions/reducer.d.ts.map +1 -0
  171. package/dist/core/sessions/schema.d.ts +82 -0
  172. package/dist/core/sessions/schema.d.ts.map +1 -0
  173. package/dist/core/sessions/session-environment.d.ts +13 -0
  174. package/dist/core/sessions/session-environment.d.ts.map +1 -0
  175. package/dist/core/sessions/session-manager.d.ts +183 -0
  176. package/dist/core/sessions/session-manager.d.ts.map +1 -0
  177. package/dist/core/sessions/session-store.d.ts +69 -0
  178. package/dist/core/sessions/session-store.d.ts.map +1 -0
  179. package/dist/core/sessions/session.d.ts +212 -0
  180. package/dist/core/sessions/session.d.ts.map +1 -0
  181. package/dist/core/sessions/session.test.d.ts +2 -0
  182. package/dist/core/sessions/session.test.d.ts.map +1 -0
  183. package/dist/core/sessions/state.d.ts +110 -0
  184. package/dist/core/sessions/state.d.ts.map +1 -0
  185. package/dist/core/system.d.ts +97 -0
  186. package/dist/core/system.d.ts.map +1 -0
  187. package/dist/core/tools/context.d.ts +3 -0
  188. package/dist/core/tools/context.d.ts.map +1 -0
  189. package/dist/core/tools/definition.d.ts +10 -0
  190. package/dist/core/tools/definition.d.ts.map +1 -0
  191. package/dist/core/tools/executor.d.ts +28 -0
  192. package/dist/core/tools/executor.d.ts.map +1 -0
  193. package/dist/core/tools/executor.test.d.ts +2 -0
  194. package/dist/core/tools/executor.test.d.ts.map +1 -0
  195. package/dist/core/tools/index.d.ts +4 -0
  196. package/dist/core/tools/index.d.ts.map +1 -0
  197. package/dist/core/tools/schema.d.ts +61 -0
  198. package/dist/core/tools/schema.d.ts.map +1 -0
  199. package/dist/core/tools/state.d.ts +28 -0
  200. package/dist/core/tools/state.d.ts.map +1 -0
  201. package/dist/index.d.ts +111 -0
  202. package/dist/index.d.ts.map +1 -0
  203. package/dist/lib/json/index.d.ts +17 -0
  204. package/dist/lib/json/index.d.ts.map +1 -0
  205. package/dist/lib/logger/console.d.ts +35 -0
  206. package/dist/lib/logger/console.d.ts.map +1 -0
  207. package/dist/lib/logger/console.test.d.ts +2 -0
  208. package/dist/lib/logger/console.test.d.ts.map +1 -0
  209. package/dist/lib/logger/file.d.ts +20 -0
  210. package/dist/lib/logger/file.d.ts.map +1 -0
  211. package/dist/lib/logger/index.d.ts +5 -0
  212. package/dist/lib/logger/index.d.ts.map +1 -0
  213. package/dist/lib/logger/logger.d.ts +87 -0
  214. package/dist/lib/logger/logger.d.ts.map +1 -0
  215. package/dist/lib/logger/ring-buffer.d.ts +33 -0
  216. package/dist/lib/logger/ring-buffer.d.ts.map +1 -0
  217. package/dist/lib/logger/tee.d.ts +15 -0
  218. package/dist/lib/logger/tee.d.ts.map +1 -0
  219. package/dist/lib/mime.d.ts +9 -0
  220. package/dist/lib/mime.d.ts.map +1 -0
  221. package/dist/lib/never.d.ts +2 -0
  222. package/dist/lib/never.d.ts.map +1 -0
  223. package/dist/lib/utils/hash.d.ts +19 -0
  224. package/dist/lib/utils/hash.d.ts.map +1 -0
  225. package/dist/lib/utils/result.d.ts +26 -0
  226. package/dist/lib/utils/result.d.ts.map +1 -0
  227. package/dist/platform/fs.d.ts +39 -0
  228. package/dist/platform/fs.d.ts.map +1 -0
  229. package/dist/platform/index.d.ts +21 -0
  230. package/dist/platform/index.d.ts.map +1 -0
  231. package/dist/platform/process.d.ts +24 -0
  232. package/dist/platform/process.d.ts.map +1 -0
  233. package/dist/plugins/agent-status/plugin.d.ts +8 -0
  234. package/dist/plugins/agent-status/plugin.d.ts.map +1 -0
  235. package/dist/plugins/agents/agents.integration.test.d.ts +2 -0
  236. package/dist/plugins/agents/agents.integration.test.d.ts.map +1 -0
  237. package/dist/plugins/agents/index.d.ts +3 -0
  238. package/dist/plugins/agents/index.d.ts.map +1 -0
  239. package/dist/plugins/agents/plugin.d.ts +57 -0
  240. package/dist/plugins/agents/plugin.d.ts.map +1 -0
  241. package/dist/plugins/context-compact/context-compact.integration.test.d.ts +2 -0
  242. package/dist/plugins/context-compact/context-compact.integration.test.d.ts.map +1 -0
  243. package/dist/plugins/context-compact/context-compactor.d.ts +88 -0
  244. package/dist/plugins/context-compact/context-compactor.d.ts.map +1 -0
  245. package/dist/plugins/context-compact/context-compactor.test.d.ts +2 -0
  246. package/dist/plugins/context-compact/context-compactor.test.d.ts.map +1 -0
  247. package/dist/plugins/context-compact/history-offloader.d.ts +17 -0
  248. package/dist/plugins/context-compact/history-offloader.d.ts.map +1 -0
  249. package/dist/plugins/context-compact/history-offloader.test.d.ts +2 -0
  250. package/dist/plugins/context-compact/history-offloader.test.d.ts.map +1 -0
  251. package/dist/plugins/context-compact/index.d.ts +4 -0
  252. package/dist/plugins/context-compact/index.d.ts.map +1 -0
  253. package/dist/plugins/context-compact/plugin.d.ts +15 -0
  254. package/dist/plugins/context-compact/plugin.d.ts.map +1 -0
  255. package/dist/plugins/filesystem/filesystem.integration.test.d.ts +2 -0
  256. package/dist/plugins/filesystem/filesystem.integration.test.d.ts.map +1 -0
  257. package/dist/plugins/filesystem/helpers.d.ts +34 -0
  258. package/dist/plugins/filesystem/helpers.d.ts.map +1 -0
  259. package/dist/plugins/filesystem/index.d.ts +5 -0
  260. package/dist/plugins/filesystem/index.d.ts.map +1 -0
  261. package/dist/plugins/filesystem/listing.d.ts +38 -0
  262. package/dist/plugins/filesystem/listing.d.ts.map +1 -0
  263. package/dist/plugins/filesystem/plugin.d.ts +53 -0
  264. package/dist/plugins/filesystem/plugin.d.ts.map +1 -0
  265. package/dist/plugins/filesystem/schema.d.ts +7 -0
  266. package/dist/plugins/filesystem/schema.d.ts.map +1 -0
  267. package/dist/plugins/git-status/index.d.ts +2 -0
  268. package/dist/plugins/git-status/index.d.ts.map +1 -0
  269. package/dist/plugins/git-status/plugin.d.ts +12 -0
  270. package/dist/plugins/git-status/plugin.d.ts.map +1 -0
  271. package/dist/plugins/limits-guard/config.d.ts +23 -0
  272. package/dist/plugins/limits-guard/config.d.ts.map +1 -0
  273. package/dist/plugins/limits-guard/index.d.ts +5 -0
  274. package/dist/plugins/limits-guard/index.d.ts.map +1 -0
  275. package/dist/plugins/limits-guard/limit-guard.d.ts +40 -0
  276. package/dist/plugins/limits-guard/limit-guard.d.ts.map +1 -0
  277. package/dist/plugins/limits-guard/limit-guard.test.d.ts +2 -0
  278. package/dist/plugins/limits-guard/limit-guard.test.d.ts.map +1 -0
  279. package/dist/plugins/limits-guard/limits-guard.integration.test.d.ts +2 -0
  280. package/dist/plugins/limits-guard/limits-guard.integration.test.d.ts.map +1 -0
  281. package/dist/plugins/limits-guard/plugin.d.ts +39 -0
  282. package/dist/plugins/limits-guard/plugin.d.ts.map +1 -0
  283. package/dist/plugins/llm-debug/index.d.ts +2 -0
  284. package/dist/plugins/llm-debug/index.d.ts.map +1 -0
  285. package/dist/plugins/llm-debug/llm-debug.integration.test.d.ts +2 -0
  286. package/dist/plugins/llm-debug/llm-debug.integration.test.d.ts.map +1 -0
  287. package/dist/plugins/llm-debug/plugin.d.ts +31 -0
  288. package/dist/plugins/llm-debug/plugin.d.ts.map +1 -0
  289. package/dist/plugins/logs/index.d.ts +2 -0
  290. package/dist/plugins/logs/index.d.ts.map +1 -0
  291. package/dist/plugins/logs/plugin.d.ts +10 -0
  292. package/dist/plugins/logs/plugin.d.ts.map +1 -0
  293. package/dist/plugins/mailbox/helpers.d.ts +20 -0
  294. package/dist/plugins/mailbox/helpers.d.ts.map +1 -0
  295. package/dist/plugins/mailbox/index.d.ts +9 -0
  296. package/dist/plugins/mailbox/index.d.ts.map +1 -0
  297. package/dist/plugins/mailbox/mailbox.integration.test.d.ts +2 -0
  298. package/dist/plugins/mailbox/mailbox.integration.test.d.ts.map +1 -0
  299. package/dist/plugins/mailbox/plugin.d.ts +31 -0
  300. package/dist/plugins/mailbox/plugin.d.ts.map +1 -0
  301. package/dist/plugins/mailbox/prompts.d.ts +21 -0
  302. package/dist/plugins/mailbox/prompts.d.ts.map +1 -0
  303. package/dist/plugins/mailbox/query.d.ts +33 -0
  304. package/dist/plugins/mailbox/query.d.ts.map +1 -0
  305. package/dist/plugins/mailbox/schema.d.ts +54 -0
  306. package/dist/plugins/mailbox/schema.d.ts.map +1 -0
  307. package/dist/plugins/mailbox/state.d.ts +40 -0
  308. package/dist/plugins/mailbox/state.d.ts.map +1 -0
  309. package/dist/plugins/resources/index.d.ts +7 -0
  310. package/dist/plugins/resources/index.d.ts.map +1 -0
  311. package/dist/plugins/resources/manifest.d.ts +23 -0
  312. package/dist/plugins/resources/manifest.d.ts.map +1 -0
  313. package/dist/plugins/resources/plugin.d.ts +28 -0
  314. package/dist/plugins/resources/plugin.d.ts.map +1 -0
  315. package/dist/plugins/resources/post-inject.d.ts +39 -0
  316. package/dist/plugins/resources/post-inject.d.ts.map +1 -0
  317. package/dist/plugins/resources/state.d.ts +25 -0
  318. package/dist/plugins/resources/state.d.ts.map +1 -0
  319. package/dist/plugins/result-eviction/index.d.ts +3 -0
  320. package/dist/plugins/result-eviction/index.d.ts.map +1 -0
  321. package/dist/plugins/result-eviction/plugin.d.ts +19 -0
  322. package/dist/plugins/result-eviction/plugin.d.ts.map +1 -0
  323. package/dist/plugins/result-eviction/result-eviction.integration.test.d.ts +2 -0
  324. package/dist/plugins/result-eviction/result-eviction.integration.test.d.ts.map +1 -0
  325. package/dist/plugins/services/plugin.d.ts +85 -0
  326. package/dist/plugins/services/plugin.d.ts.map +1 -0
  327. package/dist/plugins/services/port-pool.d.ts +32 -0
  328. package/dist/plugins/services/port-pool.d.ts.map +1 -0
  329. package/dist/plugins/services/prompt.d.ts +13 -0
  330. package/dist/plugins/services/prompt.d.ts.map +1 -0
  331. package/dist/plugins/services/schema.d.ts +70 -0
  332. package/dist/plugins/services/schema.d.ts.map +1 -0
  333. package/dist/plugins/services/service.d.ts +86 -0
  334. package/dist/plugins/services/service.d.ts.map +1 -0
  335. package/dist/plugins/services/services.integration.test.d.ts +2 -0
  336. package/dist/plugins/services/services.integration.test.d.ts.map +1 -0
  337. package/dist/plugins/session-lifecycle/index.d.ts +2 -0
  338. package/dist/plugins/session-lifecycle/index.d.ts.map +1 -0
  339. package/dist/plugins/session-lifecycle/plugin.d.ts +97 -0
  340. package/dist/plugins/session-lifecycle/plugin.d.ts.map +1 -0
  341. package/dist/plugins/session-lifecycle/session-lifecycle.integration.test.d.ts +2 -0
  342. package/dist/plugins/session-lifecycle/session-lifecycle.integration.test.d.ts.map +1 -0
  343. package/dist/plugins/session-state/plugin.d.ts +48 -0
  344. package/dist/plugins/session-state/plugin.d.ts.map +1 -0
  345. package/dist/plugins/session-stats/index.d.ts +4 -0
  346. package/dist/plugins/session-stats/index.d.ts.map +1 -0
  347. package/dist/plugins/session-stats/plugin.d.ts +29 -0
  348. package/dist/plugins/session-stats/plugin.d.ts.map +1 -0
  349. package/dist/plugins/shell/executor.d.ts +78 -0
  350. package/dist/plugins/shell/executor.d.ts.map +1 -0
  351. package/dist/plugins/shell/index.d.ts +6 -0
  352. package/dist/plugins/shell/index.d.ts.map +1 -0
  353. package/dist/plugins/shell/plugin.d.ts +51 -0
  354. package/dist/plugins/shell/plugin.d.ts.map +1 -0
  355. package/dist/plugins/shell/shell.integration.test.d.ts +2 -0
  356. package/dist/plugins/shell/shell.integration.test.d.ts.map +1 -0
  357. package/dist/plugins/shell/shell.test.d.ts +2 -0
  358. package/dist/plugins/shell/shell.test.d.ts.map +1 -0
  359. package/dist/plugins/skills/discovery.d.ts +69 -0
  360. package/dist/plugins/skills/discovery.d.ts.map +1 -0
  361. package/dist/plugins/skills/discovery.test.d.ts +2 -0
  362. package/dist/plugins/skills/discovery.test.d.ts.map +1 -0
  363. package/dist/plugins/skills/index.d.ts +11 -0
  364. package/dist/plugins/skills/index.d.ts.map +1 -0
  365. package/dist/plugins/skills/plugin.d.ts +94 -0
  366. package/dist/plugins/skills/plugin.d.ts.map +1 -0
  367. package/dist/plugins/skills/prompts.d.ts +29 -0
  368. package/dist/plugins/skills/prompts.d.ts.map +1 -0
  369. package/dist/plugins/skills/schema.d.ts +64 -0
  370. package/dist/plugins/skills/schema.d.ts.map +1 -0
  371. package/dist/plugins/skills/skills.integration.test.d.ts +2 -0
  372. package/dist/plugins/skills/skills.integration.test.d.ts.map +1 -0
  373. package/dist/plugins/snapshotting/index.d.ts +4 -0
  374. package/dist/plugins/snapshotting/index.d.ts.map +1 -0
  375. package/dist/plugins/snapshotting/jj-snapshotter.d.ts +27 -0
  376. package/dist/plugins/snapshotting/jj-snapshotter.d.ts.map +1 -0
  377. package/dist/plugins/snapshotting/plugin.d.ts +15 -0
  378. package/dist/plugins/snapshotting/plugin.d.ts.map +1 -0
  379. package/dist/plugins/snapshotting/snapshotter.d.ts +19 -0
  380. package/dist/plugins/snapshotting/snapshotter.d.ts.map +1 -0
  381. package/dist/plugins/todo/index.d.ts +7 -0
  382. package/dist/plugins/todo/index.d.ts.map +1 -0
  383. package/dist/plugins/todo/plugin.d.ts +95 -0
  384. package/dist/plugins/todo/plugin.d.ts.map +1 -0
  385. package/dist/plugins/todo/prompts.d.ts +13 -0
  386. package/dist/plugins/todo/prompts.d.ts.map +1 -0
  387. package/dist/plugins/todo/schema.d.ts +44 -0
  388. package/dist/plugins/todo/schema.d.ts.map +1 -0
  389. package/dist/plugins/todo/todo.integration.test.d.ts +2 -0
  390. package/dist/plugins/todo/todo.integration.test.d.ts.map +1 -0
  391. package/dist/plugins/uploads/index.d.ts +9 -0
  392. package/dist/plugins/uploads/index.d.ts.map +1 -0
  393. package/dist/plugins/uploads/plugin.d.ts +56 -0
  394. package/dist/plugins/uploads/plugin.d.ts.map +1 -0
  395. package/dist/plugins/uploads/preprocessor.d.ts +70 -0
  396. package/dist/plugins/uploads/preprocessor.d.ts.map +1 -0
  397. package/dist/plugins/uploads/preprocessors/image-classifier.d.ts +49 -0
  398. package/dist/plugins/uploads/preprocessors/image-classifier.d.ts.map +1 -0
  399. package/dist/plugins/uploads/preprocessors/index.d.ts +7 -0
  400. package/dist/plugins/uploads/preprocessors/index.d.ts.map +1 -0
  401. package/dist/plugins/uploads/preprocessors/markitdown-preprocessor.d.ts +43 -0
  402. package/dist/plugins/uploads/preprocessors/markitdown-preprocessor.d.ts.map +1 -0
  403. package/dist/plugins/uploads/preprocessors/zip-preprocessor.d.ts +30 -0
  404. package/dist/plugins/uploads/preprocessors/zip-preprocessor.d.ts.map +1 -0
  405. package/dist/plugins/uploads/schema.d.ts +72 -0
  406. package/dist/plugins/uploads/schema.d.ts.map +1 -0
  407. package/dist/plugins/uploads/state.d.ts +38 -0
  408. package/dist/plugins/uploads/state.d.ts.map +1 -0
  409. package/dist/plugins/uploads/uploads.integration.test.d.ts +2 -0
  410. package/dist/plugins/uploads/uploads.integration.test.d.ts.map +1 -0
  411. package/dist/plugins/user-chat/index.d.ts +7 -0
  412. package/dist/plugins/user-chat/index.d.ts.map +1 -0
  413. package/dist/plugins/user-chat/plugin.d.ts +221 -0
  414. package/dist/plugins/user-chat/plugin.d.ts.map +1 -0
  415. package/dist/plugins/user-chat/prompts.d.ts +13 -0
  416. package/dist/plugins/user-chat/prompts.d.ts.map +1 -0
  417. package/dist/plugins/user-chat/schema.d.ts +82 -0
  418. package/dist/plugins/user-chat/schema.d.ts.map +1 -0
  419. package/dist/plugins/user-chat/user-chat.integration.test.d.ts +2 -0
  420. package/dist/plugins/user-chat/user-chat.integration.test.d.ts.map +1 -0
  421. package/dist/plugins/workers/context.d.ts +159 -0
  422. package/dist/plugins/workers/context.d.ts.map +1 -0
  423. package/dist/plugins/workers/definition.d.ts +118 -0
  424. package/dist/plugins/workers/definition.d.ts.map +1 -0
  425. package/dist/plugins/workers/index.d.ts +11 -0
  426. package/dist/plugins/workers/index.d.ts.map +1 -0
  427. package/dist/plugins/workers/plugin.d.ts +125 -0
  428. package/dist/plugins/workers/plugin.d.ts.map +1 -0
  429. package/dist/plugins/workers/worker.d.ts +76 -0
  430. package/dist/plugins/workers/worker.d.ts.map +1 -0
  431. package/dist/plugins/workers/workers.integration.test.d.ts +2 -0
  432. package/dist/plugins/workers/workers.integration.test.d.ts.map +1 -0
  433. package/dist/prompts/base.d.ts +75 -0
  434. package/dist/prompts/base.d.ts.map +1 -0
  435. package/dist/prompts/builder.d.ts +73 -0
  436. package/dist/prompts/builder.d.ts.map +1 -0
  437. package/dist/prompts/index.d.ts +10 -0
  438. package/dist/prompts/index.d.ts.map +1 -0
  439. package/dist/prompts/macros.d.ts +15 -0
  440. package/dist/prompts/macros.d.ts.map +1 -0
  441. package/dist/prompts/macros.test.d.ts +2 -0
  442. package/dist/prompts/macros.test.d.ts.map +1 -0
  443. package/dist/testing/bootstrap-for-testing.d.ts +12 -0
  444. package/dist/testing/bootstrap-for-testing.d.ts.map +1 -0
  445. package/dist/testing/index.d.ts +8 -0
  446. package/dist/testing/index.d.ts.map +1 -0
  447. package/dist/testing/node-platform.d.ts +12 -0
  448. package/dist/testing/node-platform.d.ts.map +1 -0
  449. package/dist/testing/notification-collector.d.ts +46 -0
  450. package/dist/testing/notification-collector.d.ts.map +1 -0
  451. package/dist/testing/preset-helpers.d.ts +28 -0
  452. package/dist/testing/preset-helpers.d.ts.map +1 -0
  453. package/dist/testing/test-harness.d.ts +117 -0
  454. package/dist/testing/test-harness.d.ts.map +1 -0
  455. package/dist/testing/test-harness.test.d.ts +2 -0
  456. package/dist/testing/test-harness.test.d.ts.map +1 -0
  457. package/dist/testing/wait-helpers.d.ts +17 -0
  458. package/dist/testing/wait-helpers.d.ts.map +1 -0
  459. package/dist/transport/adapter/client-adapter.d.ts +32 -0
  460. package/dist/transport/adapter/client-adapter.d.ts.map +1 -0
  461. package/dist/transport/adapter/index.d.ts +23 -0
  462. package/dist/transport/adapter/index.d.ts.map +1 -0
  463. package/dist/transport/adapter/server-adapter.d.ts +29 -0
  464. package/dist/transport/adapter/server-adapter.d.ts.map +1 -0
  465. package/dist/transport/adapter/types.d.ts +14 -0
  466. package/dist/transport/adapter/types.d.ts.map +1 -0
  467. package/dist/transport/http/app.d.ts +41 -0
  468. package/dist/transport/http/app.d.ts.map +1 -0
  469. package/dist/transport/http/index.d.ts +7 -0
  470. package/dist/transport/http/index.d.ts.map +1 -0
  471. package/dist/transport/http/middleware/bearer-auth.d.ts +16 -0
  472. package/dist/transport/http/middleware/bearer-auth.d.ts.map +1 -0
  473. package/dist/transport/http/middleware/error-handler.d.ts +21 -0
  474. package/dist/transport/http/middleware/error-handler.d.ts.map +1 -0
  475. package/dist/transport/http/routes/files.d.ts +16 -0
  476. package/dist/transport/http/routes/files.d.ts.map +1 -0
  477. package/dist/transport/http/routes/resources.d.ts +12 -0
  478. package/dist/transport/http/routes/resources.d.ts.map +1 -0
  479. package/dist/transport/http/routes/rpc.d.ts +22 -0
  480. package/dist/transport/http/routes/rpc.d.ts.map +1 -0
  481. package/dist/transport/http/routes/rpc.integration.test.d.ts +2 -0
  482. package/dist/transport/http/routes/rpc.integration.test.d.ts.map +1 -0
  483. package/dist/transport/http/routes/rpc.test.d.ts +2 -0
  484. package/dist/transport/http/routes/rpc.test.d.ts.map +1 -0
  485. package/dist/transport/http/routes/upload.d.ts +14 -0
  486. package/dist/transport/http/routes/upload.d.ts.map +1 -0
  487. package/dist/transport/rpc/index.d.ts +8 -0
  488. package/dist/transport/rpc/index.d.ts.map +1 -0
  489. package/dist/transport/rpc/methods.d.ts +25 -0
  490. package/dist/transport/rpc/methods.d.ts.map +1 -0
  491. package/dist/user-config.d.ts +29 -0
  492. package/dist/user-config.d.ts.map +1 -0
  493. package/package.json +154 -0
  494. package/src/bootstrap.ts +268 -0
  495. package/src/builtin-events.ts +25 -0
  496. package/src/bun-platform/fs.ts +50 -0
  497. package/src/bun-platform/index.ts +21 -0
  498. package/src/bun-platform/process.ts +25 -0
  499. package/src/config.test.ts +174 -0
  500. package/src/config.ts +103 -0
  501. package/src/core/agent-loop.integration.test.ts +503 -0
  502. package/src/core/agents/agent-config.test.ts +240 -0
  503. package/src/core/agents/agent-roles.ts +41 -0
  504. package/src/core/agents/agent-shutdown.test.ts +236 -0
  505. package/src/core/agents/agent.test.ts +387 -0
  506. package/src/core/agents/agent.ts +1472 -0
  507. package/src/core/agents/communicator.ts +16 -0
  508. package/src/core/agents/config.ts +98 -0
  509. package/src/core/agents/context.ts +19 -0
  510. package/src/core/agents/debounce.ts +116 -0
  511. package/src/core/agents/handler-events.test.ts +167 -0
  512. package/src/core/agents/index.ts +1 -0
  513. package/src/core/agents/response-sanitizer.test.ts +137 -0
  514. package/src/core/agents/response-sanitizer.ts +67 -0
  515. package/src/core/agents/retry.ts +164 -0
  516. package/src/core/agents/schema.ts +75 -0
  517. package/src/core/agents/state.ts +272 -0
  518. package/src/core/context/state.ts +38 -0
  519. package/src/core/errors.ts +55 -0
  520. package/src/core/event-sourcing.integration.test.ts +191 -0
  521. package/src/core/events/base-event-store.ts +264 -0
  522. package/src/core/events/event-store.ts +143 -0
  523. package/src/core/events/file.test.ts +436 -0
  524. package/src/core/events/file.ts +372 -0
  525. package/src/core/events/index.ts +3 -0
  526. package/src/core/events/memory.test.ts +741 -0
  527. package/src/core/events/memory.ts +131 -0
  528. package/src/core/events/metadata-utils.ts +133 -0
  529. package/src/core/events/test-helpers.ts +31 -0
  530. package/src/core/events/types.ts +64 -0
  531. package/src/core/file-store/file-store.ts +275 -0
  532. package/src/core/file-store/types.ts +52 -0
  533. package/src/core/image/image-processor.test.ts +218 -0
  534. package/src/core/image/image-processor.ts +127 -0
  535. package/src/core/image/index.ts +4 -0
  536. package/src/core/image/noop-resizer.ts +7 -0
  537. package/src/core/image/types.ts +24 -0
  538. package/src/core/image/vips-resizer.test.ts +377 -0
  539. package/src/core/image/vips-resizer.ts +124 -0
  540. package/src/core/llm/__snapshots__/anthropic-assistant-text-with-tool-calls.json +156 -0
  541. package/src/core/llm/__snapshots__/anthropic-consecutive-tool-results.json +152 -0
  542. package/src/core/llm/__snapshots__/anthropic-image-data-url.json +105 -0
  543. package/src/core/llm/__snapshots__/anthropic-image-http-url.json +104 -0
  544. package/src/core/llm/__snapshots__/anthropic-multiple-images.json +113 -0
  545. package/src/core/llm/__snapshots__/anthropic-multiple-tool-calls-requested.json +151 -0
  546. package/src/core/llm/__snapshots__/anthropic-simple-user-message.json +97 -0
  547. package/src/core/llm/__snapshots__/anthropic-system-message-in-conversation.json +119 -0
  548. package/src/core/llm/__snapshots__/anthropic-tool-call.json +123 -0
  549. package/src/core/llm/__snapshots__/anthropic-tool-error-result.json +139 -0
  550. package/src/core/llm/__snapshots__/anthropic-tool-result-with-image.json +152 -0
  551. package/src/core/llm/__snapshots__/anthropic-tool-roundtrip.json +139 -0
  552. package/src/core/llm/__snapshots__/openrouter-assistant-text-with-tool-calls.json +150 -0
  553. package/src/core/llm/__snapshots__/openrouter-consecutive-tool-results.json +150 -0
  554. package/src/core/llm/__snapshots__/openrouter-image-data-url.json +107 -0
  555. package/src/core/llm/__snapshots__/openrouter-image-http-url.json +107 -0
  556. package/src/core/llm/__snapshots__/openrouter-multiple-images.json +113 -0
  557. package/src/core/llm/__snapshots__/openrouter-multiple-tool-calls-requested.json +158 -0
  558. package/src/core/llm/__snapshots__/openrouter-simple-user-message.json +96 -0
  559. package/src/core/llm/__snapshots__/openrouter-system-message-in-conversation.json +108 -0
  560. package/src/core/llm/__snapshots__/openrouter-tool-call.json +129 -0
  561. package/src/core/llm/__snapshots__/openrouter-tool-error-result.json +137 -0
  562. package/src/core/llm/__snapshots__/openrouter-tool-result-with-image.json +148 -0
  563. package/src/core/llm/__snapshots__/openrouter-tool-roundtrip.json +137 -0
  564. package/src/core/llm/anthropic.test.ts +509 -0
  565. package/src/core/llm/anthropic.ts +578 -0
  566. package/src/core/llm/cache-breakpoints.ts +38 -0
  567. package/src/core/llm/cache-live.test.ts +155 -0
  568. package/src/core/llm/index.ts +30 -0
  569. package/src/core/llm/llm-log-types.ts +167 -0
  570. package/src/core/llm/logger.test.ts +270 -0
  571. package/src/core/llm/logger.ts +306 -0
  572. package/src/core/llm/logging-provider.ts +73 -0
  573. package/src/core/llm/middleware.ts +172 -0
  574. package/src/core/llm/mock.test.ts +402 -0
  575. package/src/core/llm/mock.ts +234 -0
  576. package/src/core/llm/openrouter-mapping.test.ts +153 -0
  577. package/src/core/llm/openrouter.test.ts +499 -0
  578. package/src/core/llm/openrouter.ts +458 -0
  579. package/src/core/llm/provider-integration.test.ts +408 -0
  580. package/src/core/llm/provider.ts +238 -0
  581. package/src/core/llm/routing-provider.test.ts +113 -0
  582. package/src/core/llm/routing-provider.ts +86 -0
  583. package/src/core/llm/schema.ts +47 -0
  584. package/src/core/llm/snapshot-fetch.ts +158 -0
  585. package/src/core/llm/snapshot-middleware.test.ts +168 -0
  586. package/src/core/llm/snapshot-middleware.ts +185 -0
  587. package/src/core/llm/state.ts +92 -0
  588. package/src/core/llm/tokens.ts +61 -0
  589. package/src/core/multi-agent.integration.test.ts +340 -0
  590. package/src/core/plugin-hooks.integration.test.ts +428 -0
  591. package/src/core/plugins/hook-types.ts +49 -0
  592. package/src/core/plugins/index.ts +52 -0
  593. package/src/core/plugins/plugin-builder.ts +967 -0
  594. package/src/core/preset/config.ts +110 -0
  595. package/src/core/preset/index.ts +11 -0
  596. package/src/core/preset/preset-builder.ts +111 -0
  597. package/src/core/session-lifecycle.integration.test.ts +202 -0
  598. package/src/core/sessions/apply-event.ts +46 -0
  599. package/src/core/sessions/context.ts +36 -0
  600. package/src/core/sessions/fork-utils.test.ts +158 -0
  601. package/src/core/sessions/fork-utils.ts +71 -0
  602. package/src/core/sessions/reducer.ts +95 -0
  603. package/src/core/sessions/schema.ts +91 -0
  604. package/src/core/sessions/session-environment.ts +12 -0
  605. package/src/core/sessions/session-manager.ts +883 -0
  606. package/src/core/sessions/session-store.ts +141 -0
  607. package/src/core/sessions/session.test.ts +1730 -0
  608. package/src/core/sessions/session.ts +833 -0
  609. package/src/core/sessions/state.ts +520 -0
  610. package/src/core/system.ts +206 -0
  611. package/src/core/tools/context.ts +3 -0
  612. package/src/core/tools/definition.ts +15 -0
  613. package/src/core/tools/executor.test.ts +160 -0
  614. package/src/core/tools/executor.ts +117 -0
  615. package/src/core/tools/index.ts +3 -0
  616. package/src/core/tools/schema.ts +80 -0
  617. package/src/core/tools/state.ts +34 -0
  618. package/src/index.ts +165 -0
  619. package/src/lib/json/index.ts +20 -0
  620. package/src/lib/logger/console.test.ts +348 -0
  621. package/src/lib/logger/console.ts +185 -0
  622. package/src/lib/logger/file.ts +65 -0
  623. package/src/lib/logger/index.ts +4 -0
  624. package/src/lib/logger/logger.ts +114 -0
  625. package/src/lib/logger/ring-buffer.ts +65 -0
  626. package/src/lib/logger/tee.ts +51 -0
  627. package/src/lib/mime.ts +23 -0
  628. package/src/lib/never.ts +3 -0
  629. package/src/lib/utils/hash.ts +38 -0
  630. package/src/lib/utils/result.ts +35 -0
  631. package/src/platform/fs.ts +38 -0
  632. package/src/platform/index.ts +23 -0
  633. package/src/platform/process.ts +28 -0
  634. package/src/plugins/agent-status/plugin.ts +77 -0
  635. package/src/plugins/agents/agents.integration.test.ts +808 -0
  636. package/src/plugins/agents/index.ts +2 -0
  637. package/src/plugins/agents/plugin.ts +242 -0
  638. package/src/plugins/context-compact/context-compact.integration.test.ts +207 -0
  639. package/src/plugins/context-compact/context-compactor.test.ts +923 -0
  640. package/src/plugins/context-compact/context-compactor.ts +343 -0
  641. package/src/plugins/context-compact/history-offloader.test.ts +100 -0
  642. package/src/plugins/context-compact/history-offloader.ts +47 -0
  643. package/src/plugins/context-compact/index.ts +3 -0
  644. package/src/plugins/context-compact/plugin.ts +62 -0
  645. package/src/plugins/filesystem/filesystem.integration.test.ts +485 -0
  646. package/src/plugins/filesystem/helpers.ts +216 -0
  647. package/src/plugins/filesystem/index.ts +4 -0
  648. package/src/plugins/filesystem/listing.ts +276 -0
  649. package/src/plugins/filesystem/plugin.ts +468 -0
  650. package/src/plugins/filesystem/schema.ts +6 -0
  651. package/src/plugins/git-status/index.ts +1 -0
  652. package/src/plugins/git-status/plugin.ts +166 -0
  653. package/src/plugins/limits-guard/config.ts +26 -0
  654. package/src/plugins/limits-guard/index.ts +4 -0
  655. package/src/plugins/limits-guard/limit-guard.test.ts +161 -0
  656. package/src/plugins/limits-guard/limit-guard.ts +164 -0
  657. package/src/plugins/limits-guard/limits-guard.integration.test.ts +437 -0
  658. package/src/plugins/limits-guard/plugin.ts +306 -0
  659. package/src/plugins/llm-debug/index.ts +1 -0
  660. package/src/plugins/llm-debug/llm-debug.integration.test.ts +192 -0
  661. package/src/plugins/llm-debug/plugin.ts +172 -0
  662. package/src/plugins/logs/index.ts +1 -0
  663. package/src/plugins/logs/plugin.ts +39 -0
  664. package/src/plugins/mailbox/helpers.ts +83 -0
  665. package/src/plugins/mailbox/index.ts +13 -0
  666. package/src/plugins/mailbox/mailbox.integration.test.ts +705 -0
  667. package/src/plugins/mailbox/plugin.ts +270 -0
  668. package/src/plugins/mailbox/prompts.ts +104 -0
  669. package/src/plugins/mailbox/query.ts +53 -0
  670. package/src/plugins/mailbox/schema.ts +81 -0
  671. package/src/plugins/mailbox/state.ts +52 -0
  672. package/src/plugins/resources/index.ts +11 -0
  673. package/src/plugins/resources/manifest.ts +25 -0
  674. package/src/plugins/resources/plugin.ts +201 -0
  675. package/src/plugins/resources/post-inject.ts +74 -0
  676. package/src/plugins/resources/state.ts +30 -0
  677. package/src/plugins/result-eviction/index.ts +2 -0
  678. package/src/plugins/result-eviction/plugin.ts +65 -0
  679. package/src/plugins/result-eviction/result-eviction.integration.test.ts +244 -0
  680. package/src/plugins/services/plugin.ts +485 -0
  681. package/src/plugins/services/port-pool.ts +73 -0
  682. package/src/plugins/services/prompt.ts +53 -0
  683. package/src/plugins/services/schema.ts +83 -0
  684. package/src/plugins/services/service.ts +578 -0
  685. package/src/plugins/services/services.integration.test.ts +595 -0
  686. package/src/plugins/session-lifecycle/index.ts +1 -0
  687. package/src/plugins/session-lifecycle/plugin.ts +302 -0
  688. package/src/plugins/session-lifecycle/session-lifecycle.integration.test.ts +639 -0
  689. package/src/plugins/session-state/plugin.ts +179 -0
  690. package/src/plugins/session-stats/index.ts +3 -0
  691. package/src/plugins/session-stats/plugin.ts +120 -0
  692. package/src/plugins/shell/executor.ts +462 -0
  693. package/src/plugins/shell/index.ts +23 -0
  694. package/src/plugins/shell/plugin.ts +118 -0
  695. package/src/plugins/shell/shell.integration.test.ts +273 -0
  696. package/src/plugins/shell/shell.test.ts +314 -0
  697. package/src/plugins/skills/discovery.test.ts +395 -0
  698. package/src/plugins/skills/discovery.ts +276 -0
  699. package/src/plugins/skills/index.ts +32 -0
  700. package/src/plugins/skills/plugin.ts +381 -0
  701. package/src/plugins/skills/prompts.ts +84 -0
  702. package/src/plugins/skills/schema.ts +88 -0
  703. package/src/plugins/skills/skills.integration.test.ts +580 -0
  704. package/src/plugins/snapshotting/index.ts +3 -0
  705. package/src/plugins/snapshotting/jj-snapshotter.ts +122 -0
  706. package/src/plugins/snapshotting/plugin.ts +36 -0
  707. package/src/plugins/snapshotting/snapshotter.ts +20 -0
  708. package/src/plugins/todo/index.ts +11 -0
  709. package/src/plugins/todo/plugin.ts +363 -0
  710. package/src/plugins/todo/prompts.ts +63 -0
  711. package/src/plugins/todo/schema.ts +71 -0
  712. package/src/plugins/todo/todo.integration.test.ts +687 -0
  713. package/src/plugins/uploads/index.ts +13 -0
  714. package/src/plugins/uploads/plugin.ts +406 -0
  715. package/src/plugins/uploads/preprocessor.ts +111 -0
  716. package/src/plugins/uploads/preprocessors/image-classifier.ts +176 -0
  717. package/src/plugins/uploads/preprocessors/index.ts +7 -0
  718. package/src/plugins/uploads/preprocessors/markitdown-preprocessor.ts +262 -0
  719. package/src/plugins/uploads/preprocessors/zip-preprocessor.ts +210 -0
  720. package/src/plugins/uploads/schema.ts +92 -0
  721. package/src/plugins/uploads/state.ts +42 -0
  722. package/src/plugins/uploads/uploads.integration.test.ts +654 -0
  723. package/src/plugins/user-chat/index.ts +17 -0
  724. package/src/plugins/user-chat/plugin.ts +728 -0
  725. package/src/plugins/user-chat/prompts.ts +30 -0
  726. package/src/plugins/user-chat/schema.ts +80 -0
  727. package/src/plugins/user-chat/user-chat.integration.test.ts +794 -0
  728. package/src/plugins/workers/context.ts +251 -0
  729. package/src/plugins/workers/definition.ts +169 -0
  730. package/src/plugins/workers/index.ts +21 -0
  731. package/src/plugins/workers/plugin.ts +722 -0
  732. package/src/plugins/workers/worker.ts +115 -0
  733. package/src/plugins/workers/workers.integration.test.ts +778 -0
  734. package/src/prompts/base.ts +266 -0
  735. package/src/prompts/builder.ts +194 -0
  736. package/src/prompts/index.ts +36 -0
  737. package/src/prompts/macros.test.ts +107 -0
  738. package/src/prompts/macros.ts +27 -0
  739. package/src/testing/bootstrap-for-testing.ts +39 -0
  740. package/src/testing/index.ts +7 -0
  741. package/src/testing/node-platform.ts +80 -0
  742. package/src/testing/notification-collector.ts +100 -0
  743. package/src/testing/preset-helpers.ts +57 -0
  744. package/src/testing/test-harness.test.ts +63 -0
  745. package/src/testing/test-harness.ts +279 -0
  746. package/src/testing/wait-helpers.ts +84 -0
  747. package/src/transport/adapter/client-adapter.ts +81 -0
  748. package/src/transport/adapter/index.ts +42 -0
  749. package/src/transport/adapter/server-adapter.ts +93 -0
  750. package/src/transport/adapter/types.ts +20 -0
  751. package/src/transport/http/app.ts +129 -0
  752. package/src/transport/http/index.ts +7 -0
  753. package/src/transport/http/middleware/bearer-auth.ts +40 -0
  754. package/src/transport/http/middleware/error-handler.ts +76 -0
  755. package/src/transport/http/routes/files.ts +274 -0
  756. package/src/transport/http/routes/resources.ts +115 -0
  757. package/src/transport/http/routes/rpc.integration.test.ts +243 -0
  758. package/src/transport/http/routes/rpc.test.ts +372 -0
  759. package/src/transport/http/routes/rpc.ts +156 -0
  760. package/src/transport/http/routes/upload.ts +286 -0
  761. package/src/transport/rpc/index.ts +11 -0
  762. package/src/transport/rpc/methods.ts +38 -0
  763. package/src/user-config.ts +33 -0
@@ -0,0 +1,1730 @@
1
+ import { describe, expect, it } from 'bun:test'
2
+ import { COMMUNICATOR_ROLE, ORCHESTRATOR_ROLE } from '~/core/agents/agent-roles.js'
3
+ import { generateTestAgentId } from '~/core/agents/schema.js'
4
+ import { agentEvents } from '~/core/agents/state.js'
5
+ import { withSessionId } from '~/core/events/test-helpers.js'
6
+ import type { DomainEvent } from '~/core/events/types.js'
7
+ import { llmEvents } from '~/core/llm/state.js'
8
+ import { generateSessionId } from '~/core/sessions/schema.js'
9
+ import { sessionEvents } from '~/core/sessions/state.js'
10
+ import { generateToolCallId } from '~/core/tools/schema.js'
11
+ import { toolEvents } from '~/core/tools/state.js'
12
+ import { mailboxPlugin } from '~/plugins/mailbox/plugin.js'
13
+ import { getAgentMailbox, selectMailboxState } from '~/plugins/mailbox/query.js'
14
+ import { generateTestMessageId } from '~/plugins/mailbox/schema.js'
15
+ import { mailboxEvents } from '~/plugins/mailbox/state.js'
16
+ import { createApplyEvent } from './apply-event.js'
17
+ import { createSessionState, getAgentState, getCommunicatorId, getEntryAgentId, getOrchestratorId, reconstructSessionState } from './state.js'
18
+
19
+ const applyEvent = createApplyEvent([mailboxPlugin.create({})])
20
+
21
+ const SESSION_ID = generateSessionId()
22
+ const TIMESTAMP = Date.now()
23
+
24
+ describe('createSessionState', () => {
25
+ it('creates an empty session with correct properties', () => {
26
+ const session = createSessionState(SESSION_ID, 'test-preset', TIMESTAMP)
27
+
28
+ expect(session.id).toBe(SESSION_ID)
29
+ expect(session.presetId).toBe('test-preset')
30
+ expect(session.status).toBe('active')
31
+ expect(session.agents.size).toBe(0)
32
+ expect(session.createdAt).toBe(TIMESTAMP)
33
+ expect(session.closedAt).toBeUndefined()
34
+ })
35
+ })
36
+
37
+ describe('applyEvent', () => {
38
+ const baseSession = createSessionState(SESSION_ID, 'test-preset', TIMESTAMP)
39
+
40
+ describe('session_closed', () => {
41
+ it('closes the session', () => {
42
+ const event = withSessionId(SESSION_ID, sessionEvents.create('session_closed', {}))
43
+
44
+ const result = applyEvent(baseSession, event)
45
+
46
+ expect(result.status).toBe('closed')
47
+ expect(result.closedAt).toBe(event.timestamp)
48
+ })
49
+ })
50
+
51
+ describe('agent_spawned', () => {
52
+ it('adds a new agent to the session', () => {
53
+ const agentId = generateTestAgentId()
54
+ const event = withSessionId(
55
+ SESSION_ID,
56
+ agentEvents.create('agent_spawned', {
57
+ agentId,
58
+ definitionName: ORCHESTRATOR_ROLE,
59
+ parentId: null,
60
+ }),
61
+ )
62
+
63
+ const result = applyEvent(baseSession, event)
64
+
65
+ expect(result.agents.size).toBe(1)
66
+ const agent = result.agents.get(agentId)
67
+ expect(agent).toBeDefined()
68
+ expect(agent!.id).toBe(agentId)
69
+ expect(agent!.definitionName).toBe(ORCHESTRATOR_ROLE)
70
+ expect(agent!.parentId).toBeNull()
71
+ expect(agent!.status).toBe('pending')
72
+ expect(getAgentMailbox(selectMailboxState(result), agentId)).toHaveLength(0)
73
+ })
74
+
75
+ it('adds child agent with parent reference', () => {
76
+ const parentId = generateTestAgentId()
77
+ const childId = generateTestAgentId()
78
+
79
+ let session = applyEvent(
80
+ baseSession,
81
+ withSessionId(
82
+ SESSION_ID,
83
+ agentEvents.create('agent_spawned', {
84
+ agentId: parentId,
85
+ definitionName: ORCHESTRATOR_ROLE,
86
+ parentId: null,
87
+ }),
88
+ ),
89
+ )
90
+
91
+ session = applyEvent(
92
+ session,
93
+ withSessionId(
94
+ SESSION_ID,
95
+ agentEvents.create('agent_spawned', {
96
+ agentId: childId,
97
+ definitionName: 'worker',
98
+ parentId: parentId,
99
+ }),
100
+ ),
101
+ )
102
+
103
+ const child = session.agents.get(childId)
104
+ expect(child!.parentId).toBe(parentId)
105
+ })
106
+ })
107
+
108
+ describe('agent_state_changed', () => {
109
+ it('updates agent status', () => {
110
+ const agentId = generateTestAgentId()
111
+ let session = applyEvent(
112
+ baseSession,
113
+ withSessionId(
114
+ SESSION_ID,
115
+ agentEvents.create('agent_spawned', {
116
+ agentId,
117
+ definitionName: 'test',
118
+ parentId: null,
119
+ }),
120
+ ),
121
+ )
122
+
123
+ const event = withSessionId(
124
+ SESSION_ID,
125
+ agentEvents.create('agent_state_changed', {
126
+ agentId,
127
+ fromState: 'pending',
128
+ toState: 'inferring',
129
+ }),
130
+ )
131
+
132
+ session = applyEvent(session, event)
133
+
134
+ expect(session.agents.get(agentId)!.status).toBe('inferring')
135
+ })
136
+
137
+ it('throws for unknown agent', () => {
138
+ const unknownAgentId = generateTestAgentId()
139
+ const event = withSessionId(
140
+ SESSION_ID,
141
+ agentEvents.create('agent_state_changed', {
142
+ agentId: unknownAgentId,
143
+ fromState: 'pending',
144
+ toState: 'inferring',
145
+ }),
146
+ )
147
+
148
+ expect(() => applyEvent(baseSession, event)).toThrow(
149
+ `Agent not found: ${unknownAgentId}`,
150
+ )
151
+ })
152
+ })
153
+
154
+ describe('mailbox_message', () => {
155
+ it('adds message to agent mailbox', () => {
156
+ const agentId = generateTestAgentId()
157
+ let session = applyEvent(
158
+ baseSession,
159
+ withSessionId(
160
+ SESSION_ID,
161
+ agentEvents.create('agent_spawned', {
162
+ agentId,
163
+ definitionName: 'test',
164
+ parentId: null,
165
+ }),
166
+ ),
167
+ )
168
+
169
+ const messageId = generateTestMessageId()
170
+ const event = withSessionId(
171
+ SESSION_ID,
172
+ mailboxEvents.create('mailbox_message', {
173
+ toAgentId: agentId,
174
+ message: {
175
+ id: messageId,
176
+ from: 'user',
177
+ content: 'Hello',
178
+ timestamp: TIMESTAMP,
179
+ consumed: false,
180
+ },
181
+ }),
182
+ )
183
+
184
+ session = applyEvent(session, event)
185
+
186
+ const mailbox = getAgentMailbox(selectMailboxState(session), agentId)
187
+ expect(mailbox).toHaveLength(1)
188
+ expect(mailbox[0].id).toBe(messageId)
189
+ expect(mailbox[0].content).toBe('Hello')
190
+ })
191
+ })
192
+
193
+ describe('mailbox_consumed', () => {
194
+ it('marks messages as consumed', () => {
195
+ const agentId = generateTestAgentId()
196
+ const messageId = generateTestMessageId()
197
+
198
+ let session = applyEvent(
199
+ baseSession,
200
+ withSessionId(
201
+ SESSION_ID,
202
+ agentEvents.create('agent_spawned', {
203
+ agentId,
204
+ definitionName: 'test',
205
+ parentId: null,
206
+ }),
207
+ ),
208
+ )
209
+
210
+ session = applyEvent(
211
+ session,
212
+ withSessionId(
213
+ SESSION_ID,
214
+ mailboxEvents.create('mailbox_message', {
215
+ toAgentId: agentId,
216
+ message: {
217
+ id: messageId,
218
+ from: 'user',
219
+ content: 'Hello',
220
+ timestamp: TIMESTAMP,
221
+ consumed: false,
222
+ },
223
+ }),
224
+ ),
225
+ )
226
+
227
+ const event = withSessionId(
228
+ SESSION_ID,
229
+ mailboxEvents.create('mailbox_consumed', {
230
+ agentId,
231
+ messageIds: [messageId],
232
+ }),
233
+ )
234
+
235
+ session = applyEvent(session, event)
236
+
237
+ expect(getAgentMailbox(selectMailboxState(session), agentId)[0].consumed).toBe(true)
238
+ })
239
+ })
240
+
241
+ describe('inference_started', () => {
242
+ it('sets agent status to inferring', () => {
243
+ const agentId = generateTestAgentId()
244
+ let session = applyEvent(
245
+ baseSession,
246
+ withSessionId(
247
+ SESSION_ID,
248
+ agentEvents.create('agent_spawned', {
249
+ agentId,
250
+ definitionName: 'test',
251
+ parentId: null,
252
+ }),
253
+ ),
254
+ )
255
+
256
+ const event = withSessionId(
257
+ SESSION_ID,
258
+ llmEvents.create('inference_started', {
259
+ agentId,
260
+ messages: [],
261
+ consumedMessageIds: [],
262
+ }),
263
+ )
264
+
265
+ session = applyEvent(session, event)
266
+
267
+ expect(session.agents.get(agentId)!.status).toBe('inferring')
268
+ })
269
+
270
+ it('appends to pendingMessages (defense-in-depth against uncommitted messages)', () => {
271
+ // If for any reason pendingMessages are not empty when inference_started fires,
272
+ // they must not be overwritten — append instead.
273
+ const agentId = generateTestAgentId()
274
+
275
+ let session = applyEvent(
276
+ baseSession,
277
+ withSessionId(
278
+ SESSION_ID,
279
+ agentEvents.create('agent_spawned', {
280
+ agentId,
281
+ definitionName: 'test',
282
+ parentId: null,
283
+ }),
284
+ ),
285
+ )
286
+
287
+ const msg1 = { role: 'user' as const, content: 'First message' }
288
+ const msg2 = { role: 'user' as const, content: 'Second message' }
289
+
290
+ // First inference_started sets pendingMessages
291
+ session = applyEvent(
292
+ session,
293
+ withSessionId(
294
+ SESSION_ID,
295
+ llmEvents.create('inference_started', {
296
+ agentId,
297
+ messages: [msg1],
298
+ consumedMessageIds: [],
299
+ }),
300
+ ),
301
+ )
302
+
303
+ expect(session.agents.get(agentId)!.pendingMessages).toHaveLength(1)
304
+
305
+ // Second inference_started without inference_completed in between —
306
+ // must append, not overwrite
307
+ session = applyEvent(
308
+ session,
309
+ withSessionId(
310
+ SESSION_ID,
311
+ llmEvents.create('inference_started', {
312
+ agentId,
313
+ messages: [msg2],
314
+ consumedMessageIds: [],
315
+ }),
316
+ ),
317
+ )
318
+
319
+ const agent = session.agents.get(agentId)!
320
+ expect(agent.pendingMessages).toHaveLength(2)
321
+ expect(agent.pendingMessages[0].content).toBe(msg1.content)
322
+ expect(agent.pendingMessages[1].content).toBe(msg2.content)
323
+ })
324
+
325
+ it('afterInference pause: inference_completed commits turn, then agent_paused preserves pause', () => {
326
+ // Real scenario: planner receives subagent response, LLM says "WAITING",
327
+ // limits-guard pauses. The correct event sequence is:
328
+ // inference_started → inference_completed → agent_paused
329
+ // This ensures conversationHistory includes the turn and pendingMessages are cleared.
330
+ const agentId = generateTestAgentId()
331
+
332
+ let session = applyEvent(
333
+ baseSession,
334
+ withSessionId(
335
+ SESSION_ID,
336
+ agentEvents.create('agent_spawned', {
337
+ agentId,
338
+ definitionName: 'test',
339
+ parentId: null,
340
+ }),
341
+ ),
342
+ )
343
+
344
+ const userMsg = { role: 'user' as const, content: 'Response from collection-analyzer_1 (locations)' }
345
+
346
+ // 1. inference_started — message goes to pendingMessages
347
+ session = applyEvent(
348
+ session,
349
+ withSessionId(
350
+ SESSION_ID,
351
+ llmEvents.create('inference_started', {
352
+ agentId,
353
+ messages: [userMsg],
354
+ consumedMessageIds: [],
355
+ }),
356
+ ),
357
+ )
358
+
359
+ expect(session.agents.get(agentId)!.pendingMessages).toHaveLength(1)
360
+ expect(session.agents.get(agentId)!.status).toBe('inferring')
361
+
362
+ // 2. inference_completed — pendingMessages + response committed to conversationHistory
363
+ session = applyEvent(
364
+ session,
365
+ withSessionId(
366
+ SESSION_ID,
367
+ llmEvents.create('inference_completed', {
368
+ agentId,
369
+ consumedMessageIds: [],
370
+ response: { content: 'WAITING', toolCalls: [] },
371
+ metrics: { promptTokens: 200, completionTokens: 1, totalTokens: 201, latencyMs: 500, model: 'test' },
372
+ }),
373
+ ),
374
+ )
375
+
376
+ expect(session.agents.get(agentId)!.pendingMessages).toHaveLength(0)
377
+ expect(session.agents.get(agentId)!.conversationHistory).toHaveLength(2) // user + assistant
378
+ expect(session.agents.get(agentId)!.status).toBe('pending')
379
+
380
+ // 3. agent_paused — status set to paused, conversationHistory preserved
381
+ session = applyEvent(
382
+ session,
383
+ withSessionId(
384
+ SESSION_ID,
385
+ agentEvents.create('agent_paused', {
386
+ agentId,
387
+ reason: 'handler',
388
+ message: 'No progress for 5 turns',
389
+ }),
390
+ ),
391
+ )
392
+
393
+ const agent = session.agents.get(agentId)!
394
+ expect(agent.status).toBe('paused')
395
+ expect(agent.pendingMessages).toHaveLength(0)
396
+ expect(agent.conversationHistory).toHaveLength(2)
397
+ expect(agent.conversationHistory[0].content).toBe(userMsg.content)
398
+ expect(agent.conversationHistory[1].content).toBe('WAITING')
399
+
400
+ // 4. Resume + next message — starts fresh with empty pendingMessages
401
+ session = applyEvent(
402
+ session,
403
+ withSessionId(
404
+ SESSION_ID,
405
+ agentEvents.create('agent_resumed', { agentId }),
406
+ ),
407
+ )
408
+
409
+ const nextMsg = { role: 'user' as const, content: 'Response from collection-analyzer_2 (team)' }
410
+
411
+ session = applyEvent(
412
+ session,
413
+ withSessionId(
414
+ SESSION_ID,
415
+ llmEvents.create('inference_started', {
416
+ agentId,
417
+ messages: [nextMsg],
418
+ consumedMessageIds: [],
419
+ }),
420
+ ),
421
+ )
422
+
423
+ // Only the new message in pendingMessages (previous turn was committed)
424
+ expect(session.agents.get(agentId)!.pendingMessages).toHaveLength(1)
425
+ expect(session.agents.get(agentId)!.pendingMessages[0].content).toBe(nextMsg.content)
426
+ // Previous turn preserved in conversationHistory
427
+ expect(session.agents.get(agentId)!.conversationHistory).toHaveLength(2)
428
+ })
429
+ })
430
+
431
+ describe('inference_completed', () => {
432
+ it('updates conversation history and status when no tool calls', () => {
433
+ const agentId = generateTestAgentId()
434
+ let session = applyEvent(
435
+ baseSession,
436
+ withSessionId(
437
+ SESSION_ID,
438
+ agentEvents.create('agent_spawned', {
439
+ agentId,
440
+ definitionName: 'test',
441
+ parentId: null,
442
+ }),
443
+ ),
444
+ )
445
+
446
+ const event = withSessionId(
447
+ SESSION_ID,
448
+ llmEvents.create('inference_completed', {
449
+ agentId,
450
+ consumedMessageIds: [],
451
+ response: {
452
+ content: 'Hello there!',
453
+ toolCalls: [],
454
+ },
455
+ metrics: {
456
+ promptTokens: 100,
457
+ completionTokens: 10,
458
+ totalTokens: 110,
459
+ latencyMs: 500,
460
+ model: 'test-model',
461
+ },
462
+ }),
463
+ )
464
+
465
+ session = applyEvent(session, event)
466
+
467
+ const agent = session.agents.get(agentId)!
468
+ expect(agent.status).toBe('pending')
469
+ expect(agent.conversationHistory).toHaveLength(1)
470
+ expect(agent.conversationHistory[0].role).toBe('assistant')
471
+ expect(agent.conversationHistory[0].content).toBe('Hello there!')
472
+ expect(agent.pendingToolCalls).toHaveLength(0)
473
+ })
474
+
475
+ it('marks consumed messages via mailbox_consumed event', () => {
476
+ const agentId = generateTestAgentId()
477
+ const messageId = generateTestMessageId()
478
+
479
+ let session = applyEvent(
480
+ baseSession,
481
+ withSessionId(
482
+ SESSION_ID,
483
+ agentEvents.create('agent_spawned', {
484
+ agentId,
485
+ definitionName: 'test',
486
+ parentId: null,
487
+ }),
488
+ ),
489
+ )
490
+
491
+ session = applyEvent(
492
+ session,
493
+ withSessionId(
494
+ SESSION_ID,
495
+ mailboxEvents.create('mailbox_message', {
496
+ toAgentId: agentId,
497
+ message: {
498
+ id: messageId,
499
+ from: 'user',
500
+ content: 'Hello',
501
+ timestamp: TIMESTAMP,
502
+ consumed: false,
503
+ },
504
+ }),
505
+ ),
506
+ )
507
+
508
+ // Before consumption, message should be unconsumed
509
+ expect(getAgentMailbox(selectMailboxState(session), agentId)[0].consumed).toBe(false)
510
+
511
+ // Consume via mailbox_consumed (the dequeue mechanism)
512
+ session = applyEvent(
513
+ session,
514
+ withSessionId(
515
+ SESSION_ID,
516
+ mailboxEvents.create('mailbox_consumed', {
517
+ agentId,
518
+ messageIds: [messageId],
519
+ }),
520
+ ),
521
+ )
522
+
523
+ // After mailbox_consumed, message should be consumed
524
+ expect(getAgentMailbox(selectMailboxState(session), agentId)[0].consumed).toBe(true)
525
+
526
+ // Inference still works after consumption
527
+ session = applyEvent(
528
+ session,
529
+ withSessionId(
530
+ SESSION_ID,
531
+ llmEvents.create('inference_completed', {
532
+ agentId,
533
+ consumedMessageIds: [],
534
+ response: {
535
+ content: 'Hi!',
536
+ toolCalls: [],
537
+ },
538
+ metrics: {
539
+ promptTokens: 100,
540
+ completionTokens: 10,
541
+ totalTokens: 110,
542
+ latencyMs: 500,
543
+ model: 'test-model',
544
+ },
545
+ }),
546
+ ),
547
+ )
548
+
549
+ expect(session.agents.get(agentId)!.conversationHistory).toHaveLength(1)
550
+ })
551
+
552
+ it('sets tool_exec status when there are tool calls', () => {
553
+ const agentId = generateTestAgentId()
554
+ const toolCallId = generateToolCallId()
555
+ let session = applyEvent(
556
+ baseSession,
557
+ withSessionId(
558
+ SESSION_ID,
559
+ agentEvents.create('agent_spawned', {
560
+ agentId,
561
+ definitionName: 'test',
562
+ parentId: null,
563
+ }),
564
+ ),
565
+ )
566
+
567
+ const event = withSessionId(
568
+ SESSION_ID,
569
+ llmEvents.create('inference_completed', {
570
+ agentId,
571
+ consumedMessageIds: [],
572
+ response: {
573
+ content: null,
574
+ toolCalls: [
575
+ {
576
+ id: toolCallId,
577
+ name: 'send_message',
578
+ input: { to: 'other', content: 'hi' },
579
+ },
580
+ ],
581
+ },
582
+ metrics: {
583
+ promptTokens: 100,
584
+ completionTokens: 20,
585
+ totalTokens: 120,
586
+ latencyMs: 600,
587
+ model: 'test-model',
588
+ },
589
+ }),
590
+ )
591
+
592
+ session = applyEvent(session, event)
593
+
594
+ const agent = session.agents.get(agentId)!
595
+ expect(agent.status).toBe('tool_exec')
596
+ expect(agent.pendingToolCalls).toHaveLength(1)
597
+ expect(agent.pendingToolCalls[0].name).toBe('send_message')
598
+ })
599
+ })
600
+
601
+ describe('inference_failed', () => {
602
+ it('sets agent status to errored', () => {
603
+ const agentId = generateTestAgentId()
604
+ let session = applyEvent(
605
+ baseSession,
606
+ withSessionId(
607
+ SESSION_ID,
608
+ agentEvents.create('agent_spawned', {
609
+ agentId,
610
+ definitionName: 'test',
611
+ parentId: null,
612
+ }),
613
+ ),
614
+ )
615
+
616
+ const event = withSessionId(
617
+ SESSION_ID,
618
+ llmEvents.create('inference_failed', {
619
+ agentId,
620
+ error: 'API error',
621
+ }),
622
+ )
623
+
624
+ session = applyEvent(session, event)
625
+
626
+ expect(session.agents.get(agentId)!.status).toBe('errored')
627
+ })
628
+
629
+ it('clears pendingMessages but preserves pendingToolResults and mailbox for retry', () => {
630
+ const agentId = generateTestAgentId()
631
+ const messageId = generateTestMessageId()
632
+ const toolCallId = generateToolCallId()
633
+
634
+ // 1. Spawn agent
635
+ let session = applyEvent(
636
+ baseSession,
637
+ withSessionId(
638
+ SESSION_ID,
639
+ agentEvents.create('agent_spawned', {
640
+ agentId,
641
+ definitionName: 'test',
642
+ parentId: null,
643
+ }),
644
+ ),
645
+ )
646
+
647
+ // 2. Add mailbox message
648
+ session = applyEvent(
649
+ session,
650
+ withSessionId(
651
+ SESSION_ID,
652
+ mailboxEvents.create('mailbox_message', {
653
+ toAgentId: agentId,
654
+ message: {
655
+ id: messageId,
656
+ from: 'user',
657
+ content: 'Hello',
658
+ timestamp: TIMESTAMP,
659
+ consumed: false,
660
+ },
661
+ }),
662
+ ),
663
+ )
664
+
665
+ // 3. First inference with tool call
666
+ session = applyEvent(
667
+ session,
668
+ withSessionId(
669
+ SESSION_ID,
670
+ llmEvents.create('inference_completed', {
671
+ agentId,
672
+ consumedMessageIds: [],
673
+ response: {
674
+ content: null,
675
+ toolCalls: [{ id: toolCallId, name: 'test_tool', input: {} }],
676
+ },
677
+ metrics: { promptTokens: 100, completionTokens: 10, totalTokens: 110, latencyMs: 500, model: 'test' },
678
+ }),
679
+ ),
680
+ )
681
+
682
+ // 4. Tool completes - creates pendingToolResult
683
+ session = applyEvent(
684
+ session,
685
+ withSessionId(
686
+ SESSION_ID,
687
+ toolEvents.create('tool_completed', {
688
+ agentId,
689
+ toolCallId,
690
+ result: 'tool output',
691
+ }),
692
+ ),
693
+ )
694
+
695
+ // 5. Start inference with pending messages
696
+ session = applyEvent(
697
+ session,
698
+ withSessionId(
699
+ SESSION_ID,
700
+ llmEvents.create('inference_started', {
701
+ agentId,
702
+ messages: [
703
+ { role: 'tool' as const, toolCallId, toolName: 'test_tool', content: 'tool output', isError: false, timestamp: TIMESTAMP },
704
+ { role: 'user' as const, content: 'formatted mailbox', sourceMessageIds: [messageId] },
705
+ ],
706
+ consumedMessageIds: [messageId],
707
+ }),
708
+ ),
709
+ )
710
+
711
+ // Verify pendingMessages are set
712
+ expect(session.agents.get(agentId)!.pendingMessages).toHaveLength(2)
713
+ expect(session.agents.get(agentId)!.status).toBe('inferring')
714
+
715
+ // 6. Inference fails
716
+ session = applyEvent(
717
+ session,
718
+ withSessionId(
719
+ SESSION_ID,
720
+ llmEvents.create('inference_failed', {
721
+ agentId,
722
+ error: 'API timeout',
723
+ }),
724
+ ),
725
+ )
726
+
727
+ const agent = session.agents.get(agentId)!
728
+
729
+ // pendingMessages cleared
730
+ expect(agent.pendingMessages).toHaveLength(0)
731
+ // pendingToolResults preserved for rebuild
732
+ expect(agent.pendingToolResults).toHaveLength(1)
733
+ expect(agent.pendingToolResults[0].toolCallId).toBe(toolCallId)
734
+ // mailbox NOT consumed (can rebuild user message)
735
+ expect(getAgentMailbox(selectMailboxState(session), agentId)[0].consumed).toBe(false)
736
+ // status is errored
737
+ expect(agent.status).toBe('errored')
738
+ })
739
+ })
740
+
741
+ describe('session_restarted', () => {
742
+ it('clears pendingMessages for inferring agents, preserves pendingToolResults and mailbox', () => {
743
+ const agentId = generateTestAgentId()
744
+ const messageId = generateTestMessageId()
745
+ const toolCallId = generateToolCallId()
746
+
747
+ // 1. Spawn agent
748
+ let session = applyEvent(
749
+ baseSession,
750
+ withSessionId(
751
+ SESSION_ID,
752
+ agentEvents.create('agent_spawned', {
753
+ agentId,
754
+ definitionName: 'test',
755
+ parentId: null,
756
+ }),
757
+ ),
758
+ )
759
+
760
+ // 2. Add mailbox message
761
+ session = applyEvent(
762
+ session,
763
+ withSessionId(
764
+ SESSION_ID,
765
+ mailboxEvents.create('mailbox_message', {
766
+ toAgentId: agentId,
767
+ message: {
768
+ id: messageId,
769
+ from: 'user',
770
+ content: 'Hello',
771
+ timestamp: TIMESTAMP,
772
+ consumed: false,
773
+ },
774
+ }),
775
+ ),
776
+ )
777
+
778
+ // 3. First inference with tool call
779
+ session = applyEvent(
780
+ session,
781
+ withSessionId(
782
+ SESSION_ID,
783
+ llmEvents.create('inference_completed', {
784
+ agentId,
785
+ consumedMessageIds: [],
786
+ response: {
787
+ content: null,
788
+ toolCalls: [{ id: toolCallId, name: 'test_tool', input: {} }],
789
+ },
790
+ metrics: { promptTokens: 100, completionTokens: 10, totalTokens: 110, latencyMs: 500, model: 'test' },
791
+ }),
792
+ ),
793
+ )
794
+
795
+ // 4. Tool completes
796
+ session = applyEvent(
797
+ session,
798
+ withSessionId(
799
+ SESSION_ID,
800
+ toolEvents.create('tool_completed', {
801
+ agentId,
802
+ toolCallId,
803
+ result: 'tool output',
804
+ }),
805
+ ),
806
+ )
807
+
808
+ // 5. Start inference
809
+ session = applyEvent(
810
+ session,
811
+ withSessionId(
812
+ SESSION_ID,
813
+ llmEvents.create('inference_started', {
814
+ agentId,
815
+ messages: [
816
+ { role: 'tool' as const, toolCallId, toolName: 'test_tool', content: 'tool output', isError: false, timestamp: TIMESTAMP },
817
+ { role: 'user' as const, content: 'formatted mailbox', sourceMessageIds: [messageId] },
818
+ ],
819
+ consumedMessageIds: [messageId],
820
+ }),
821
+ ),
822
+ )
823
+
824
+ // Verify state before restart
825
+ expect(session.agents.get(agentId)!.pendingMessages).toHaveLength(2)
826
+ expect(session.agents.get(agentId)!.status).toBe('inferring')
827
+
828
+ // 6. Server restarts
829
+ session = applyEvent(session, {
830
+ ...withSessionId(
831
+ SESSION_ID,
832
+ sessionEvents.create('session_restarted', {
833
+ resetAgentIds: [agentId],
834
+ clearedToolAgentIds: [],
835
+ }),
836
+ ),
837
+ timestamp: TIMESTAMP + 1000,
838
+ })
839
+
840
+ const agent = session.agents.get(agentId)!
841
+
842
+ // pendingMessages cleared
843
+ expect(agent.pendingMessages).toHaveLength(0)
844
+ // pendingToolResults preserved - can rebuild tool messages
845
+ expect(agent.pendingToolResults).toHaveLength(1)
846
+ expect(agent.pendingToolResults[0].toolCallId).toBe(toolCallId)
847
+ // mailbox NOT consumed - can rebuild user message
848
+ expect(getAgentMailbox(selectMailboxState(session), agentId)[0].consumed).toBe(false)
849
+ // status reset to pending
850
+ expect(agent.status).toBe('pending')
851
+ })
852
+
853
+ it('allows successful retry after restart - full cycle', () => {
854
+ const agentId = generateTestAgentId()
855
+ const messageId = generateTestMessageId()
856
+ const toolCallId = generateToolCallId()
857
+
858
+ // 1. Setup: spawn, message, tool call, tool complete
859
+ let session = applyEvent(
860
+ baseSession,
861
+ withSessionId(
862
+ SESSION_ID,
863
+ agentEvents.create('agent_spawned', {
864
+ agentId,
865
+ definitionName: 'test',
866
+ parentId: null,
867
+ }),
868
+ ),
869
+ )
870
+
871
+ session = applyEvent(
872
+ session,
873
+ withSessionId(
874
+ SESSION_ID,
875
+ mailboxEvents.create('mailbox_message', {
876
+ toAgentId: agentId,
877
+ message: { id: messageId, from: 'user', content: 'Hello', timestamp: TIMESTAMP, consumed: false },
878
+ }),
879
+ ),
880
+ )
881
+
882
+ session = applyEvent(
883
+ session,
884
+ withSessionId(
885
+ SESSION_ID,
886
+ llmEvents.create('inference_completed', {
887
+ agentId,
888
+ consumedMessageIds: [],
889
+ response: { content: null, toolCalls: [{ id: toolCallId, name: 'test_tool', input: {} }] },
890
+ metrics: { promptTokens: 100, completionTokens: 10, totalTokens: 110, latencyMs: 500, model: 'test' },
891
+ }),
892
+ ),
893
+ )
894
+
895
+ session = applyEvent(
896
+ session,
897
+ withSessionId(
898
+ SESSION_ID,
899
+ toolEvents.create('tool_completed', {
900
+ agentId,
901
+ toolCallId,
902
+ result: 'tool output',
903
+ }),
904
+ ),
905
+ )
906
+
907
+ // 2. First inference attempt - interrupted by restart
908
+ session = applyEvent(
909
+ session,
910
+ withSessionId(
911
+ SESSION_ID,
912
+ llmEvents.create('inference_started', {
913
+ agentId,
914
+ messages: [
915
+ { role: 'tool' as const, toolCallId, toolName: 'test_tool', content: 'tool output', isError: false, timestamp: TIMESTAMP },
916
+ { role: 'user' as const, content: 'formatted mailbox v1', sourceMessageIds: [messageId] },
917
+ ],
918
+ consumedMessageIds: [messageId],
919
+ }),
920
+ ),
921
+ )
922
+
923
+ session = applyEvent(session, {
924
+ ...withSessionId(
925
+ SESSION_ID,
926
+ sessionEvents.create('session_restarted', {
927
+ resetAgentIds: [agentId],
928
+ clearedToolAgentIds: [],
929
+ }),
930
+ ),
931
+ timestamp: TIMESTAMP + 1000,
932
+ })
933
+
934
+ // 3. Retry inference - rebuild messages from preserved state
935
+ const agentAfterRestart = session.agents.get(agentId)!
936
+ expect(agentAfterRestart.status).toBe('pending')
937
+ expect(agentAfterRestart.pendingToolResults).toHaveLength(1)
938
+ expect(getAgentMailbox(selectMailboxState(session), agentId)[0].consumed).toBe(false)
939
+
940
+ // Runtime would rebuild: tool message from pendingToolResults + user message from unconsumed mailbox
941
+ session = applyEvent(session, {
942
+ ...withSessionId(
943
+ SESSION_ID,
944
+ llmEvents.create('inference_started', {
945
+ agentId,
946
+ messages: [
947
+ { role: 'tool' as const, toolCallId, toolName: 'test_tool', content: 'tool output', isError: false, timestamp: TIMESTAMP },
948
+ { role: 'user' as const, content: 'formatted mailbox v2', sourceMessageIds: [messageId] },
949
+ ],
950
+ consumedMessageIds: [messageId],
951
+ }),
952
+ ),
953
+ timestamp: TIMESTAMP + 2000,
954
+ })
955
+
956
+ // 4. Dequeue mechanism marks messages consumed
957
+ session = applyEvent(
958
+ session,
959
+ withSessionId(
960
+ SESSION_ID,
961
+ mailboxEvents.create('mailbox_consumed', {
962
+ agentId,
963
+ messageIds: [messageId],
964
+ }),
965
+ ),
966
+ )
967
+
968
+ // 5. This time inference completes successfully
969
+ session = applyEvent(session, {
970
+ ...withSessionId(
971
+ SESSION_ID,
972
+ llmEvents.create('inference_completed', {
973
+ agentId,
974
+ consumedMessageIds: [],
975
+ response: { content: 'Done!', toolCalls: [] },
976
+ metrics: { promptTokens: 100, completionTokens: 10, totalTokens: 110, latencyMs: 500, model: 'test' },
977
+ }),
978
+ ),
979
+ timestamp: TIMESTAMP + 3000,
980
+ })
981
+
982
+ const finalAgent = session.agents.get(agentId)!
983
+
984
+ // Everything committed
985
+ expect(finalAgent.status).toBe('pending')
986
+ expect(finalAgent.pendingMessages).toHaveLength(0)
987
+ expect(finalAgent.pendingToolResults).toHaveLength(0)
988
+ expect(getAgentMailbox(selectMailboxState(session), agentId)[0].consumed).toBe(true)
989
+
990
+ // History contains: assistant (tool call) + tool + user + assistant (final)
991
+ expect(finalAgent.conversationHistory).toHaveLength(4)
992
+ expect(finalAgent.conversationHistory[0].role).toBe('assistant')
993
+ expect(finalAgent.conversationHistory[1].role).toBe('tool')
994
+ expect(finalAgent.conversationHistory[2].role).toBe('user')
995
+ expect(finalAgent.conversationHistory[3].role).toBe('assistant')
996
+ expect(finalAgent.conversationHistory[3].content).toBe('Done!')
997
+ })
998
+ })
999
+
1000
+ describe('tool_completed', () => {
1001
+ it('removes tool call from pending and updates status', () => {
1002
+ const agentId = generateTestAgentId()
1003
+ const toolCallId = generateToolCallId()
1004
+
1005
+ let session = applyEvent(
1006
+ baseSession,
1007
+ withSessionId(
1008
+ SESSION_ID,
1009
+ agentEvents.create('agent_spawned', {
1010
+ agentId,
1011
+ definitionName: 'test',
1012
+ parentId: null,
1013
+ }),
1014
+ ),
1015
+ )
1016
+
1017
+ session = applyEvent(
1018
+ session,
1019
+ withSessionId(
1020
+ SESSION_ID,
1021
+ llmEvents.create('inference_completed', {
1022
+ agentId,
1023
+ consumedMessageIds: [],
1024
+ response: {
1025
+ content: null,
1026
+ toolCalls: [{ id: toolCallId, name: 'test', input: {} }],
1027
+ },
1028
+ metrics: {
1029
+ promptTokens: 100,
1030
+ completionTokens: 10,
1031
+ totalTokens: 110,
1032
+ latencyMs: 500,
1033
+ model: 'test-model',
1034
+ },
1035
+ }),
1036
+ ),
1037
+ )
1038
+
1039
+ const event = withSessionId(
1040
+ SESSION_ID,
1041
+ toolEvents.create('tool_completed', {
1042
+ agentId,
1043
+ toolCallId,
1044
+ result: 'success',
1045
+ }),
1046
+ )
1047
+
1048
+ session = applyEvent(session, event)
1049
+
1050
+ const agent = session.agents.get(agentId)!
1051
+ expect(agent.pendingToolCalls).toHaveLength(0)
1052
+ expect(agent.status).toBe('pending')
1053
+ })
1054
+
1055
+ it('keeps tool_exec status when there are remaining tool calls', () => {
1056
+ const agentId = generateTestAgentId()
1057
+ const toolCallId1 = generateToolCallId()
1058
+ const toolCallId2 = generateToolCallId()
1059
+
1060
+ let session = applyEvent(
1061
+ baseSession,
1062
+ withSessionId(
1063
+ SESSION_ID,
1064
+ agentEvents.create('agent_spawned', {
1065
+ agentId,
1066
+ definitionName: 'test',
1067
+ parentId: null,
1068
+ }),
1069
+ ),
1070
+ )
1071
+
1072
+ session = applyEvent(
1073
+ session,
1074
+ withSessionId(
1075
+ SESSION_ID,
1076
+ llmEvents.create('inference_completed', {
1077
+ agentId,
1078
+ consumedMessageIds: [],
1079
+ response: {
1080
+ content: null,
1081
+ toolCalls: [
1082
+ { id: toolCallId1, name: 'test1', input: {} },
1083
+ { id: toolCallId2, name: 'test2', input: {} },
1084
+ ],
1085
+ },
1086
+ metrics: {
1087
+ promptTokens: 100,
1088
+ completionTokens: 10,
1089
+ totalTokens: 110,
1090
+ latencyMs: 500,
1091
+ model: 'test-model',
1092
+ },
1093
+ }),
1094
+ ),
1095
+ )
1096
+
1097
+ session = applyEvent(
1098
+ session,
1099
+ withSessionId(
1100
+ SESSION_ID,
1101
+ toolEvents.create('tool_completed', {
1102
+ agentId,
1103
+ toolCallId: toolCallId1,
1104
+ result: 'success',
1105
+ }),
1106
+ ),
1107
+ )
1108
+
1109
+ const agent = session.agents.get(agentId)!
1110
+ expect(agent.pendingToolCalls).toHaveLength(1)
1111
+ expect(agent.status).toBe('tool_exec')
1112
+ })
1113
+
1114
+ it('stores content in pendingToolResults (deferred to history by inference_completed)', () => {
1115
+ const agentId = generateTestAgentId()
1116
+ const toolCallId = generateToolCallId()
1117
+ const toolTimestamp = Date.now()
1118
+
1119
+ let session = applyEvent(
1120
+ baseSession,
1121
+ withSessionId(
1122
+ SESSION_ID,
1123
+ agentEvents.create('agent_spawned', {
1124
+ agentId,
1125
+ definitionName: 'test',
1126
+ parentId: null,
1127
+ }),
1128
+ ),
1129
+ )
1130
+
1131
+ session = applyEvent(
1132
+ session,
1133
+ withSessionId(
1134
+ SESSION_ID,
1135
+ llmEvents.create('inference_completed', {
1136
+ agentId,
1137
+ consumedMessageIds: [],
1138
+ response: {
1139
+ content: null,
1140
+ toolCalls: [{ id: toolCallId, name: 'send_message', input: { to: 'agent-1', message: 'hello' } }],
1141
+ },
1142
+ metrics: {
1143
+ promptTokens: 100,
1144
+ completionTokens: 10,
1145
+ totalTokens: 110,
1146
+ latencyMs: 500,
1147
+ model: 'test-model',
1148
+ },
1149
+ }),
1150
+ ),
1151
+ )
1152
+
1153
+ session = applyEvent(session, {
1154
+ ...withSessionId(
1155
+ SESSION_ID,
1156
+ toolEvents.create('tool_completed', {
1157
+ agentId,
1158
+ toolCallId,
1159
+ result: 'success message',
1160
+ }),
1161
+ ),
1162
+ timestamp: toolTimestamp,
1163
+ })
1164
+
1165
+ const agent = session.agents.get(agentId)!
1166
+
1167
+ // Tool result should NOT be in conversationHistory yet (deferred pattern)
1168
+ const toolMessage = agent.conversationHistory.find(m => m.role === 'tool')
1169
+ expect(toolMessage).toBeUndefined()
1170
+
1171
+ // Check pendingToolResults has full content
1172
+ expect(agent.pendingToolResults).toHaveLength(1)
1173
+ expect(agent.pendingToolResults[0].toolCallId).toBe(toolCallId)
1174
+ expect(agent.pendingToolResults[0].toolName).toBe('send_message')
1175
+ expect(agent.pendingToolResults[0].timestamp).toBe(toolTimestamp)
1176
+ expect(agent.pendingToolResults[0].isError).toBe(false)
1177
+ expect(agent.pendingToolResults[0].content).toBe('success message')
1178
+ })
1179
+
1180
+ it('preserves rich content (array) in pendingToolResults', () => {
1181
+ const agentId = generateTestAgentId()
1182
+ const toolCallId = generateToolCallId()
1183
+
1184
+ let session = applyEvent(
1185
+ baseSession,
1186
+ withSessionId(
1187
+ SESSION_ID,
1188
+ agentEvents.create('agent_spawned', {
1189
+ agentId,
1190
+ definitionName: 'test',
1191
+ parentId: null,
1192
+ }),
1193
+ ),
1194
+ )
1195
+
1196
+ session = applyEvent(
1197
+ session,
1198
+ withSessionId(
1199
+ SESSION_ID,
1200
+ llmEvents.create('inference_completed', {
1201
+ agentId,
1202
+ consumedMessageIds: [],
1203
+ response: {
1204
+ content: null,
1205
+ toolCalls: [{ id: toolCallId, name: 'read_file', input: { path: '/test' } }],
1206
+ },
1207
+ metrics: {
1208
+ promptTokens: 100,
1209
+ completionTokens: 10,
1210
+ totalTokens: 110,
1211
+ latencyMs: 500,
1212
+ model: 'test-model',
1213
+ },
1214
+ }),
1215
+ ),
1216
+ )
1217
+
1218
+ // Rich content array (like image + text)
1219
+ const richContent = [
1220
+ { type: 'text' as const, text: 'File contents:' },
1221
+ { type: 'image_url' as const, imageUrl: { url: 'data:image/png;base64,abc123' } },
1222
+ ]
1223
+
1224
+ session = applyEvent(
1225
+ session,
1226
+ withSessionId(
1227
+ SESSION_ID,
1228
+ toolEvents.create('tool_completed', {
1229
+ agentId,
1230
+ toolCallId,
1231
+ result: richContent,
1232
+ }),
1233
+ ),
1234
+ )
1235
+
1236
+ const agent = session.agents.get(agentId)!
1237
+
1238
+ // Rich content should be preserved (not stringified)
1239
+ expect(agent.pendingToolResults[0].content).toEqual(richContent)
1240
+ })
1241
+
1242
+ it('inference_started does NOT clear pendingToolResults (deferred to inference_completed)', () => {
1243
+ const agentId = generateTestAgentId()
1244
+ const toolCallId = generateToolCallId()
1245
+
1246
+ let session = applyEvent(
1247
+ baseSession,
1248
+ withSessionId(
1249
+ SESSION_ID,
1250
+ agentEvents.create('agent_spawned', {
1251
+ agentId,
1252
+ definitionName: 'test',
1253
+ parentId: null,
1254
+ }),
1255
+ ),
1256
+ )
1257
+
1258
+ session = applyEvent(
1259
+ session,
1260
+ withSessionId(
1261
+ SESSION_ID,
1262
+ llmEvents.create('inference_completed', {
1263
+ agentId,
1264
+ consumedMessageIds: [],
1265
+ response: {
1266
+ content: null,
1267
+ toolCalls: [{ id: toolCallId, name: 'send_message', input: {} }],
1268
+ },
1269
+ metrics: {
1270
+ promptTokens: 100,
1271
+ completionTokens: 10,
1272
+ totalTokens: 110,
1273
+ latencyMs: 500,
1274
+ model: 'test-model',
1275
+ },
1276
+ }),
1277
+ ),
1278
+ )
1279
+
1280
+ session = applyEvent(
1281
+ session,
1282
+ withSessionId(
1283
+ SESSION_ID,
1284
+ toolEvents.create('tool_completed', {
1285
+ agentId,
1286
+ toolCallId,
1287
+ result: 'success',
1288
+ }),
1289
+ ),
1290
+ )
1291
+
1292
+ // Verify pendingToolResults has entry
1293
+ expect(session.agents.get(agentId)!.pendingToolResults).toHaveLength(1)
1294
+
1295
+ // Start new inference - should NOT clear pendingToolResults
1296
+ session = applyEvent(
1297
+ session,
1298
+ withSessionId(
1299
+ SESSION_ID,
1300
+ llmEvents.create('inference_started', {
1301
+ agentId,
1302
+ messages: [
1303
+ { role: 'tool' as const, toolCallId, toolName: 'send_message', content: 'success', isError: false, timestamp: TIMESTAMP },
1304
+ ],
1305
+ consumedMessageIds: [],
1306
+ }),
1307
+ ),
1308
+ )
1309
+
1310
+ // pendingToolResults preserved until inference_completed
1311
+ expect(session.agents.get(agentId)!.pendingToolResults).toHaveLength(1)
1312
+ })
1313
+
1314
+ it('inference_completed moves pendingToolResults to conversationHistory', () => {
1315
+ const agentId = generateTestAgentId()
1316
+ const toolCallId = generateToolCallId()
1317
+ const toolTimestamp = TIMESTAMP + 1000
1318
+
1319
+ let session = applyEvent(
1320
+ baseSession,
1321
+ withSessionId(
1322
+ SESSION_ID,
1323
+ agentEvents.create('agent_spawned', {
1324
+ agentId,
1325
+ definitionName: 'test',
1326
+ parentId: null,
1327
+ }),
1328
+ ),
1329
+ )
1330
+
1331
+ // First inference with tool call
1332
+ session = applyEvent(
1333
+ session,
1334
+ withSessionId(
1335
+ SESSION_ID,
1336
+ llmEvents.create('inference_completed', {
1337
+ agentId,
1338
+ consumedMessageIds: [],
1339
+ response: {
1340
+ content: null,
1341
+ toolCalls: [{ id: toolCallId, name: 'send_message', input: {} }],
1342
+ },
1343
+ metrics: {
1344
+ promptTokens: 100,
1345
+ completionTokens: 10,
1346
+ totalTokens: 110,
1347
+ latencyMs: 500,
1348
+ model: 'test-model',
1349
+ },
1350
+ }),
1351
+ ),
1352
+ )
1353
+
1354
+ // Tool completes
1355
+ session = applyEvent(session, {
1356
+ ...withSessionId(
1357
+ SESSION_ID,
1358
+ toolEvents.create('tool_completed', {
1359
+ agentId,
1360
+ toolCallId,
1361
+ result: 'tool result content',
1362
+ }),
1363
+ ),
1364
+ timestamp: toolTimestamp,
1365
+ })
1366
+
1367
+ // Verify: not in history yet, in pendingToolResults
1368
+ let agent = session.agents.get(agentId)!
1369
+ expect(agent.conversationHistory.filter(m => m.role === 'tool')).toHaveLength(0)
1370
+ expect(agent.pendingToolResults).toHaveLength(1)
1371
+
1372
+ // Start and complete new inference
1373
+ session = applyEvent(session, {
1374
+ ...withSessionId(
1375
+ SESSION_ID,
1376
+ llmEvents.create('inference_started', {
1377
+ agentId,
1378
+ messages: [
1379
+ { role: 'tool' as const, toolCallId, toolName: 'send_message', content: 'tool result content', isError: false, timestamp: toolTimestamp },
1380
+ ],
1381
+ consumedMessageIds: [],
1382
+ }),
1383
+ ),
1384
+ timestamp: TIMESTAMP + 2000,
1385
+ })
1386
+
1387
+ session = applyEvent(session, {
1388
+ ...withSessionId(
1389
+ SESSION_ID,
1390
+ llmEvents.create('inference_completed', {
1391
+ agentId,
1392
+ consumedMessageIds: [],
1393
+ response: {
1394
+ content: 'Response after tool',
1395
+ toolCalls: [],
1396
+ },
1397
+ metrics: {
1398
+ promptTokens: 100,
1399
+ completionTokens: 10,
1400
+ totalTokens: 110,
1401
+ latencyMs: 500,
1402
+ model: 'test-model',
1403
+ },
1404
+ }),
1405
+ ),
1406
+ timestamp: TIMESTAMP + 3000,
1407
+ })
1408
+
1409
+ // Verify: pendingToolResults moved to history, then cleared
1410
+ agent = session.agents.get(agentId)!
1411
+ expect(agent.pendingToolResults).toHaveLength(0)
1412
+
1413
+ const toolMessages = agent.conversationHistory.filter(m => m.role === 'tool')
1414
+ expect(toolMessages).toHaveLength(1)
1415
+
1416
+ const toolMessage = toolMessages[0]
1417
+ if (toolMessage.role === 'tool') {
1418
+ expect(toolMessage.toolCallId).toBe(toolCallId)
1419
+ expect(toolMessage.toolName).toBe('send_message')
1420
+ expect(toolMessage.content).toBe('tool result content')
1421
+ expect(toolMessage.timestamp).toBe(toolTimestamp)
1422
+ expect(toolMessage.isError).toBe(false)
1423
+ }
1424
+ })
1425
+ })
1426
+
1427
+ describe('tool_failed', () => {
1428
+ it('removes tool call from pending', () => {
1429
+ const agentId = generateTestAgentId()
1430
+ const toolCallId = generateToolCallId()
1431
+
1432
+ let session = applyEvent(
1433
+ baseSession,
1434
+ withSessionId(
1435
+ SESSION_ID,
1436
+ agentEvents.create('agent_spawned', {
1437
+ agentId,
1438
+ definitionName: 'test',
1439
+ parentId: null,
1440
+ }),
1441
+ ),
1442
+ )
1443
+
1444
+ session = applyEvent(
1445
+ session,
1446
+ withSessionId(
1447
+ SESSION_ID,
1448
+ llmEvents.create('inference_completed', {
1449
+ agentId,
1450
+ consumedMessageIds: [],
1451
+ response: {
1452
+ content: null,
1453
+ toolCalls: [{ id: toolCallId, name: 'test', input: {} }],
1454
+ },
1455
+ metrics: {
1456
+ promptTokens: 100,
1457
+ completionTokens: 10,
1458
+ totalTokens: 110,
1459
+ latencyMs: 500,
1460
+ model: 'test-model',
1461
+ },
1462
+ }),
1463
+ ),
1464
+ )
1465
+
1466
+ const event = withSessionId(
1467
+ SESSION_ID,
1468
+ toolEvents.create('tool_failed', {
1469
+ agentId,
1470
+ toolCallId,
1471
+ error: 'Tool error',
1472
+ }),
1473
+ )
1474
+
1475
+ session = applyEvent(session, event)
1476
+
1477
+ const agent = session.agents.get(agentId)!
1478
+ expect(agent.pendingToolCalls).toHaveLength(0)
1479
+ expect(agent.status).toBe('pending')
1480
+ })
1481
+
1482
+ it('stores error in pendingToolResults with isError=true (deferred to history by inference_completed)', () => {
1483
+ const agentId = generateTestAgentId()
1484
+ const toolCallId = generateToolCallId()
1485
+ const toolTimestamp = Date.now()
1486
+
1487
+ let session = applyEvent(
1488
+ baseSession,
1489
+ withSessionId(
1490
+ SESSION_ID,
1491
+ agentEvents.create('agent_spawned', {
1492
+ agentId,
1493
+ definitionName: 'test',
1494
+ parentId: null,
1495
+ }),
1496
+ ),
1497
+ )
1498
+
1499
+ session = applyEvent(
1500
+ session,
1501
+ withSessionId(
1502
+ SESSION_ID,
1503
+ llmEvents.create('inference_completed', {
1504
+ agentId,
1505
+ consumedMessageIds: [],
1506
+ response: {
1507
+ content: null,
1508
+ toolCalls: [{ id: toolCallId, name: 'reveal_secret', input: { password: 'test' } }],
1509
+ },
1510
+ metrics: {
1511
+ promptTokens: 100,
1512
+ completionTokens: 10,
1513
+ totalTokens: 110,
1514
+ latencyMs: 500,
1515
+ model: 'test-model',
1516
+ },
1517
+ }),
1518
+ ),
1519
+ )
1520
+
1521
+ session = applyEvent(session, {
1522
+ ...withSessionId(
1523
+ SESSION_ID,
1524
+ toolEvents.create('tool_failed', {
1525
+ agentId,
1526
+ toolCallId,
1527
+ error: 'Invalid password',
1528
+ }),
1529
+ ),
1530
+ timestamp: toolTimestamp,
1531
+ })
1532
+
1533
+ const agent = session.agents.get(agentId)!
1534
+
1535
+ // Tool result should NOT be in conversationHistory yet (deferred pattern)
1536
+ const toolMessage = agent.conversationHistory.find(m => m.role === 'tool')
1537
+ expect(toolMessage).toBeUndefined()
1538
+
1539
+ // Check pendingToolResults has error content with isError=true
1540
+ expect(agent.pendingToolResults).toHaveLength(1)
1541
+ expect(agent.pendingToolResults[0].toolCallId).toBe(toolCallId)
1542
+ expect(agent.pendingToolResults[0].toolName).toBe('reveal_secret')
1543
+ expect(agent.pendingToolResults[0].timestamp).toBe(toolTimestamp)
1544
+ expect(agent.pendingToolResults[0].isError).toBe(true)
1545
+ expect(agent.pendingToolResults[0].content).toBe('Invalid password')
1546
+ })
1547
+ })
1548
+ })
1549
+
1550
+ describe('reconstructSessionState', () => {
1551
+ it('returns null for empty events array', () => {
1552
+ expect(reconstructSessionState([], applyEvent)).toBeNull()
1553
+ })
1554
+
1555
+ it('throws error when first event is not session_created', () => {
1556
+ const events: DomainEvent[] = [
1557
+ withSessionId(SESSION_ID, sessionEvents.create('session_closed', {})),
1558
+ ]
1559
+
1560
+ expect(() => reconstructSessionState(events, applyEvent)).toThrow(
1561
+ 'First event must be session_created',
1562
+ )
1563
+ })
1564
+
1565
+ it('reconstructs session from events', () => {
1566
+ const agentId = generateTestAgentId()
1567
+ const events: DomainEvent[] = [
1568
+ withSessionId(SESSION_ID, sessionEvents.create('session_created', { presetId: 'test-preset' })),
1569
+ withSessionId(
1570
+ SESSION_ID,
1571
+ agentEvents.create('agent_spawned', {
1572
+ agentId,
1573
+ definitionName: ORCHESTRATOR_ROLE,
1574
+ parentId: null,
1575
+ }),
1576
+ ),
1577
+ ]
1578
+
1579
+ const session = reconstructSessionState(events, applyEvent)
1580
+
1581
+ expect(session).not.toBeNull()
1582
+ expect(session!.id).toBe(SESSION_ID)
1583
+ expect(session!.agents.size).toBe(1)
1584
+ expect(session!.agents.get(agentId)!.definitionName).toBe(ORCHESTRATOR_ROLE)
1585
+ })
1586
+ })
1587
+
1588
+ describe('Query helpers', () => {
1589
+ describe('getOrchestratorId', () => {
1590
+ it('returns orchestrator agent id', () => {
1591
+ const orchestratorId = generateTestAgentId()
1592
+ const events: DomainEvent[] = [
1593
+ withSessionId(SESSION_ID, sessionEvents.create('session_created', { presetId: 'test' })),
1594
+ withSessionId(
1595
+ SESSION_ID,
1596
+ agentEvents.create('agent_spawned', {
1597
+ agentId: orchestratorId,
1598
+ definitionName: ORCHESTRATOR_ROLE,
1599
+ parentId: null,
1600
+ }),
1601
+ ),
1602
+ ]
1603
+
1604
+ const session = reconstructSessionState(events, applyEvent)!
1605
+ expect(getOrchestratorId(session)).toBe(orchestratorId)
1606
+ })
1607
+
1608
+ it('returns null when no orchestrator', () => {
1609
+ const session = createSessionState(SESSION_ID, 'test', TIMESTAMP)
1610
+ expect(getOrchestratorId(session)).toBeNull()
1611
+ })
1612
+ })
1613
+
1614
+ describe('getCommunicatorId', () => {
1615
+ it('returns communicator agent id', () => {
1616
+ const communicatorId = generateTestAgentId()
1617
+ const events: DomainEvent[] = [
1618
+ withSessionId(SESSION_ID, sessionEvents.create('session_created', { presetId: 'test' })),
1619
+ withSessionId(
1620
+ SESSION_ID,
1621
+ agentEvents.create('agent_spawned', {
1622
+ agentId: communicatorId,
1623
+ definitionName: COMMUNICATOR_ROLE,
1624
+ parentId: null,
1625
+ }),
1626
+ ),
1627
+ ]
1628
+
1629
+ const session = reconstructSessionState(events, applyEvent)!
1630
+ expect(getCommunicatorId(session)).toBe(communicatorId)
1631
+ })
1632
+ })
1633
+
1634
+ describe('getEntryAgentId', () => {
1635
+ it('returns communicator when both exist', () => {
1636
+ const orchestratorId = generateTestAgentId()
1637
+ const communicatorId = generateTestAgentId()
1638
+ const events: DomainEvent[] = [
1639
+ withSessionId(SESSION_ID, sessionEvents.create('session_created', { presetId: 'test' })),
1640
+ withSessionId(
1641
+ SESSION_ID,
1642
+ agentEvents.create('agent_spawned', {
1643
+ agentId: orchestratorId,
1644
+ definitionName: ORCHESTRATOR_ROLE,
1645
+ parentId: null,
1646
+ }),
1647
+ ),
1648
+ withSessionId(
1649
+ SESSION_ID,
1650
+ agentEvents.create('agent_spawned', {
1651
+ agentId: communicatorId,
1652
+ definitionName: COMMUNICATOR_ROLE,
1653
+ parentId: orchestratorId,
1654
+ }),
1655
+ ),
1656
+ ]
1657
+
1658
+ const session = reconstructSessionState(events, applyEvent)!
1659
+ expect(getEntryAgentId(session)).toBe(communicatorId)
1660
+ })
1661
+
1662
+ it('returns orchestrator when no communicator', () => {
1663
+ const orchestratorId = generateTestAgentId()
1664
+ const events: DomainEvent[] = [
1665
+ withSessionId(SESSION_ID, sessionEvents.create('session_created', { presetId: 'test' })),
1666
+ withSessionId(
1667
+ SESSION_ID,
1668
+ agentEvents.create('agent_spawned', {
1669
+ agentId: orchestratorId,
1670
+ definitionName: ORCHESTRATOR_ROLE,
1671
+ parentId: null,
1672
+ }),
1673
+ ),
1674
+ ]
1675
+
1676
+ const session = reconstructSessionState(events, applyEvent)!
1677
+ expect(getEntryAgentId(session)).toBe(orchestratorId)
1678
+ })
1679
+ })
1680
+
1681
+ describe('getAgentState', () => {
1682
+ it('returns agent by id', () => {
1683
+ const agentId = generateTestAgentId()
1684
+ const events: DomainEvent[] = [
1685
+ withSessionId(SESSION_ID, sessionEvents.create('session_created', { presetId: 'test' })),
1686
+ withSessionId(
1687
+ SESSION_ID,
1688
+ agentEvents.create('agent_spawned', {
1689
+ agentId,
1690
+ definitionName: 'test',
1691
+ parentId: null,
1692
+ }),
1693
+ ),
1694
+ ]
1695
+
1696
+ const session = reconstructSessionState(events, applyEvent)!
1697
+ const agent = getAgentState(session, agentId)
1698
+
1699
+ expect(agent).not.toBeNull()
1700
+ expect(agent!.id).toBe(agentId)
1701
+ })
1702
+
1703
+ it('returns null for unknown agent', () => {
1704
+ const session = createSessionState(SESSION_ID, 'test', TIMESTAMP)
1705
+ expect(getAgentState(session, generateTestAgentId())).toBeNull()
1706
+ })
1707
+ })
1708
+ })
1709
+
1710
+ describe('Immutability', () => {
1711
+ it('applyEvent does not mutate original session', () => {
1712
+ const session = createSessionState(SESSION_ID, 'test', TIMESTAMP)
1713
+ const agentId = generateTestAgentId()
1714
+
1715
+ const event = withSessionId(
1716
+ SESSION_ID,
1717
+ agentEvents.create('agent_spawned', {
1718
+ agentId,
1719
+ definitionName: 'test',
1720
+ parentId: null,
1721
+ }),
1722
+ )
1723
+
1724
+ const newSession = applyEvent(session, event)
1725
+
1726
+ expect(session.agents.size).toBe(0)
1727
+ expect(newSession.agents.size).toBe(1)
1728
+ expect(session).not.toBe(newSession)
1729
+ })
1730
+ })