@roj-ai/sdk 0.1.3 → 0.1.5

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 (493) hide show
  1. package/dist/bootstrap.js +191 -0
  2. package/dist/bootstrap.js.map +1 -0
  3. package/dist/builtin-events.js +8 -0
  4. package/dist/builtin-events.js.map +1 -0
  5. package/dist/bun-platform/fs.js +39 -0
  6. package/dist/bun-platform/fs.js.map +1 -0
  7. package/dist/bun-platform/index.js +18 -0
  8. package/dist/bun-platform/index.js.map +1 -0
  9. package/dist/bun-platform/process.js +21 -0
  10. package/dist/bun-platform/process.js.map +1 -0
  11. package/dist/config.js +54 -0
  12. package/dist/config.js.map +1 -0
  13. package/dist/config.test.js +155 -0
  14. package/dist/config.test.js.map +1 -0
  15. package/dist/core/agent-loop.integration.test.js +414 -0
  16. package/dist/core/agent-loop.integration.test.js.map +1 -0
  17. package/dist/core/agents/agent-config.test.js +194 -0
  18. package/dist/core/agents/agent-config.test.js.map +1 -0
  19. package/dist/core/agents/agent-roles.js +25 -0
  20. package/dist/core/agents/agent-roles.js.map +1 -0
  21. package/dist/core/agents/agent-shutdown.test.js +180 -0
  22. package/dist/core/agents/agent-shutdown.test.js.map +1 -0
  23. package/dist/core/agents/agent.js +1205 -0
  24. package/dist/core/agents/agent.js.map +1 -0
  25. package/dist/core/agents/agent.test.js +313 -0
  26. package/dist/core/agents/agent.test.js.map +1 -0
  27. package/dist/core/agents/communicator.js +13 -0
  28. package/dist/core/agents/communicator.js.map +1 -0
  29. package/dist/core/agents/config.js +5 -0
  30. package/dist/core/agents/config.js.map +1 -0
  31. package/dist/core/agents/context.js +2 -0
  32. package/dist/core/agents/context.js.map +1 -0
  33. package/dist/core/agents/debounce.js +74 -0
  34. package/dist/core/agents/debounce.js.map +1 -0
  35. package/dist/core/agents/handler-events.test.js +115 -0
  36. package/dist/core/agents/handler-events.test.js.map +1 -0
  37. package/dist/core/agents/index.js +2 -0
  38. package/dist/core/agents/index.js.map +1 -0
  39. package/dist/core/agents/response-sanitizer.js +46 -0
  40. package/dist/core/agents/response-sanitizer.js.map +1 -0
  41. package/dist/core/agents/response-sanitizer.test.js +101 -0
  42. package/dist/core/agents/response-sanitizer.test.js.map +1 -0
  43. package/dist/core/agents/retry.js +105 -0
  44. package/dist/core/agents/retry.js.map +1 -0
  45. package/dist/core/agents/schema.js +39 -0
  46. package/dist/core/agents/schema.js.map +1 -0
  47. package/dist/core/agents/state.js +90 -0
  48. package/dist/core/agents/state.js.map +1 -0
  49. package/dist/core/context/state.js +23 -0
  50. package/dist/core/context/state.js.map +1 -0
  51. package/dist/core/errors.js +38 -0
  52. package/dist/core/errors.js.map +1 -0
  53. package/dist/core/event-sourcing.integration.test.js +154 -0
  54. package/dist/core/event-sourcing.integration.test.js.map +1 -0
  55. package/dist/core/events/base-event-store.js +201 -0
  56. package/dist/core/events/base-event-store.js.map +1 -0
  57. package/dist/core/events/event-store.js +26 -0
  58. package/dist/core/events/event-store.js.map +1 -0
  59. package/dist/core/events/file.js +320 -0
  60. package/dist/core/events/file.js.map +1 -0
  61. package/dist/core/events/file.test.js +284 -0
  62. package/dist/core/events/file.test.js.map +1 -0
  63. package/dist/core/events/index.js +3 -0
  64. package/dist/core/events/index.js.map +1 -0
  65. package/dist/core/events/memory.js +101 -0
  66. package/dist/core/events/memory.js.map +1 -0
  67. package/dist/core/events/memory.test.js +502 -0
  68. package/dist/core/events/memory.test.js.map +1 -0
  69. package/dist/core/events/metadata-utils.js +107 -0
  70. package/dist/core/events/metadata-utils.js.map +1 -0
  71. package/dist/core/events/test-helpers.js +15 -0
  72. package/dist/core/events/test-helpers.js.map +1 -0
  73. package/dist/core/events/types.js +21 -0
  74. package/dist/core/events/types.js.map +1 -0
  75. package/dist/core/file-store/file-store.js +250 -0
  76. package/dist/core/file-store/file-store.js.map +1 -0
  77. package/dist/core/file-store/types.js +7 -0
  78. package/dist/core/file-store/types.js.map +1 -0
  79. package/dist/core/image/image-processor.js +106 -0
  80. package/dist/core/image/image-processor.js.map +1 -0
  81. package/dist/core/image/image-processor.test.js +171 -0
  82. package/dist/core/image/image-processor.test.js.map +1 -0
  83. package/dist/core/image/index.js +4 -0
  84. package/dist/core/image/index.js.map +1 -0
  85. package/dist/core/image/noop-resizer.js +6 -0
  86. package/dist/core/image/noop-resizer.js.map +1 -0
  87. package/dist/core/image/types.js +2 -0
  88. package/dist/core/image/types.js.map +1 -0
  89. package/dist/core/image/vips-resizer.js +100 -0
  90. package/dist/core/image/vips-resizer.js.map +1 -0
  91. package/dist/core/image/vips-resizer.test.js +324 -0
  92. package/dist/core/image/vips-resizer.test.js.map +1 -0
  93. package/dist/core/llm/anthropic.js +396 -0
  94. package/dist/core/llm/anthropic.js.map +1 -0
  95. package/dist/core/llm/anthropic.test.js +434 -0
  96. package/dist/core/llm/anthropic.test.js.map +1 -0
  97. package/dist/core/llm/cache-breakpoints.js +37 -0
  98. package/dist/core/llm/cache-breakpoints.js.map +1 -0
  99. package/dist/core/llm/cache-live.test.js +137 -0
  100. package/dist/core/llm/cache-live.test.js.map +1 -0
  101. package/dist/core/llm/index.js +9 -0
  102. package/dist/core/llm/index.js.map +1 -0
  103. package/dist/core/llm/llm-log-types.js +12 -0
  104. package/dist/core/llm/llm-log-types.js.map +1 -0
  105. package/dist/core/llm/logger.js +241 -0
  106. package/dist/core/llm/logger.js.map +1 -0
  107. package/dist/core/llm/logger.test.js +228 -0
  108. package/dist/core/llm/logger.test.js.map +1 -0
  109. package/dist/core/llm/logging-provider.js +49 -0
  110. package/dist/core/llm/logging-provider.js.map +1 -0
  111. package/dist/core/llm/middleware.js +114 -0
  112. package/dist/core/llm/middleware.js.map +1 -0
  113. package/dist/core/llm/mock.js +186 -0
  114. package/dist/core/llm/mock.js.map +1 -0
  115. package/dist/core/llm/mock.test.js +318 -0
  116. package/dist/core/llm/mock.test.js.map +1 -0
  117. package/dist/core/llm/openrouter-mapping.test.js +125 -0
  118. package/dist/core/llm/openrouter-mapping.test.js.map +1 -0
  119. package/dist/core/llm/openrouter.js +298 -0
  120. package/dist/core/llm/openrouter.js.map +1 -0
  121. package/dist/core/llm/openrouter.test.js +377 -0
  122. package/dist/core/llm/openrouter.test.js.map +1 -0
  123. package/dist/core/llm/provider-integration.test.js +350 -0
  124. package/dist/core/llm/provider-integration.test.js.map +1 -0
  125. package/dist/core/llm/provider.js +18 -0
  126. package/dist/core/llm/provider.js.map +1 -0
  127. package/dist/core/llm/routing-provider.js +52 -0
  128. package/dist/core/llm/routing-provider.js.map +1 -0
  129. package/dist/core/llm/routing-provider.test.js +94 -0
  130. package/dist/core/llm/routing-provider.test.js.map +1 -0
  131. package/dist/core/llm/schema.js +31 -0
  132. package/dist/core/llm/schema.js.map +1 -0
  133. package/dist/core/llm/snapshot-fetch.js +122 -0
  134. package/dist/core/llm/snapshot-fetch.js.map +1 -0
  135. package/dist/core/llm/snapshot-middleware.js +142 -0
  136. package/dist/core/llm/snapshot-middleware.js.map +1 -0
  137. package/dist/core/llm/snapshot-middleware.test.js +144 -0
  138. package/dist/core/llm/snapshot-middleware.test.js.map +1 -0
  139. package/dist/core/llm/state.js +48 -0
  140. package/dist/core/llm/state.js.map +1 -0
  141. package/dist/core/llm/tokens.js +40 -0
  142. package/dist/core/llm/tokens.js.map +1 -0
  143. package/dist/core/multi-agent.integration.test.js +298 -0
  144. package/dist/core/multi-agent.integration.test.js.map +1 -0
  145. package/dist/core/plugin-hooks.integration.test.js +344 -0
  146. package/dist/core/plugin-hooks.integration.test.js.map +1 -0
  147. package/dist/core/plugins/hook-types.js +5 -0
  148. package/dist/core/plugins/hook-types.js.map +1 -0
  149. package/dist/core/plugins/index.js +5 -0
  150. package/dist/core/plugins/index.js.map +1 -0
  151. package/dist/core/plugins/plugin-builder.js +321 -0
  152. package/dist/core/plugins/plugin-builder.js.map +1 -0
  153. package/dist/core/preset/config.js +54 -0
  154. package/dist/core/preset/config.js.map +1 -0
  155. package/dist/core/preset/index.js +6 -0
  156. package/dist/core/preset/index.js.map +1 -0
  157. package/dist/core/preset/preset-builder.js +63 -0
  158. package/dist/core/preset/preset-builder.js.map +1 -0
  159. package/dist/core/session-lifecycle.integration.test.js +159 -0
  160. package/dist/core/session-lifecycle.integration.test.js.map +1 -0
  161. package/dist/core/sessions/apply-event.js +41 -0
  162. package/dist/core/sessions/apply-event.js.map +1 -0
  163. package/dist/core/sessions/context.js +2 -0
  164. package/dist/core/sessions/context.js.map +1 -0
  165. package/dist/core/sessions/fork-utils.js +42 -0
  166. package/dist/core/sessions/fork-utils.js.map +1 -0
  167. package/dist/core/sessions/fork-utils.test.js +129 -0
  168. package/dist/core/sessions/fork-utils.test.js.map +1 -0
  169. package/dist/core/sessions/reducer.js +55 -0
  170. package/dist/core/sessions/reducer.js.map +1 -0
  171. package/dist/core/sessions/schema.js +66 -0
  172. package/dist/core/sessions/schema.js.map +1 -0
  173. package/dist/core/sessions/session-environment.js +2 -0
  174. package/dist/core/sessions/session-environment.js.map +1 -0
  175. package/dist/core/sessions/session-manager.js +650 -0
  176. package/dist/core/sessions/session-manager.js.map +1 -0
  177. package/dist/core/sessions/session-store.js +118 -0
  178. package/dist/core/sessions/session-store.js.map +1 -0
  179. package/dist/core/sessions/session.js +675 -0
  180. package/dist/core/sessions/session.js.map +1 -0
  181. package/dist/core/sessions/session.test.js +1095 -0
  182. package/dist/core/sessions/session.test.js.map +1 -0
  183. package/dist/core/sessions/state.js +377 -0
  184. package/dist/core/sessions/state.js.map +1 -0
  185. package/dist/core/system.js +66 -0
  186. package/dist/core/system.js.map +1 -0
  187. package/dist/core/tools/context.js +2 -0
  188. package/dist/core/tools/context.js.map +1 -0
  189. package/dist/core/tools/definition.js +4 -0
  190. package/dist/core/tools/definition.js.map +1 -0
  191. package/dist/core/tools/executor.js +82 -0
  192. package/dist/core/tools/executor.js.map +1 -0
  193. package/dist/core/tools/executor.test.js +143 -0
  194. package/dist/core/tools/executor.test.js.map +1 -0
  195. package/dist/core/tools/index.js +4 -0
  196. package/dist/core/tools/index.js.map +1 -0
  197. package/dist/core/tools/schema.js +20 -0
  198. package/dist/core/tools/schema.js.map +1 -0
  199. package/dist/core/tools/state.js +29 -0
  200. package/dist/core/tools/state.js.map +1 -0
  201. package/dist/index.js +70 -0
  202. package/dist/index.js.map +1 -0
  203. package/dist/lib/json/index.js +5 -0
  204. package/dist/lib/json/index.js.map +1 -0
  205. package/dist/lib/logger/console.js +147 -0
  206. package/dist/lib/logger/console.js.map +1 -0
  207. package/dist/lib/logger/console.test.js +258 -0
  208. package/dist/lib/logger/console.test.js.map +1 -0
  209. package/dist/lib/logger/file.js +54 -0
  210. package/dist/lib/logger/file.js.map +1 -0
  211. package/dist/lib/logger/index.js +4 -0
  212. package/dist/lib/logger/index.js.map +1 -0
  213. package/dist/lib/logger/logger.js +28 -0
  214. package/dist/lib/logger/logger.js.map +1 -0
  215. package/dist/lib/logger/ring-buffer.js +61 -0
  216. package/dist/lib/logger/ring-buffer.js.map +1 -0
  217. package/dist/lib/logger/tee.js +43 -0
  218. package/dist/lib/logger/tee.js.map +1 -0
  219. package/dist/lib/mime.js +22 -0
  220. package/dist/lib/mime.js.map +1 -0
  221. package/dist/lib/never.js +4 -0
  222. package/dist/lib/never.js.map +1 -0
  223. package/dist/lib/utils/hash.js +35 -0
  224. package/dist/lib/utils/hash.js.map +1 -0
  225. package/dist/lib/utils/result.js +21 -0
  226. package/dist/lib/utils/result.js.map +1 -0
  227. package/dist/platform/fs.js +8 -0
  228. package/dist/platform/fs.js.map +1 -0
  229. package/dist/platform/index.js +9 -0
  230. package/dist/platform/index.js.map +1 -0
  231. package/dist/platform/process.js +8 -0
  232. package/dist/platform/process.js.map +1 -0
  233. package/dist/plugins/agent-status/plugin.js +77 -0
  234. package/dist/plugins/agent-status/plugin.js.map +1 -0
  235. package/dist/plugins/agents/agents.integration.test.js +683 -0
  236. package/dist/plugins/agents/agents.integration.test.js.map +1 -0
  237. package/dist/plugins/agents/index.js +2 -0
  238. package/dist/plugins/agents/index.js.map +1 -0
  239. package/dist/plugins/agents/plugin.js +199 -0
  240. package/dist/plugins/agents/plugin.js.map +1 -0
  241. package/dist/plugins/context-compact/context-compact.integration.test.js +174 -0
  242. package/dist/plugins/context-compact/context-compact.integration.test.js.map +1 -0
  243. package/dist/plugins/context-compact/context-compactor.js +238 -0
  244. package/dist/plugins/context-compact/context-compactor.js.map +1 -0
  245. package/dist/plugins/context-compact/context-compactor.test.js +763 -0
  246. package/dist/plugins/context-compact/context-compactor.test.js.map +1 -0
  247. package/dist/plugins/context-compact/history-offloader.js +42 -0
  248. package/dist/plugins/context-compact/history-offloader.js.map +1 -0
  249. package/dist/plugins/context-compact/history-offloader.test.js +77 -0
  250. package/dist/plugins/context-compact/history-offloader.test.js.map +1 -0
  251. package/dist/plugins/context-compact/index.js +4 -0
  252. package/dist/plugins/context-compact/index.js.map +1 -0
  253. package/dist/plugins/context-compact/plugin.js +37 -0
  254. package/dist/plugins/context-compact/plugin.js.map +1 -0
  255. package/dist/plugins/filesystem/filesystem.integration.test.js +411 -0
  256. package/dist/plugins/filesystem/filesystem.integration.test.js.map +1 -0
  257. package/dist/plugins/filesystem/helpers.js +170 -0
  258. package/dist/plugins/filesystem/helpers.js.map +1 -0
  259. package/dist/plugins/filesystem/index.js +3 -0
  260. package/dist/plugins/filesystem/index.js.map +1 -0
  261. package/dist/plugins/filesystem/listing.js +247 -0
  262. package/dist/plugins/filesystem/listing.js.map +1 -0
  263. package/dist/plugins/filesystem/plugin.js +364 -0
  264. package/dist/plugins/filesystem/plugin.js.map +1 -0
  265. package/dist/plugins/filesystem/schema.js +2 -0
  266. package/dist/plugins/filesystem/schema.js.map +1 -0
  267. package/dist/plugins/git-status/index.js +2 -0
  268. package/dist/plugins/git-status/index.js.map +1 -0
  269. package/dist/plugins/git-status/plugin.js +144 -0
  270. package/dist/plugins/git-status/plugin.js.map +1 -0
  271. package/dist/plugins/limits-guard/config.js +5 -0
  272. package/dist/plugins/limits-guard/config.js.map +1 -0
  273. package/dist/plugins/limits-guard/index.js +3 -0
  274. package/dist/plugins/limits-guard/index.js.map +1 -0
  275. package/dist/plugins/limits-guard/limit-guard.js +125 -0
  276. package/dist/plugins/limits-guard/limit-guard.js.map +1 -0
  277. package/dist/plugins/limits-guard/limit-guard.test.js +121 -0
  278. package/dist/plugins/limits-guard/limit-guard.test.js.map +1 -0
  279. package/dist/plugins/limits-guard/limits-guard.integration.test.js +378 -0
  280. package/dist/plugins/limits-guard/limits-guard.integration.test.js.map +1 -0
  281. package/dist/plugins/limits-guard/plugin.js +240 -0
  282. package/dist/plugins/limits-guard/plugin.js.map +1 -0
  283. package/dist/plugins/llm-debug/index.js +2 -0
  284. package/dist/plugins/llm-debug/index.js.map +1 -0
  285. package/dist/plugins/llm-debug/llm-debug.integration.test.js +157 -0
  286. package/dist/plugins/llm-debug/llm-debug.integration.test.js.map +1 -0
  287. package/dist/plugins/llm-debug/plugin.js +148 -0
  288. package/dist/plugins/llm-debug/plugin.js.map +1 -0
  289. package/dist/plugins/logs/index.js +2 -0
  290. package/dist/plugins/logs/index.js.map +1 -0
  291. package/dist/plugins/logs/plugin.js +38 -0
  292. package/dist/plugins/logs/plugin.js.map +1 -0
  293. package/dist/plugins/mailbox/helpers.js +66 -0
  294. package/dist/plugins/mailbox/helpers.js.map +1 -0
  295. package/dist/plugins/mailbox/index.js +9 -0
  296. package/dist/plugins/mailbox/index.js.map +1 -0
  297. package/dist/plugins/mailbox/mailbox.integration.test.js +605 -0
  298. package/dist/plugins/mailbox/mailbox.integration.test.js.map +1 -0
  299. package/dist/plugins/mailbox/plugin.js +204 -0
  300. package/dist/plugins/mailbox/plugin.js.map +1 -0
  301. package/dist/plugins/mailbox/prompts.js +93 -0
  302. package/dist/plugins/mailbox/prompts.js.map +1 -0
  303. package/dist/plugins/mailbox/query.js +38 -0
  304. package/dist/plugins/mailbox/query.js.map +1 -0
  305. package/dist/plugins/mailbox/schema.js +32 -0
  306. package/dist/plugins/mailbox/schema.js.map +1 -0
  307. package/dist/plugins/mailbox/state.js +41 -0
  308. package/dist/plugins/mailbox/state.js.map +1 -0
  309. package/dist/plugins/resources/index.js +4 -0
  310. package/dist/plugins/resources/index.js.map +1 -0
  311. package/dist/plugins/resources/manifest.js +20 -0
  312. package/dist/plugins/resources/manifest.js.map +1 -0
  313. package/dist/plugins/resources/plugin.js +171 -0
  314. package/dist/plugins/resources/plugin.js.map +1 -0
  315. package/dist/plugins/resources/post-inject.js +32 -0
  316. package/dist/plugins/resources/post-inject.js.map +1 -0
  317. package/dist/plugins/resources/state.js +16 -0
  318. package/dist/plugins/resources/state.js.map +1 -0
  319. package/dist/plugins/result-eviction/index.js +2 -0
  320. package/dist/plugins/result-eviction/index.js.map +1 -0
  321. package/dist/plugins/result-eviction/plugin.js +43 -0
  322. package/dist/plugins/result-eviction/plugin.js.map +1 -0
  323. package/dist/plugins/result-eviction/result-eviction.integration.test.js +217 -0
  324. package/dist/plugins/result-eviction/result-eviction.integration.test.js.map +1 -0
  325. package/dist/plugins/services/plugin.js +453 -0
  326. package/dist/plugins/services/plugin.js.map +1 -0
  327. package/dist/plugins/services/port-pool.js +70 -0
  328. package/dist/plugins/services/port-pool.js.map +1 -0
  329. package/dist/plugins/services/prompt.js +40 -0
  330. package/dist/plugins/services/prompt.js.map +1 -0
  331. package/dist/plugins/services/schema.js +9 -0
  332. package/dist/plugins/services/schema.js.map +1 -0
  333. package/dist/plugins/services/service.js +470 -0
  334. package/dist/plugins/services/service.js.map +1 -0
  335. package/dist/plugins/services/services.integration.test.js +485 -0
  336. package/dist/plugins/services/services.integration.test.js.map +1 -0
  337. package/dist/plugins/session-lifecycle/index.js +2 -0
  338. package/dist/plugins/session-lifecycle/index.js.map +1 -0
  339. package/dist/plugins/session-lifecycle/plugin.js +273 -0
  340. package/dist/plugins/session-lifecycle/plugin.js.map +1 -0
  341. package/dist/plugins/session-lifecycle/session-lifecycle.integration.test.js +498 -0
  342. package/dist/plugins/session-lifecycle/session-lifecycle.integration.test.js.map +1 -0
  343. package/dist/plugins/session-state/plugin.js +159 -0
  344. package/dist/plugins/session-state/plugin.js.map +1 -0
  345. package/dist/plugins/session-stats/index.js +3 -0
  346. package/dist/plugins/session-stats/index.js.map +1 -0
  347. package/dist/plugins/session-stats/plugin.js +81 -0
  348. package/dist/plugins/session-stats/plugin.js.map +1 -0
  349. package/dist/plugins/shell/executor.js +339 -0
  350. package/dist/plugins/shell/executor.js.map +1 -0
  351. package/dist/plugins/shell/index.js +6 -0
  352. package/dist/plugins/shell/index.js.map +1 -0
  353. package/dist/plugins/shell/plugin.js +66 -0
  354. package/dist/plugins/shell/plugin.js.map +1 -0
  355. package/dist/plugins/shell/shell.integration.test.js +234 -0
  356. package/dist/plugins/shell/shell.integration.test.js.map +1 -0
  357. package/dist/plugins/shell/shell.test.js +236 -0
  358. package/dist/plugins/shell/shell.test.js.map +1 -0
  359. package/dist/plugins/skills/discovery.js +205 -0
  360. package/dist/plugins/skills/discovery.js.map +1 -0
  361. package/dist/plugins/skills/discovery.test.js +312 -0
  362. package/dist/plugins/skills/discovery.test.js.map +1 -0
  363. package/dist/plugins/skills/index.js +12 -0
  364. package/dist/plugins/skills/index.js.map +1 -0
  365. package/dist/plugins/skills/plugin.js +293 -0
  366. package/dist/plugins/skills/plugin.js.map +1 -0
  367. package/dist/plugins/skills/prompts.js +70 -0
  368. package/dist/plugins/skills/prompts.js.map +1 -0
  369. package/dist/plugins/skills/schema.js +18 -0
  370. package/dist/plugins/skills/schema.js.map +1 -0
  371. package/dist/plugins/skills/skills.integration.test.js +475 -0
  372. package/dist/plugins/skills/skills.integration.test.js.map +1 -0
  373. package/dist/plugins/snapshotting/index.js +3 -0
  374. package/dist/plugins/snapshotting/index.js.map +1 -0
  375. package/dist/plugins/snapshotting/jj-snapshotter.js +106 -0
  376. package/dist/plugins/snapshotting/jj-snapshotter.js.map +1 -0
  377. package/dist/plugins/snapshotting/plugin.js +28 -0
  378. package/dist/plugins/snapshotting/plugin.js.map +1 -0
  379. package/dist/plugins/snapshotting/snapshotter.js +2 -0
  380. package/dist/plugins/snapshotting/snapshotter.js.map +1 -0
  381. package/dist/plugins/todo/index.js +7 -0
  382. package/dist/plugins/todo/index.js.map +1 -0
  383. package/dist/plugins/todo/plugin.js +319 -0
  384. package/dist/plugins/todo/plugin.js.map +1 -0
  385. package/dist/plugins/todo/prompts.js +54 -0
  386. package/dist/plugins/todo/prompts.js.map +1 -0
  387. package/dist/plugins/todo/schema.js +18 -0
  388. package/dist/plugins/todo/schema.js.map +1 -0
  389. package/dist/plugins/todo/todo.integration.test.js +605 -0
  390. package/dist/plugins/todo/todo.integration.test.js.map +1 -0
  391. package/dist/plugins/uploads/index.js +8 -0
  392. package/dist/plugins/uploads/index.js.map +1 -0
  393. package/dist/plugins/uploads/plugin.js +346 -0
  394. package/dist/plugins/uploads/plugin.js.map +1 -0
  395. package/dist/plugins/uploads/preprocessor.js +44 -0
  396. package/dist/plugins/uploads/preprocessor.js.map +1 -0
  397. package/dist/plugins/uploads/preprocessors/image-classifier.js +127 -0
  398. package/dist/plugins/uploads/preprocessors/image-classifier.js.map +1 -0
  399. package/dist/plugins/uploads/preprocessors/index.js +7 -0
  400. package/dist/plugins/uploads/preprocessors/index.js.map +1 -0
  401. package/dist/plugins/uploads/preprocessors/markitdown-preprocessor.js +204 -0
  402. package/dist/plugins/uploads/preprocessors/markitdown-preprocessor.js.map +1 -0
  403. package/dist/plugins/uploads/preprocessors/zip-preprocessor.js +172 -0
  404. package/dist/plugins/uploads/preprocessors/zip-preprocessor.js.map +1 -0
  405. package/dist/plugins/uploads/schema.js +20 -0
  406. package/dist/plugins/uploads/schema.js.map +1 -0
  407. package/dist/plugins/uploads/state.js +22 -0
  408. package/dist/plugins/uploads/state.js.map +1 -0
  409. package/dist/plugins/uploads/uploads.integration.test.js +496 -0
  410. package/dist/plugins/uploads/uploads.integration.test.js.map +1 -0
  411. package/dist/plugins/user-chat/index.js +5 -0
  412. package/dist/plugins/user-chat/index.js.map +1 -0
  413. package/dist/plugins/user-chat/plugin.js +544 -0
  414. package/dist/plugins/user-chat/plugin.js.map +1 -0
  415. package/dist/plugins/user-chat/prompts.js +29 -0
  416. package/dist/plugins/user-chat/prompts.js.map +1 -0
  417. package/dist/plugins/user-chat/schema.js +46 -0
  418. package/dist/plugins/user-chat/schema.js.map +1 -0
  419. package/dist/plugins/user-chat/user-chat.integration.test.js +668 -0
  420. package/dist/plugins/user-chat/user-chat.integration.test.js.map +1 -0
  421. package/dist/plugins/workers/context.js +143 -0
  422. package/dist/plugins/workers/context.js.map +1 -0
  423. package/dist/plugins/workers/definition.js +30 -0
  424. package/dist/plugins/workers/definition.js.map +1 -0
  425. package/dist/plugins/workers/index.js +7 -0
  426. package/dist/plugins/workers/index.js.map +1 -0
  427. package/dist/plugins/workers/plugin.js +578 -0
  428. package/dist/plugins/workers/plugin.js.map +1 -0
  429. package/dist/plugins/workers/worker.js +18 -0
  430. package/dist/plugins/workers/worker.js.map +1 -0
  431. package/dist/plugins/workers/workers.integration.test.js +629 -0
  432. package/dist/plugins/workers/workers.integration.test.js.map +1 -0
  433. package/dist/prompts/base.js +239 -0
  434. package/dist/prompts/base.js.map +1 -0
  435. package/dist/prompts/builder.js +131 -0
  436. package/dist/prompts/builder.js.map +1 -0
  437. package/dist/prompts/index.js +20 -0
  438. package/dist/prompts/index.js.map +1 -0
  439. package/dist/prompts/macros.js +26 -0
  440. package/dist/prompts/macros.js.map +1 -0
  441. package/dist/prompts/macros.test.js +80 -0
  442. package/dist/prompts/macros.test.js.map +1 -0
  443. package/dist/testing/bootstrap-for-testing.js +28 -0
  444. package/dist/testing/bootstrap-for-testing.js.map +1 -0
  445. package/dist/testing/index.js +7 -0
  446. package/dist/testing/index.js.map +1 -0
  447. package/dist/testing/node-platform.js +65 -0
  448. package/dist/testing/node-platform.js.map +1 -0
  449. package/dist/testing/notification-collector.js +82 -0
  450. package/dist/testing/notification-collector.js.map +1 -0
  451. package/dist/testing/preset-helpers.js +37 -0
  452. package/dist/testing/preset-helpers.js.map +1 -0
  453. package/dist/testing/test-harness.js +226 -0
  454. package/dist/testing/test-harness.js.map +1 -0
  455. package/dist/testing/test-harness.test.js +51 -0
  456. package/dist/testing/test-harness.test.js.map +1 -0
  457. package/dist/testing/wait-helpers.js +64 -0
  458. package/dist/testing/wait-helpers.js.map +1 -0
  459. package/dist/transport/adapter/client-adapter.js +64 -0
  460. package/dist/transport/adapter/client-adapter.js.map +1 -0
  461. package/dist/transport/adapter/index.js +24 -0
  462. package/dist/transport/adapter/index.js.map +1 -0
  463. package/dist/transport/adapter/server-adapter.js +73 -0
  464. package/dist/transport/adapter/server-adapter.js.map +1 -0
  465. package/dist/transport/adapter/types.js +8 -0
  466. package/dist/transport/adapter/types.js.map +1 -0
  467. package/dist/transport/http/app.js +86 -0
  468. package/dist/transport/http/app.js.map +1 -0
  469. package/dist/transport/http/index.js +6 -0
  470. package/dist/transport/http/index.js.map +1 -0
  471. package/dist/transport/http/middleware/bearer-auth.js +33 -0
  472. package/dist/transport/http/middleware/bearer-auth.js.map +1 -0
  473. package/dist/transport/http/middleware/error-handler.js +56 -0
  474. package/dist/transport/http/middleware/error-handler.js.map +1 -0
  475. package/dist/transport/http/routes/files.js +237 -0
  476. package/dist/transport/http/routes/files.js.map +1 -0
  477. package/dist/transport/http/routes/resources.js +77 -0
  478. package/dist/transport/http/routes/resources.js.map +1 -0
  479. package/dist/transport/http/routes/rpc.integration.test.js +189 -0
  480. package/dist/transport/http/routes/rpc.integration.test.js.map +1 -0
  481. package/dist/transport/http/routes/rpc.js +110 -0
  482. package/dist/transport/http/routes/rpc.js.map +1 -0
  483. package/dist/transport/http/routes/rpc.test.js +316 -0
  484. package/dist/transport/http/routes/rpc.test.js.map +1 -0
  485. package/dist/transport/http/routes/upload.js +205 -0
  486. package/dist/transport/http/routes/upload.js.map +1 -0
  487. package/dist/transport/rpc/index.js +7 -0
  488. package/dist/transport/rpc/index.js.map +1 -0
  489. package/dist/transport/rpc/methods.js +8 -0
  490. package/dist/transport/rpc/methods.js.map +1 -0
  491. package/dist/user-config.js +14 -0
  492. package/dist/user-config.js.map +1 -0
  493. package/package.json +47 -57
@@ -0,0 +1,1095 @@
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 { llmEvents } from '../../core/llm/state.js';
7
+ import { generateSessionId } from '../../core/sessions/schema.js';
8
+ import { sessionEvents } from '../../core/sessions/state.js';
9
+ import { generateToolCallId } from '../../core/tools/schema.js';
10
+ import { toolEvents } from '../../core/tools/state.js';
11
+ import { mailboxPlugin } from '../../plugins/mailbox/plugin.js';
12
+ import { getAgentMailbox, selectMailboxState } from '../../plugins/mailbox/query.js';
13
+ import { generateTestMessageId } from '../../plugins/mailbox/schema.js';
14
+ import { mailboxEvents } from '../../plugins/mailbox/state.js';
15
+ import { createApplyEvent } from './apply-event.js';
16
+ import { createSessionState, getAgentState, getCommunicatorId, getEntryAgentId, getOrchestratorId, reconstructSessionState } from './state.js';
17
+ const applyEvent = createApplyEvent([mailboxPlugin.create({})]);
18
+ const SESSION_ID = generateSessionId();
19
+ const TIMESTAMP = Date.now();
20
+ describe('createSessionState', () => {
21
+ it('creates an empty session with correct properties', () => {
22
+ const session = createSessionState(SESSION_ID, 'test-preset', TIMESTAMP);
23
+ expect(session.id).toBe(SESSION_ID);
24
+ expect(session.presetId).toBe('test-preset');
25
+ expect(session.status).toBe('active');
26
+ expect(session.agents.size).toBe(0);
27
+ expect(session.createdAt).toBe(TIMESTAMP);
28
+ expect(session.closedAt).toBeUndefined();
29
+ });
30
+ });
31
+ describe('applyEvent', () => {
32
+ const baseSession = createSessionState(SESSION_ID, 'test-preset', TIMESTAMP);
33
+ describe('session_closed', () => {
34
+ it('closes the session', () => {
35
+ const event = withSessionId(SESSION_ID, sessionEvents.create('session_closed', {}));
36
+ const result = applyEvent(baseSession, event);
37
+ expect(result.status).toBe('closed');
38
+ expect(result.closedAt).toBe(event.timestamp);
39
+ });
40
+ });
41
+ describe('agent_spawned', () => {
42
+ it('adds a new agent to the session', () => {
43
+ const agentId = generateTestAgentId();
44
+ const event = withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
45
+ agentId,
46
+ definitionName: ORCHESTRATOR_ROLE,
47
+ parentId: null,
48
+ }));
49
+ const result = applyEvent(baseSession, event);
50
+ expect(result.agents.size).toBe(1);
51
+ const agent = result.agents.get(agentId);
52
+ expect(agent).toBeDefined();
53
+ expect(agent.id).toBe(agentId);
54
+ expect(agent.definitionName).toBe(ORCHESTRATOR_ROLE);
55
+ expect(agent.parentId).toBeNull();
56
+ expect(agent.status).toBe('pending');
57
+ expect(getAgentMailbox(selectMailboxState(result), agentId)).toHaveLength(0);
58
+ });
59
+ it('adds child agent with parent reference', () => {
60
+ const parentId = generateTestAgentId();
61
+ const childId = generateTestAgentId();
62
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
63
+ agentId: parentId,
64
+ definitionName: ORCHESTRATOR_ROLE,
65
+ parentId: null,
66
+ })));
67
+ session = applyEvent(session, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
68
+ agentId: childId,
69
+ definitionName: 'worker',
70
+ parentId: parentId,
71
+ })));
72
+ const child = session.agents.get(childId);
73
+ expect(child.parentId).toBe(parentId);
74
+ });
75
+ });
76
+ describe('agent_state_changed', () => {
77
+ it('updates agent status', () => {
78
+ const agentId = generateTestAgentId();
79
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
80
+ agentId,
81
+ definitionName: 'test',
82
+ parentId: null,
83
+ })));
84
+ const event = withSessionId(SESSION_ID, agentEvents.create('agent_state_changed', {
85
+ agentId,
86
+ fromState: 'pending',
87
+ toState: 'inferring',
88
+ }));
89
+ session = applyEvent(session, event);
90
+ expect(session.agents.get(agentId).status).toBe('inferring');
91
+ });
92
+ it('throws for unknown agent', () => {
93
+ const unknownAgentId = generateTestAgentId();
94
+ const event = withSessionId(SESSION_ID, agentEvents.create('agent_state_changed', {
95
+ agentId: unknownAgentId,
96
+ fromState: 'pending',
97
+ toState: 'inferring',
98
+ }));
99
+ expect(() => applyEvent(baseSession, event)).toThrow(`Agent not found: ${unknownAgentId}`);
100
+ });
101
+ });
102
+ describe('mailbox_message', () => {
103
+ it('adds message to agent mailbox', () => {
104
+ const agentId = generateTestAgentId();
105
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
106
+ agentId,
107
+ definitionName: 'test',
108
+ parentId: null,
109
+ })));
110
+ const messageId = generateTestMessageId();
111
+ const event = withSessionId(SESSION_ID, mailboxEvents.create('mailbox_message', {
112
+ toAgentId: agentId,
113
+ message: {
114
+ id: messageId,
115
+ from: 'user',
116
+ content: 'Hello',
117
+ timestamp: TIMESTAMP,
118
+ consumed: false,
119
+ },
120
+ }));
121
+ session = applyEvent(session, event);
122
+ const mailbox = getAgentMailbox(selectMailboxState(session), agentId);
123
+ expect(mailbox).toHaveLength(1);
124
+ expect(mailbox[0].id).toBe(messageId);
125
+ expect(mailbox[0].content).toBe('Hello');
126
+ });
127
+ });
128
+ describe('mailbox_consumed', () => {
129
+ it('marks messages as consumed', () => {
130
+ const agentId = generateTestAgentId();
131
+ const messageId = generateTestMessageId();
132
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
133
+ agentId,
134
+ definitionName: 'test',
135
+ parentId: null,
136
+ })));
137
+ session = applyEvent(session, withSessionId(SESSION_ID, mailboxEvents.create('mailbox_message', {
138
+ toAgentId: agentId,
139
+ message: {
140
+ id: messageId,
141
+ from: 'user',
142
+ content: 'Hello',
143
+ timestamp: TIMESTAMP,
144
+ consumed: false,
145
+ },
146
+ })));
147
+ const event = withSessionId(SESSION_ID, mailboxEvents.create('mailbox_consumed', {
148
+ agentId,
149
+ messageIds: [messageId],
150
+ }));
151
+ session = applyEvent(session, event);
152
+ expect(getAgentMailbox(selectMailboxState(session), agentId)[0].consumed).toBe(true);
153
+ });
154
+ });
155
+ describe('inference_started', () => {
156
+ it('sets agent status to inferring', () => {
157
+ const agentId = generateTestAgentId();
158
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
159
+ agentId,
160
+ definitionName: 'test',
161
+ parentId: null,
162
+ })));
163
+ const event = withSessionId(SESSION_ID, llmEvents.create('inference_started', {
164
+ agentId,
165
+ messages: [],
166
+ consumedMessageIds: [],
167
+ }));
168
+ session = applyEvent(session, event);
169
+ expect(session.agents.get(agentId).status).toBe('inferring');
170
+ });
171
+ it('appends to pendingMessages (defense-in-depth against uncommitted messages)', () => {
172
+ // If for any reason pendingMessages are not empty when inference_started fires,
173
+ // they must not be overwritten — append instead.
174
+ const agentId = generateTestAgentId();
175
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
176
+ agentId,
177
+ definitionName: 'test',
178
+ parentId: null,
179
+ })));
180
+ const msg1 = { role: 'user', content: 'First message' };
181
+ const msg2 = { role: 'user', content: 'Second message' };
182
+ // First inference_started sets pendingMessages
183
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_started', {
184
+ agentId,
185
+ messages: [msg1],
186
+ consumedMessageIds: [],
187
+ })));
188
+ expect(session.agents.get(agentId).pendingMessages).toHaveLength(1);
189
+ // Second inference_started without inference_completed in between —
190
+ // must append, not overwrite
191
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_started', {
192
+ agentId,
193
+ messages: [msg2],
194
+ consumedMessageIds: [],
195
+ })));
196
+ const agent = session.agents.get(agentId);
197
+ expect(agent.pendingMessages).toHaveLength(2);
198
+ expect(agent.pendingMessages[0].content).toBe(msg1.content);
199
+ expect(agent.pendingMessages[1].content).toBe(msg2.content);
200
+ });
201
+ it('afterInference pause: inference_completed commits turn, then agent_paused preserves pause', () => {
202
+ // Real scenario: planner receives subagent response, LLM says "WAITING",
203
+ // limits-guard pauses. The correct event sequence is:
204
+ // inference_started → inference_completed → agent_paused
205
+ // This ensures conversationHistory includes the turn and pendingMessages are cleared.
206
+ const agentId = generateTestAgentId();
207
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
208
+ agentId,
209
+ definitionName: 'test',
210
+ parentId: null,
211
+ })));
212
+ const userMsg = { role: 'user', content: 'Response from collection-analyzer_1 (locations)' };
213
+ // 1. inference_started — message goes to pendingMessages
214
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_started', {
215
+ agentId,
216
+ messages: [userMsg],
217
+ consumedMessageIds: [],
218
+ })));
219
+ expect(session.agents.get(agentId).pendingMessages).toHaveLength(1);
220
+ expect(session.agents.get(agentId).status).toBe('inferring');
221
+ // 2. inference_completed — pendingMessages + response committed to conversationHistory
222
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
223
+ agentId,
224
+ consumedMessageIds: [],
225
+ response: { content: 'WAITING', toolCalls: [] },
226
+ metrics: { promptTokens: 200, completionTokens: 1, totalTokens: 201, latencyMs: 500, model: 'test' },
227
+ })));
228
+ expect(session.agents.get(agentId).pendingMessages).toHaveLength(0);
229
+ expect(session.agents.get(agentId).conversationHistory).toHaveLength(2); // user + assistant
230
+ expect(session.agents.get(agentId).status).toBe('pending');
231
+ // 3. agent_paused — status set to paused, conversationHistory preserved
232
+ session = applyEvent(session, withSessionId(SESSION_ID, agentEvents.create('agent_paused', {
233
+ agentId,
234
+ reason: 'handler',
235
+ message: 'No progress for 5 turns',
236
+ })));
237
+ const agent = session.agents.get(agentId);
238
+ expect(agent.status).toBe('paused');
239
+ expect(agent.pendingMessages).toHaveLength(0);
240
+ expect(agent.conversationHistory).toHaveLength(2);
241
+ expect(agent.conversationHistory[0].content).toBe(userMsg.content);
242
+ expect(agent.conversationHistory[1].content).toBe('WAITING');
243
+ // 4. Resume + next message — starts fresh with empty pendingMessages
244
+ session = applyEvent(session, withSessionId(SESSION_ID, agentEvents.create('agent_resumed', { agentId })));
245
+ const nextMsg = { role: 'user', content: 'Response from collection-analyzer_2 (team)' };
246
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_started', {
247
+ agentId,
248
+ messages: [nextMsg],
249
+ consumedMessageIds: [],
250
+ })));
251
+ // Only the new message in pendingMessages (previous turn was committed)
252
+ expect(session.agents.get(agentId).pendingMessages).toHaveLength(1);
253
+ expect(session.agents.get(agentId).pendingMessages[0].content).toBe(nextMsg.content);
254
+ // Previous turn preserved in conversationHistory
255
+ expect(session.agents.get(agentId).conversationHistory).toHaveLength(2);
256
+ });
257
+ });
258
+ describe('inference_completed', () => {
259
+ it('updates conversation history and status when no tool calls', () => {
260
+ const agentId = generateTestAgentId();
261
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
262
+ agentId,
263
+ definitionName: 'test',
264
+ parentId: null,
265
+ })));
266
+ const event = withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
267
+ agentId,
268
+ consumedMessageIds: [],
269
+ response: {
270
+ content: 'Hello there!',
271
+ toolCalls: [],
272
+ },
273
+ metrics: {
274
+ promptTokens: 100,
275
+ completionTokens: 10,
276
+ totalTokens: 110,
277
+ latencyMs: 500,
278
+ model: 'test-model',
279
+ },
280
+ }));
281
+ session = applyEvent(session, event);
282
+ const agent = session.agents.get(agentId);
283
+ expect(agent.status).toBe('pending');
284
+ expect(agent.conversationHistory).toHaveLength(1);
285
+ expect(agent.conversationHistory[0].role).toBe('assistant');
286
+ expect(agent.conversationHistory[0].content).toBe('Hello there!');
287
+ expect(agent.pendingToolCalls).toHaveLength(0);
288
+ });
289
+ it('marks consumed messages via mailbox_consumed event', () => {
290
+ const agentId = generateTestAgentId();
291
+ const messageId = generateTestMessageId();
292
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
293
+ agentId,
294
+ definitionName: 'test',
295
+ parentId: null,
296
+ })));
297
+ session = applyEvent(session, withSessionId(SESSION_ID, mailboxEvents.create('mailbox_message', {
298
+ toAgentId: agentId,
299
+ message: {
300
+ id: messageId,
301
+ from: 'user',
302
+ content: 'Hello',
303
+ timestamp: TIMESTAMP,
304
+ consumed: false,
305
+ },
306
+ })));
307
+ // Before consumption, message should be unconsumed
308
+ expect(getAgentMailbox(selectMailboxState(session), agentId)[0].consumed).toBe(false);
309
+ // Consume via mailbox_consumed (the dequeue mechanism)
310
+ session = applyEvent(session, withSessionId(SESSION_ID, mailboxEvents.create('mailbox_consumed', {
311
+ agentId,
312
+ messageIds: [messageId],
313
+ })));
314
+ // After mailbox_consumed, message should be consumed
315
+ expect(getAgentMailbox(selectMailboxState(session), agentId)[0].consumed).toBe(true);
316
+ // Inference still works after consumption
317
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
318
+ agentId,
319
+ consumedMessageIds: [],
320
+ response: {
321
+ content: 'Hi!',
322
+ toolCalls: [],
323
+ },
324
+ metrics: {
325
+ promptTokens: 100,
326
+ completionTokens: 10,
327
+ totalTokens: 110,
328
+ latencyMs: 500,
329
+ model: 'test-model',
330
+ },
331
+ })));
332
+ expect(session.agents.get(agentId).conversationHistory).toHaveLength(1);
333
+ });
334
+ it('sets tool_exec status when there are tool calls', () => {
335
+ const agentId = generateTestAgentId();
336
+ const toolCallId = generateToolCallId();
337
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
338
+ agentId,
339
+ definitionName: 'test',
340
+ parentId: null,
341
+ })));
342
+ const event = withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
343
+ agentId,
344
+ consumedMessageIds: [],
345
+ response: {
346
+ content: null,
347
+ toolCalls: [
348
+ {
349
+ id: toolCallId,
350
+ name: 'send_message',
351
+ input: { to: 'other', content: 'hi' },
352
+ },
353
+ ],
354
+ },
355
+ metrics: {
356
+ promptTokens: 100,
357
+ completionTokens: 20,
358
+ totalTokens: 120,
359
+ latencyMs: 600,
360
+ model: 'test-model',
361
+ },
362
+ }));
363
+ session = applyEvent(session, event);
364
+ const agent = session.agents.get(agentId);
365
+ expect(agent.status).toBe('tool_exec');
366
+ expect(agent.pendingToolCalls).toHaveLength(1);
367
+ expect(agent.pendingToolCalls[0].name).toBe('send_message');
368
+ });
369
+ });
370
+ describe('inference_failed', () => {
371
+ it('sets agent status to errored', () => {
372
+ const agentId = generateTestAgentId();
373
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
374
+ agentId,
375
+ definitionName: 'test',
376
+ parentId: null,
377
+ })));
378
+ const event = withSessionId(SESSION_ID, llmEvents.create('inference_failed', {
379
+ agentId,
380
+ error: 'API error',
381
+ }));
382
+ session = applyEvent(session, event);
383
+ expect(session.agents.get(agentId).status).toBe('errored');
384
+ });
385
+ it('clears pendingMessages but preserves pendingToolResults and mailbox for retry', () => {
386
+ const agentId = generateTestAgentId();
387
+ const messageId = generateTestMessageId();
388
+ const toolCallId = generateToolCallId();
389
+ // 1. Spawn agent
390
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
391
+ agentId,
392
+ definitionName: 'test',
393
+ parentId: null,
394
+ })));
395
+ // 2. Add mailbox message
396
+ session = applyEvent(session, withSessionId(SESSION_ID, mailboxEvents.create('mailbox_message', {
397
+ toAgentId: agentId,
398
+ message: {
399
+ id: messageId,
400
+ from: 'user',
401
+ content: 'Hello',
402
+ timestamp: TIMESTAMP,
403
+ consumed: false,
404
+ },
405
+ })));
406
+ // 3. First inference with tool call
407
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
408
+ agentId,
409
+ consumedMessageIds: [],
410
+ response: {
411
+ content: null,
412
+ toolCalls: [{ id: toolCallId, name: 'test_tool', input: {} }],
413
+ },
414
+ metrics: { promptTokens: 100, completionTokens: 10, totalTokens: 110, latencyMs: 500, model: 'test' },
415
+ })));
416
+ // 4. Tool completes - creates pendingToolResult
417
+ session = applyEvent(session, withSessionId(SESSION_ID, toolEvents.create('tool_completed', {
418
+ agentId,
419
+ toolCallId,
420
+ result: 'tool output',
421
+ })));
422
+ // 5. Start inference with pending messages
423
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_started', {
424
+ agentId,
425
+ messages: [
426
+ { role: 'tool', toolCallId, toolName: 'test_tool', content: 'tool output', isError: false, timestamp: TIMESTAMP },
427
+ { role: 'user', content: 'formatted mailbox', sourceMessageIds: [messageId] },
428
+ ],
429
+ consumedMessageIds: [messageId],
430
+ })));
431
+ // Verify pendingMessages are set
432
+ expect(session.agents.get(agentId).pendingMessages).toHaveLength(2);
433
+ expect(session.agents.get(agentId).status).toBe('inferring');
434
+ // 6. Inference fails
435
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_failed', {
436
+ agentId,
437
+ error: 'API timeout',
438
+ })));
439
+ const agent = session.agents.get(agentId);
440
+ // pendingMessages cleared
441
+ expect(agent.pendingMessages).toHaveLength(0);
442
+ // pendingToolResults preserved for rebuild
443
+ expect(agent.pendingToolResults).toHaveLength(1);
444
+ expect(agent.pendingToolResults[0].toolCallId).toBe(toolCallId);
445
+ // mailbox NOT consumed (can rebuild user message)
446
+ expect(getAgentMailbox(selectMailboxState(session), agentId)[0].consumed).toBe(false);
447
+ // status is errored
448
+ expect(agent.status).toBe('errored');
449
+ });
450
+ });
451
+ describe('session_restarted', () => {
452
+ it('clears pendingMessages for inferring agents, preserves pendingToolResults and mailbox', () => {
453
+ const agentId = generateTestAgentId();
454
+ const messageId = generateTestMessageId();
455
+ const toolCallId = generateToolCallId();
456
+ // 1. Spawn agent
457
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
458
+ agentId,
459
+ definitionName: 'test',
460
+ parentId: null,
461
+ })));
462
+ // 2. Add mailbox message
463
+ session = applyEvent(session, withSessionId(SESSION_ID, mailboxEvents.create('mailbox_message', {
464
+ toAgentId: agentId,
465
+ message: {
466
+ id: messageId,
467
+ from: 'user',
468
+ content: 'Hello',
469
+ timestamp: TIMESTAMP,
470
+ consumed: false,
471
+ },
472
+ })));
473
+ // 3. First inference with tool call
474
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
475
+ agentId,
476
+ consumedMessageIds: [],
477
+ response: {
478
+ content: null,
479
+ toolCalls: [{ id: toolCallId, name: 'test_tool', input: {} }],
480
+ },
481
+ metrics: { promptTokens: 100, completionTokens: 10, totalTokens: 110, latencyMs: 500, model: 'test' },
482
+ })));
483
+ // 4. Tool completes
484
+ session = applyEvent(session, withSessionId(SESSION_ID, toolEvents.create('tool_completed', {
485
+ agentId,
486
+ toolCallId,
487
+ result: 'tool output',
488
+ })));
489
+ // 5. Start inference
490
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_started', {
491
+ agentId,
492
+ messages: [
493
+ { role: 'tool', toolCallId, toolName: 'test_tool', content: 'tool output', isError: false, timestamp: TIMESTAMP },
494
+ { role: 'user', content: 'formatted mailbox', sourceMessageIds: [messageId] },
495
+ ],
496
+ consumedMessageIds: [messageId],
497
+ })));
498
+ // Verify state before restart
499
+ expect(session.agents.get(agentId).pendingMessages).toHaveLength(2);
500
+ expect(session.agents.get(agentId).status).toBe('inferring');
501
+ // 6. Server restarts
502
+ session = applyEvent(session, {
503
+ ...withSessionId(SESSION_ID, sessionEvents.create('session_restarted', {
504
+ resetAgentIds: [agentId],
505
+ clearedToolAgentIds: [],
506
+ })),
507
+ timestamp: TIMESTAMP + 1000,
508
+ });
509
+ const agent = session.agents.get(agentId);
510
+ // pendingMessages cleared
511
+ expect(agent.pendingMessages).toHaveLength(0);
512
+ // pendingToolResults preserved - can rebuild tool messages
513
+ expect(agent.pendingToolResults).toHaveLength(1);
514
+ expect(agent.pendingToolResults[0].toolCallId).toBe(toolCallId);
515
+ // mailbox NOT consumed - can rebuild user message
516
+ expect(getAgentMailbox(selectMailboxState(session), agentId)[0].consumed).toBe(false);
517
+ // status reset to pending
518
+ expect(agent.status).toBe('pending');
519
+ });
520
+ it('allows successful retry after restart - full cycle', () => {
521
+ const agentId = generateTestAgentId();
522
+ const messageId = generateTestMessageId();
523
+ const toolCallId = generateToolCallId();
524
+ // 1. Setup: spawn, message, tool call, tool complete
525
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
526
+ agentId,
527
+ definitionName: 'test',
528
+ parentId: null,
529
+ })));
530
+ session = applyEvent(session, withSessionId(SESSION_ID, mailboxEvents.create('mailbox_message', {
531
+ toAgentId: agentId,
532
+ message: { id: messageId, from: 'user', content: 'Hello', timestamp: TIMESTAMP, consumed: false },
533
+ })));
534
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
535
+ agentId,
536
+ consumedMessageIds: [],
537
+ response: { content: null, toolCalls: [{ id: toolCallId, name: 'test_tool', input: {} }] },
538
+ metrics: { promptTokens: 100, completionTokens: 10, totalTokens: 110, latencyMs: 500, model: 'test' },
539
+ })));
540
+ session = applyEvent(session, withSessionId(SESSION_ID, toolEvents.create('tool_completed', {
541
+ agentId,
542
+ toolCallId,
543
+ result: 'tool output',
544
+ })));
545
+ // 2. First inference attempt - interrupted by restart
546
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_started', {
547
+ agentId,
548
+ messages: [
549
+ { role: 'tool', toolCallId, toolName: 'test_tool', content: 'tool output', isError: false, timestamp: TIMESTAMP },
550
+ { role: 'user', content: 'formatted mailbox v1', sourceMessageIds: [messageId] },
551
+ ],
552
+ consumedMessageIds: [messageId],
553
+ })));
554
+ session = applyEvent(session, {
555
+ ...withSessionId(SESSION_ID, sessionEvents.create('session_restarted', {
556
+ resetAgentIds: [agentId],
557
+ clearedToolAgentIds: [],
558
+ })),
559
+ timestamp: TIMESTAMP + 1000,
560
+ });
561
+ // 3. Retry inference - rebuild messages from preserved state
562
+ const agentAfterRestart = session.agents.get(agentId);
563
+ expect(agentAfterRestart.status).toBe('pending');
564
+ expect(agentAfterRestart.pendingToolResults).toHaveLength(1);
565
+ expect(getAgentMailbox(selectMailboxState(session), agentId)[0].consumed).toBe(false);
566
+ // Runtime would rebuild: tool message from pendingToolResults + user message from unconsumed mailbox
567
+ session = applyEvent(session, {
568
+ ...withSessionId(SESSION_ID, llmEvents.create('inference_started', {
569
+ agentId,
570
+ messages: [
571
+ { role: 'tool', toolCallId, toolName: 'test_tool', content: 'tool output', isError: false, timestamp: TIMESTAMP },
572
+ { role: 'user', content: 'formatted mailbox v2', sourceMessageIds: [messageId] },
573
+ ],
574
+ consumedMessageIds: [messageId],
575
+ })),
576
+ timestamp: TIMESTAMP + 2000,
577
+ });
578
+ // 4. Dequeue mechanism marks messages consumed
579
+ session = applyEvent(session, withSessionId(SESSION_ID, mailboxEvents.create('mailbox_consumed', {
580
+ agentId,
581
+ messageIds: [messageId],
582
+ })));
583
+ // 5. This time inference completes successfully
584
+ session = applyEvent(session, {
585
+ ...withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
586
+ agentId,
587
+ consumedMessageIds: [],
588
+ response: { content: 'Done!', toolCalls: [] },
589
+ metrics: { promptTokens: 100, completionTokens: 10, totalTokens: 110, latencyMs: 500, model: 'test' },
590
+ })),
591
+ timestamp: TIMESTAMP + 3000,
592
+ });
593
+ const finalAgent = session.agents.get(agentId);
594
+ // Everything committed
595
+ expect(finalAgent.status).toBe('pending');
596
+ expect(finalAgent.pendingMessages).toHaveLength(0);
597
+ expect(finalAgent.pendingToolResults).toHaveLength(0);
598
+ expect(getAgentMailbox(selectMailboxState(session), agentId)[0].consumed).toBe(true);
599
+ // History contains: assistant (tool call) + tool + user + assistant (final)
600
+ expect(finalAgent.conversationHistory).toHaveLength(4);
601
+ expect(finalAgent.conversationHistory[0].role).toBe('assistant');
602
+ expect(finalAgent.conversationHistory[1].role).toBe('tool');
603
+ expect(finalAgent.conversationHistory[2].role).toBe('user');
604
+ expect(finalAgent.conversationHistory[3].role).toBe('assistant');
605
+ expect(finalAgent.conversationHistory[3].content).toBe('Done!');
606
+ });
607
+ });
608
+ describe('tool_completed', () => {
609
+ it('removes tool call from pending and updates status', () => {
610
+ const agentId = generateTestAgentId();
611
+ const toolCallId = generateToolCallId();
612
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
613
+ agentId,
614
+ definitionName: 'test',
615
+ parentId: null,
616
+ })));
617
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
618
+ agentId,
619
+ consumedMessageIds: [],
620
+ response: {
621
+ content: null,
622
+ toolCalls: [{ id: toolCallId, name: 'test', input: {} }],
623
+ },
624
+ metrics: {
625
+ promptTokens: 100,
626
+ completionTokens: 10,
627
+ totalTokens: 110,
628
+ latencyMs: 500,
629
+ model: 'test-model',
630
+ },
631
+ })));
632
+ const event = withSessionId(SESSION_ID, toolEvents.create('tool_completed', {
633
+ agentId,
634
+ toolCallId,
635
+ result: 'success',
636
+ }));
637
+ session = applyEvent(session, event);
638
+ const agent = session.agents.get(agentId);
639
+ expect(agent.pendingToolCalls).toHaveLength(0);
640
+ expect(agent.status).toBe('pending');
641
+ });
642
+ it('keeps tool_exec status when there are remaining tool calls', () => {
643
+ const agentId = generateTestAgentId();
644
+ const toolCallId1 = generateToolCallId();
645
+ const toolCallId2 = generateToolCallId();
646
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
647
+ agentId,
648
+ definitionName: 'test',
649
+ parentId: null,
650
+ })));
651
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
652
+ agentId,
653
+ consumedMessageIds: [],
654
+ response: {
655
+ content: null,
656
+ toolCalls: [
657
+ { id: toolCallId1, name: 'test1', input: {} },
658
+ { id: toolCallId2, name: 'test2', input: {} },
659
+ ],
660
+ },
661
+ metrics: {
662
+ promptTokens: 100,
663
+ completionTokens: 10,
664
+ totalTokens: 110,
665
+ latencyMs: 500,
666
+ model: 'test-model',
667
+ },
668
+ })));
669
+ session = applyEvent(session, withSessionId(SESSION_ID, toolEvents.create('tool_completed', {
670
+ agentId,
671
+ toolCallId: toolCallId1,
672
+ result: 'success',
673
+ })));
674
+ const agent = session.agents.get(agentId);
675
+ expect(agent.pendingToolCalls).toHaveLength(1);
676
+ expect(agent.status).toBe('tool_exec');
677
+ });
678
+ it('stores content in pendingToolResults (deferred to history by inference_completed)', () => {
679
+ const agentId = generateTestAgentId();
680
+ const toolCallId = generateToolCallId();
681
+ const toolTimestamp = Date.now();
682
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
683
+ agentId,
684
+ definitionName: 'test',
685
+ parentId: null,
686
+ })));
687
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
688
+ agentId,
689
+ consumedMessageIds: [],
690
+ response: {
691
+ content: null,
692
+ toolCalls: [{ id: toolCallId, name: 'send_message', input: { to: 'agent-1', message: 'hello' } }],
693
+ },
694
+ metrics: {
695
+ promptTokens: 100,
696
+ completionTokens: 10,
697
+ totalTokens: 110,
698
+ latencyMs: 500,
699
+ model: 'test-model',
700
+ },
701
+ })));
702
+ session = applyEvent(session, {
703
+ ...withSessionId(SESSION_ID, toolEvents.create('tool_completed', {
704
+ agentId,
705
+ toolCallId,
706
+ result: 'success message',
707
+ })),
708
+ timestamp: toolTimestamp,
709
+ });
710
+ const agent = session.agents.get(agentId);
711
+ // Tool result should NOT be in conversationHistory yet (deferred pattern)
712
+ const toolMessage = agent.conversationHistory.find(m => m.role === 'tool');
713
+ expect(toolMessage).toBeUndefined();
714
+ // Check pendingToolResults has full content
715
+ expect(agent.pendingToolResults).toHaveLength(1);
716
+ expect(agent.pendingToolResults[0].toolCallId).toBe(toolCallId);
717
+ expect(agent.pendingToolResults[0].toolName).toBe('send_message');
718
+ expect(agent.pendingToolResults[0].timestamp).toBe(toolTimestamp);
719
+ expect(agent.pendingToolResults[0].isError).toBe(false);
720
+ expect(agent.pendingToolResults[0].content).toBe('success message');
721
+ });
722
+ it('preserves rich content (array) in pendingToolResults', () => {
723
+ const agentId = generateTestAgentId();
724
+ const toolCallId = generateToolCallId();
725
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
726
+ agentId,
727
+ definitionName: 'test',
728
+ parentId: null,
729
+ })));
730
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
731
+ agentId,
732
+ consumedMessageIds: [],
733
+ response: {
734
+ content: null,
735
+ toolCalls: [{ id: toolCallId, name: 'read_file', input: { path: '/test' } }],
736
+ },
737
+ metrics: {
738
+ promptTokens: 100,
739
+ completionTokens: 10,
740
+ totalTokens: 110,
741
+ latencyMs: 500,
742
+ model: 'test-model',
743
+ },
744
+ })));
745
+ // Rich content array (like image + text)
746
+ const richContent = [
747
+ { type: 'text', text: 'File contents:' },
748
+ { type: 'image_url', imageUrl: { url: 'data:image/png;base64,abc123' } },
749
+ ];
750
+ session = applyEvent(session, withSessionId(SESSION_ID, toolEvents.create('tool_completed', {
751
+ agentId,
752
+ toolCallId,
753
+ result: richContent,
754
+ })));
755
+ const agent = session.agents.get(agentId);
756
+ // Rich content should be preserved (not stringified)
757
+ expect(agent.pendingToolResults[0].content).toEqual(richContent);
758
+ });
759
+ it('inference_started does NOT clear pendingToolResults (deferred to inference_completed)', () => {
760
+ const agentId = generateTestAgentId();
761
+ const toolCallId = generateToolCallId();
762
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
763
+ agentId,
764
+ definitionName: 'test',
765
+ parentId: null,
766
+ })));
767
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
768
+ agentId,
769
+ consumedMessageIds: [],
770
+ response: {
771
+ content: null,
772
+ toolCalls: [{ id: toolCallId, name: 'send_message', input: {} }],
773
+ },
774
+ metrics: {
775
+ promptTokens: 100,
776
+ completionTokens: 10,
777
+ totalTokens: 110,
778
+ latencyMs: 500,
779
+ model: 'test-model',
780
+ },
781
+ })));
782
+ session = applyEvent(session, withSessionId(SESSION_ID, toolEvents.create('tool_completed', {
783
+ agentId,
784
+ toolCallId,
785
+ result: 'success',
786
+ })));
787
+ // Verify pendingToolResults has entry
788
+ expect(session.agents.get(agentId).pendingToolResults).toHaveLength(1);
789
+ // Start new inference - should NOT clear pendingToolResults
790
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_started', {
791
+ agentId,
792
+ messages: [
793
+ { role: 'tool', toolCallId, toolName: 'send_message', content: 'success', isError: false, timestamp: TIMESTAMP },
794
+ ],
795
+ consumedMessageIds: [],
796
+ })));
797
+ // pendingToolResults preserved until inference_completed
798
+ expect(session.agents.get(agentId).pendingToolResults).toHaveLength(1);
799
+ });
800
+ it('inference_completed moves pendingToolResults to conversationHistory', () => {
801
+ const agentId = generateTestAgentId();
802
+ const toolCallId = generateToolCallId();
803
+ const toolTimestamp = TIMESTAMP + 1000;
804
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
805
+ agentId,
806
+ definitionName: 'test',
807
+ parentId: null,
808
+ })));
809
+ // First inference with tool call
810
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
811
+ agentId,
812
+ consumedMessageIds: [],
813
+ response: {
814
+ content: null,
815
+ toolCalls: [{ id: toolCallId, name: 'send_message', input: {} }],
816
+ },
817
+ metrics: {
818
+ promptTokens: 100,
819
+ completionTokens: 10,
820
+ totalTokens: 110,
821
+ latencyMs: 500,
822
+ model: 'test-model',
823
+ },
824
+ })));
825
+ // Tool completes
826
+ session = applyEvent(session, {
827
+ ...withSessionId(SESSION_ID, toolEvents.create('tool_completed', {
828
+ agentId,
829
+ toolCallId,
830
+ result: 'tool result content',
831
+ })),
832
+ timestamp: toolTimestamp,
833
+ });
834
+ // Verify: not in history yet, in pendingToolResults
835
+ let agent = session.agents.get(agentId);
836
+ expect(agent.conversationHistory.filter(m => m.role === 'tool')).toHaveLength(0);
837
+ expect(agent.pendingToolResults).toHaveLength(1);
838
+ // Start and complete new inference
839
+ session = applyEvent(session, {
840
+ ...withSessionId(SESSION_ID, llmEvents.create('inference_started', {
841
+ agentId,
842
+ messages: [
843
+ { role: 'tool', toolCallId, toolName: 'send_message', content: 'tool result content', isError: false, timestamp: toolTimestamp },
844
+ ],
845
+ consumedMessageIds: [],
846
+ })),
847
+ timestamp: TIMESTAMP + 2000,
848
+ });
849
+ session = applyEvent(session, {
850
+ ...withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
851
+ agentId,
852
+ consumedMessageIds: [],
853
+ response: {
854
+ content: 'Response after tool',
855
+ toolCalls: [],
856
+ },
857
+ metrics: {
858
+ promptTokens: 100,
859
+ completionTokens: 10,
860
+ totalTokens: 110,
861
+ latencyMs: 500,
862
+ model: 'test-model',
863
+ },
864
+ })),
865
+ timestamp: TIMESTAMP + 3000,
866
+ });
867
+ // Verify: pendingToolResults moved to history, then cleared
868
+ agent = session.agents.get(agentId);
869
+ expect(agent.pendingToolResults).toHaveLength(0);
870
+ const toolMessages = agent.conversationHistory.filter(m => m.role === 'tool');
871
+ expect(toolMessages).toHaveLength(1);
872
+ const toolMessage = toolMessages[0];
873
+ if (toolMessage.role === 'tool') {
874
+ expect(toolMessage.toolCallId).toBe(toolCallId);
875
+ expect(toolMessage.toolName).toBe('send_message');
876
+ expect(toolMessage.content).toBe('tool result content');
877
+ expect(toolMessage.timestamp).toBe(toolTimestamp);
878
+ expect(toolMessage.isError).toBe(false);
879
+ }
880
+ });
881
+ });
882
+ describe('tool_failed', () => {
883
+ it('removes tool call from pending', () => {
884
+ const agentId = generateTestAgentId();
885
+ const toolCallId = generateToolCallId();
886
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
887
+ agentId,
888
+ definitionName: 'test',
889
+ parentId: null,
890
+ })));
891
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
892
+ agentId,
893
+ consumedMessageIds: [],
894
+ response: {
895
+ content: null,
896
+ toolCalls: [{ id: toolCallId, name: 'test', input: {} }],
897
+ },
898
+ metrics: {
899
+ promptTokens: 100,
900
+ completionTokens: 10,
901
+ totalTokens: 110,
902
+ latencyMs: 500,
903
+ model: 'test-model',
904
+ },
905
+ })));
906
+ const event = withSessionId(SESSION_ID, toolEvents.create('tool_failed', {
907
+ agentId,
908
+ toolCallId,
909
+ error: 'Tool error',
910
+ }));
911
+ session = applyEvent(session, event);
912
+ const agent = session.agents.get(agentId);
913
+ expect(agent.pendingToolCalls).toHaveLength(0);
914
+ expect(agent.status).toBe('pending');
915
+ });
916
+ it('stores error in pendingToolResults with isError=true (deferred to history by inference_completed)', () => {
917
+ const agentId = generateTestAgentId();
918
+ const toolCallId = generateToolCallId();
919
+ const toolTimestamp = Date.now();
920
+ let session = applyEvent(baseSession, withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
921
+ agentId,
922
+ definitionName: 'test',
923
+ parentId: null,
924
+ })));
925
+ session = applyEvent(session, withSessionId(SESSION_ID, llmEvents.create('inference_completed', {
926
+ agentId,
927
+ consumedMessageIds: [],
928
+ response: {
929
+ content: null,
930
+ toolCalls: [{ id: toolCallId, name: 'reveal_secret', input: { password: 'test' } }],
931
+ },
932
+ metrics: {
933
+ promptTokens: 100,
934
+ completionTokens: 10,
935
+ totalTokens: 110,
936
+ latencyMs: 500,
937
+ model: 'test-model',
938
+ },
939
+ })));
940
+ session = applyEvent(session, {
941
+ ...withSessionId(SESSION_ID, toolEvents.create('tool_failed', {
942
+ agentId,
943
+ toolCallId,
944
+ error: 'Invalid password',
945
+ })),
946
+ timestamp: toolTimestamp,
947
+ });
948
+ const agent = session.agents.get(agentId);
949
+ // Tool result should NOT be in conversationHistory yet (deferred pattern)
950
+ const toolMessage = agent.conversationHistory.find(m => m.role === 'tool');
951
+ expect(toolMessage).toBeUndefined();
952
+ // Check pendingToolResults has error content with isError=true
953
+ expect(agent.pendingToolResults).toHaveLength(1);
954
+ expect(agent.pendingToolResults[0].toolCallId).toBe(toolCallId);
955
+ expect(agent.pendingToolResults[0].toolName).toBe('reveal_secret');
956
+ expect(agent.pendingToolResults[0].timestamp).toBe(toolTimestamp);
957
+ expect(agent.pendingToolResults[0].isError).toBe(true);
958
+ expect(agent.pendingToolResults[0].content).toBe('Invalid password');
959
+ });
960
+ });
961
+ });
962
+ describe('reconstructSessionState', () => {
963
+ it('returns null for empty events array', () => {
964
+ expect(reconstructSessionState([], applyEvent)).toBeNull();
965
+ });
966
+ it('throws error when first event is not session_created', () => {
967
+ const events = [
968
+ withSessionId(SESSION_ID, sessionEvents.create('session_closed', {})),
969
+ ];
970
+ expect(() => reconstructSessionState(events, applyEvent)).toThrow('First event must be session_created');
971
+ });
972
+ it('reconstructs session from events', () => {
973
+ const agentId = generateTestAgentId();
974
+ const events = [
975
+ withSessionId(SESSION_ID, sessionEvents.create('session_created', { presetId: 'test-preset' })),
976
+ withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
977
+ agentId,
978
+ definitionName: ORCHESTRATOR_ROLE,
979
+ parentId: null,
980
+ })),
981
+ ];
982
+ const session = reconstructSessionState(events, applyEvent);
983
+ expect(session).not.toBeNull();
984
+ expect(session.id).toBe(SESSION_ID);
985
+ expect(session.agents.size).toBe(1);
986
+ expect(session.agents.get(agentId).definitionName).toBe(ORCHESTRATOR_ROLE);
987
+ });
988
+ });
989
+ describe('Query helpers', () => {
990
+ describe('getOrchestratorId', () => {
991
+ it('returns orchestrator agent id', () => {
992
+ const orchestratorId = generateTestAgentId();
993
+ const events = [
994
+ withSessionId(SESSION_ID, sessionEvents.create('session_created', { presetId: 'test' })),
995
+ withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
996
+ agentId: orchestratorId,
997
+ definitionName: ORCHESTRATOR_ROLE,
998
+ parentId: null,
999
+ })),
1000
+ ];
1001
+ const session = reconstructSessionState(events, applyEvent);
1002
+ expect(getOrchestratorId(session)).toBe(orchestratorId);
1003
+ });
1004
+ it('returns null when no orchestrator', () => {
1005
+ const session = createSessionState(SESSION_ID, 'test', TIMESTAMP);
1006
+ expect(getOrchestratorId(session)).toBeNull();
1007
+ });
1008
+ });
1009
+ describe('getCommunicatorId', () => {
1010
+ it('returns communicator agent id', () => {
1011
+ const communicatorId = generateTestAgentId();
1012
+ const events = [
1013
+ withSessionId(SESSION_ID, sessionEvents.create('session_created', { presetId: 'test' })),
1014
+ withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
1015
+ agentId: communicatorId,
1016
+ definitionName: COMMUNICATOR_ROLE,
1017
+ parentId: null,
1018
+ })),
1019
+ ];
1020
+ const session = reconstructSessionState(events, applyEvent);
1021
+ expect(getCommunicatorId(session)).toBe(communicatorId);
1022
+ });
1023
+ });
1024
+ describe('getEntryAgentId', () => {
1025
+ it('returns communicator when both exist', () => {
1026
+ const orchestratorId = generateTestAgentId();
1027
+ const communicatorId = generateTestAgentId();
1028
+ const events = [
1029
+ withSessionId(SESSION_ID, sessionEvents.create('session_created', { presetId: 'test' })),
1030
+ withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
1031
+ agentId: orchestratorId,
1032
+ definitionName: ORCHESTRATOR_ROLE,
1033
+ parentId: null,
1034
+ })),
1035
+ withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
1036
+ agentId: communicatorId,
1037
+ definitionName: COMMUNICATOR_ROLE,
1038
+ parentId: orchestratorId,
1039
+ })),
1040
+ ];
1041
+ const session = reconstructSessionState(events, applyEvent);
1042
+ expect(getEntryAgentId(session)).toBe(communicatorId);
1043
+ });
1044
+ it('returns orchestrator when no communicator', () => {
1045
+ const orchestratorId = generateTestAgentId();
1046
+ const events = [
1047
+ withSessionId(SESSION_ID, sessionEvents.create('session_created', { presetId: 'test' })),
1048
+ withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
1049
+ agentId: orchestratorId,
1050
+ definitionName: ORCHESTRATOR_ROLE,
1051
+ parentId: null,
1052
+ })),
1053
+ ];
1054
+ const session = reconstructSessionState(events, applyEvent);
1055
+ expect(getEntryAgentId(session)).toBe(orchestratorId);
1056
+ });
1057
+ });
1058
+ describe('getAgentState', () => {
1059
+ it('returns agent by id', () => {
1060
+ const agentId = generateTestAgentId();
1061
+ const events = [
1062
+ withSessionId(SESSION_ID, sessionEvents.create('session_created', { presetId: 'test' })),
1063
+ withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
1064
+ agentId,
1065
+ definitionName: 'test',
1066
+ parentId: null,
1067
+ })),
1068
+ ];
1069
+ const session = reconstructSessionState(events, applyEvent);
1070
+ const agent = getAgentState(session, agentId);
1071
+ expect(agent).not.toBeNull();
1072
+ expect(agent.id).toBe(agentId);
1073
+ });
1074
+ it('returns null for unknown agent', () => {
1075
+ const session = createSessionState(SESSION_ID, 'test', TIMESTAMP);
1076
+ expect(getAgentState(session, generateTestAgentId())).toBeNull();
1077
+ });
1078
+ });
1079
+ });
1080
+ describe('Immutability', () => {
1081
+ it('applyEvent does not mutate original session', () => {
1082
+ const session = createSessionState(SESSION_ID, 'test', TIMESTAMP);
1083
+ const agentId = generateTestAgentId();
1084
+ const event = withSessionId(SESSION_ID, agentEvents.create('agent_spawned', {
1085
+ agentId,
1086
+ definitionName: 'test',
1087
+ parentId: null,
1088
+ }));
1089
+ const newSession = applyEvent(session, event);
1090
+ expect(session.agents.size).toBe(0);
1091
+ expect(newSession.agents.size).toBe(1);
1092
+ expect(session).not.toBe(newSession);
1093
+ });
1094
+ });
1095
+ //# sourceMappingURL=session.test.js.map