@roj-ai/sdk 0.1.1 → 0.1.4

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,378 @@
1
+ import { describe, expect, it } from 'bun:test';
2
+ import { AgentId } from '../../core/agents/schema.js';
3
+ import { agentEvents } from '../../core/agents/state.js';
4
+ import { MockLLMProvider } from '../../core/llm/mock.js';
5
+ import { selectPluginState } from '../../core/sessions/reducer.js';
6
+ import { ToolCallId } from '../../core/tools/schema.js';
7
+ import { createMultiAgentPreset, createTestPreset, TestHarness } from '../../testing/index.js';
8
+ import { limitsGuardPlugin } from './plugin.js';
9
+ function createLimitsHarness(options) {
10
+ return new TestHarness({ ...options, systemPlugins: [limitsGuardPlugin] });
11
+ }
12
+ /**
13
+ * Wait for a specific agent to reach 'paused' status (or timeout).
14
+ * Polls the TestSession's state directly (live getter).
15
+ */
16
+ async function waitForAgentPaused(session, agentId, timeoutMs = 10000) {
17
+ const deadline = Date.now() + timeoutMs;
18
+ while (Date.now() < deadline) {
19
+ const agentState = session.state.agents.get(agentId);
20
+ if (agentState?.status === 'paused')
21
+ return;
22
+ await new Promise(r => setTimeout(r, 10));
23
+ }
24
+ throw new Error(`waitForAgentPaused timed out after ${timeoutMs}ms for agent ${agentId}`);
25
+ }
26
+ /**
27
+ * Wait for all agents to be either idle (pending with no work) or paused.
28
+ */
29
+ async function waitForAllSettled(session, timeoutMs = 10000) {
30
+ const deadline = Date.now() + timeoutMs;
31
+ outer: while (Date.now() < deadline) {
32
+ for (const [, agentState] of session.state.agents) {
33
+ if (agentState.status === 'paused')
34
+ continue;
35
+ if (agentState.status === 'pending' && agentState.pendingToolCalls.length === 0 && agentState.pendingToolResults.length === 0) {
36
+ continue;
37
+ }
38
+ await new Promise(r => setTimeout(r, 10));
39
+ continue outer;
40
+ }
41
+ // Double-check after a brief delay
42
+ await new Promise(r => setTimeout(r, 10));
43
+ for (const [, agentState] of session.state.agents) {
44
+ if (agentState.status === 'paused')
45
+ continue;
46
+ if (agentState.status === 'pending' && agentState.pendingToolCalls.length === 0 && agentState.pendingToolResults.length === 0) {
47
+ continue;
48
+ }
49
+ continue outer;
50
+ }
51
+ return;
52
+ }
53
+ throw new Error(`waitForAllSettled timed out after ${timeoutMs}ms`);
54
+ }
55
+ describe('limits-guard plugin', () => {
56
+ // =========================================================================
57
+ // hard limits
58
+ // =========================================================================
59
+ describe('hard limits', () => {
60
+ it('agent exceeding inference hard limit → agent_paused event → agent stops', async () => {
61
+ let inferenceCount = 0;
62
+ const harness = createLimitsHarness({
63
+ presets: [createTestPreset({
64
+ orchestratorSystem: 'Test agent.',
65
+ orchestratorPlugins: [limitsGuardPlugin.configureAgent({ limits: { maxTurns: 3 } })],
66
+ })],
67
+ mockHandler: () => {
68
+ inferenceCount++;
69
+ return {
70
+ content: null,
71
+ toolCalls: [{ id: ToolCallId(`tc${inferenceCount}`), name: 'tell_user', input: { message: `Turn ${inferenceCount}` } }],
72
+ finishReason: 'stop',
73
+ metrics: MockLLMProvider.defaultMetrics(),
74
+ };
75
+ },
76
+ });
77
+ const session = await harness.createSession('test');
78
+ const entryAgentId = session.getEntryAgentId();
79
+ await session.sendMessage('Start');
80
+ await waitForAgentPaused(session, entryAgentId);
81
+ const entryAgent = session.state.agents.get(entryAgentId);
82
+ expect(entryAgent.status).toBe('paused');
83
+ const pausedEvents = await session.getEventsByType(agentEvents, 'agent_paused');
84
+ expect(pausedEvents.length).toBeGreaterThanOrEqual(1);
85
+ const pauseEvent = pausedEvents.find(e => e.agentId === entryAgentId);
86
+ expect(pauseEvent).toBeDefined();
87
+ expect(pauseEvent.reason).toBe('handler');
88
+ await harness.shutdown();
89
+ });
90
+ it('agent exceeding tool call hard limit → agent paused', async () => {
91
+ let callNum = 0;
92
+ const harness = createLimitsHarness({
93
+ presets: [createTestPreset({
94
+ orchestratorSystem: 'Test agent.',
95
+ orchestratorPlugins: [limitsGuardPlugin.configureAgent({ limits: { maxToolCalls: 3, maxTurns: 100 } })],
96
+ })],
97
+ mockHandler: () => {
98
+ callNum++;
99
+ return {
100
+ content: null,
101
+ toolCalls: [
102
+ { id: ToolCallId(`tc${callNum}a`), name: 'tell_user', input: { message: `A${callNum}` } },
103
+ { id: ToolCallId(`tc${callNum}b`), name: 'tell_user', input: { message: `B${callNum}` } },
104
+ ],
105
+ finishReason: 'stop',
106
+ metrics: MockLLMProvider.defaultMetrics(),
107
+ };
108
+ },
109
+ });
110
+ const session = await harness.createSession('test');
111
+ const entryAgentId = session.getEntryAgentId();
112
+ await session.sendMessage('Start');
113
+ await waitForAgentPaused(session, entryAgentId);
114
+ const entryAgent = session.state.agents.get(entryAgentId);
115
+ expect(entryAgent.status).toBe('paused');
116
+ const counters = selectPluginState(session.state, 'agentLimits')?.get(entryAgentId);
117
+ expect(counters).toBeDefined();
118
+ expect(counters.toolCallCount).toBeGreaterThanOrEqual(3);
119
+ await harness.shutdown();
120
+ });
121
+ it('agent exceeding spawned agent limit → agent paused', async () => {
122
+ let orchestratorCalls = 0;
123
+ const harness = createLimitsHarness({
124
+ presets: [createMultiAgentPreset([
125
+ { name: 'worker', system: 'Worker agent.', tools: [], agents: [] },
126
+ ], { orchestratorSystem: 'Orchestrator agent.' })],
127
+ mockHandler: (request) => {
128
+ if (request.systemPrompt.includes('Orchestrator')) {
129
+ orchestratorCalls++;
130
+ return {
131
+ content: null,
132
+ toolCalls: [
133
+ { id: ToolCallId(`tc${orchestratorCalls}a`), name: 'start_worker', input: { message: `Task ${orchestratorCalls}a` } },
134
+ { id: ToolCallId(`tc${orchestratorCalls}b`), name: 'start_worker', input: { message: `Task ${orchestratorCalls}b` } },
135
+ ],
136
+ finishReason: 'stop',
137
+ metrics: MockLLMProvider.defaultMetrics(),
138
+ };
139
+ }
140
+ return { content: 'Worker done', toolCalls: [], finishReason: 'stop', metrics: MockLLMProvider.defaultMetrics() };
141
+ },
142
+ });
143
+ const session = await harness.createSession('test');
144
+ const orchestratorId = session.getEntryAgentId();
145
+ await session.sendMessage('Start');
146
+ await waitForAgentPaused(session, orchestratorId, 15000);
147
+ const orch = session.state.agents.get(orchestratorId);
148
+ expect(orch.status).toBe('paused');
149
+ const counters = selectPluginState(session.state, 'agentLimits')?.get(orchestratorId);
150
+ expect(counters).toBeDefined();
151
+ expect(counters.spawnedAgentCount).toBeGreaterThanOrEqual(10);
152
+ await harness.shutdown();
153
+ });
154
+ });
155
+ // =========================================================================
156
+ // soft limits (status display)
157
+ // =========================================================================
158
+ describe('soft limits (status display)', () => {
159
+ it('approaching limit (80%) → status contains warning message', async () => {
160
+ let inferenceCount = 0;
161
+ let sawWarning = false;
162
+ const harness = createLimitsHarness({
163
+ presets: [createTestPreset({
164
+ orchestratorSystem: 'Test agent.',
165
+ orchestratorPlugins: [limitsGuardPlugin.configureAgent({ limits: { maxTurns: 5 } })],
166
+ })],
167
+ mockHandler: (request) => {
168
+ inferenceCount++;
169
+ const messagesStr = JSON.stringify(request.messages);
170
+ if (messagesStr.includes('Approaching maxTurns limit')) {
171
+ sawWarning = true;
172
+ }
173
+ if (inferenceCount < 5) {
174
+ return {
175
+ content: null,
176
+ toolCalls: [{ id: ToolCallId(`tc${inferenceCount}`), name: 'tell_user', input: { message: `Turn ${inferenceCount}` } }],
177
+ finishReason: 'stop',
178
+ metrics: MockLLMProvider.defaultMetrics(),
179
+ };
180
+ }
181
+ return { content: 'Done', toolCalls: [], finishReason: 'stop', metrics: MockLLMProvider.defaultMetrics() };
182
+ },
183
+ });
184
+ const session = await harness.createSession('test');
185
+ const entryAgentId = session.getEntryAgentId();
186
+ await session.sendMessage('Start');
187
+ // Agent will be paused at turn 5 or finish at turn 5 if last response has no tools
188
+ await waitForAllSettled(session);
189
+ expect(sawWarning).toBe(true);
190
+ await harness.shutdown();
191
+ });
192
+ });
193
+ // =========================================================================
194
+ // counter tracking
195
+ // =========================================================================
196
+ describe('counter tracking', () => {
197
+ it('each inference increments inferenceCount', async () => {
198
+ let callCount = 0;
199
+ const harness = createLimitsHarness({
200
+ presets: [createTestPreset({
201
+ orchestratorSystem: 'Test agent.',
202
+ orchestratorPlugins: [limitsGuardPlugin.configureAgent({ limits: { maxTurns: 100 } })],
203
+ })],
204
+ mockHandler: () => {
205
+ callCount++;
206
+ if (callCount <= 2) {
207
+ return {
208
+ content: null,
209
+ toolCalls: [{ id: ToolCallId(`tc${callCount}`), name: 'tell_user', input: { message: `Msg ${callCount}` } }],
210
+ finishReason: 'stop',
211
+ metrics: MockLLMProvider.defaultMetrics(),
212
+ };
213
+ }
214
+ return { content: 'Done', toolCalls: [], finishReason: 'stop', metrics: MockLLMProvider.defaultMetrics() };
215
+ },
216
+ });
217
+ const session = await harness.createSession('test');
218
+ await session.sendAndWaitForIdle('Start');
219
+ const counters = selectPluginState(session.state, 'agentLimits')?.get(session.getEntryAgentId());
220
+ expect(counters).toBeDefined();
221
+ expect(counters.inferenceCount).toBe(3); // 2 with tools + 1 final
222
+ await harness.shutdown();
223
+ });
224
+ it('each tool call increments toolCallCount', async () => {
225
+ let callCount = 0;
226
+ const harness = createLimitsHarness({
227
+ presets: [createTestPreset({
228
+ orchestratorSystem: 'Test agent.',
229
+ orchestratorPlugins: [limitsGuardPlugin.configureAgent({ limits: { maxTurns: 100 } })],
230
+ })],
231
+ mockHandler: () => {
232
+ callCount++;
233
+ if (callCount === 1) {
234
+ return {
235
+ content: null,
236
+ toolCalls: [
237
+ { id: ToolCallId('tc1'), name: 'tell_user', input: { message: 'A' } },
238
+ { id: ToolCallId('tc2'), name: 'tell_user', input: { message: 'B' } },
239
+ ],
240
+ finishReason: 'stop',
241
+ metrics: MockLLMProvider.defaultMetrics(),
242
+ };
243
+ }
244
+ return { content: 'Done', toolCalls: [], finishReason: 'stop', metrics: MockLLMProvider.defaultMetrics() };
245
+ },
246
+ });
247
+ const session = await harness.createSession('test');
248
+ await session.sendAndWaitForIdle('Start');
249
+ const counters = selectPluginState(session.state, 'agentLimits')?.get(session.getEntryAgentId());
250
+ expect(counters).toBeDefined();
251
+ expect(counters.toolCallCount).toBe(2);
252
+ await harness.shutdown();
253
+ });
254
+ it('each agent spawn increments parent spawnedAgentCount', async () => {
255
+ let orchestratorCalls = 0;
256
+ const harness = createLimitsHarness({
257
+ presets: [createMultiAgentPreset([
258
+ { name: 'worker', system: 'Worker agent.', tools: [], agents: [] },
259
+ ], { orchestratorSystem: 'Orchestrator agent.' })],
260
+ mockHandler: (request) => {
261
+ if (request.systemPrompt.includes('Orchestrator')) {
262
+ orchestratorCalls++;
263
+ if (orchestratorCalls === 1) {
264
+ return {
265
+ content: null,
266
+ toolCalls: [
267
+ { id: ToolCallId('tc1'), name: 'start_worker', input: { message: 'Task 1' } },
268
+ { id: ToolCallId('tc2'), name: 'start_worker', input: { message: 'Task 2' } },
269
+ ],
270
+ finishReason: 'stop',
271
+ metrics: MockLLMProvider.defaultMetrics(),
272
+ };
273
+ }
274
+ return { content: 'Done', toolCalls: [], finishReason: 'stop', metrics: MockLLMProvider.defaultMetrics() };
275
+ }
276
+ return { content: 'Worker done', toolCalls: [], finishReason: 'stop', metrics: MockLLMProvider.defaultMetrics() };
277
+ },
278
+ });
279
+ const session = await harness.createSession('test');
280
+ await session.sendAndWaitForIdle('Start', { timeoutMs: 10000 });
281
+ const orchestratorId = session.getEntryAgentId();
282
+ const counters = selectPluginState(session.state, 'agentLimits')?.get(orchestratorId);
283
+ expect(counters).toBeDefined();
284
+ expect(counters.spawnedAgentCount).toBe(2);
285
+ await harness.shutdown();
286
+ });
287
+ it('each mailbox message increments sender messagesSentCount', async () => {
288
+ let orchestratorCalls = 0;
289
+ let workerCalls = 0;
290
+ const harness = createLimitsHarness({
291
+ presets: [createMultiAgentPreset([
292
+ { name: 'worker', system: 'Worker agent.', tools: [], agents: [] },
293
+ ], { orchestratorSystem: 'Orchestrator agent.' })],
294
+ mockHandler: (request) => {
295
+ if (request.systemPrompt.includes('Orchestrator')) {
296
+ orchestratorCalls++;
297
+ if (orchestratorCalls === 1) {
298
+ return {
299
+ content: null,
300
+ toolCalls: [{ id: ToolCallId('tc1'), name: 'start_worker', input: { message: 'Task' } }],
301
+ finishReason: 'stop',
302
+ metrics: MockLLMProvider.defaultMetrics(),
303
+ };
304
+ }
305
+ return { content: 'Done', toolCalls: [], finishReason: 'stop', metrics: MockLLMProvider.defaultMetrics() };
306
+ }
307
+ workerCalls++;
308
+ if (workerCalls === 1) {
309
+ return {
310
+ content: null,
311
+ toolCalls: [{ id: ToolCallId('tc2'), name: 'send_message', input: { to: 'parent', message: 'Report' } }],
312
+ finishReason: 'stop',
313
+ metrics: MockLLMProvider.defaultMetrics(),
314
+ };
315
+ }
316
+ return { content: 'Worker done', toolCalls: [], finishReason: 'stop', metrics: MockLLMProvider.defaultMetrics() };
317
+ },
318
+ });
319
+ const session = await harness.createSession('test');
320
+ await session.sendAndWaitForIdle('Start', { timeoutMs: 10000 });
321
+ const workerCounters = selectPluginState(session.state, 'agentLimits')?.get(AgentId('worker_1'));
322
+ expect(workerCounters).toBeDefined();
323
+ expect(workerCounters.messagesSentCount).toBeGreaterThanOrEqual(1);
324
+ await harness.shutdown();
325
+ });
326
+ });
327
+ // =========================================================================
328
+ // resume resets
329
+ // =========================================================================
330
+ describe('resume resets', () => {
331
+ it('agent_resumed → pattern-based counters reset, cumulative counters preserved', async () => {
332
+ const harness = createLimitsHarness({
333
+ presets: [createTestPreset({
334
+ orchestratorSystem: 'Test agent.',
335
+ orchestratorPlugins: [limitsGuardPlugin.configureAgent({
336
+ limits: { maxRepeatedResponses: 2, maxTurns: 100 },
337
+ })],
338
+ })],
339
+ // Always return the same text → triggers maxRepeatedResponses
340
+ mockHandler: () => ({
341
+ content: 'I am stuck',
342
+ toolCalls: [],
343
+ finishReason: 'stop',
344
+ metrics: MockLLMProvider.defaultMetrics(),
345
+ }),
346
+ });
347
+ const session = await harness.createSession('test');
348
+ const entryAgentId = session.getEntryAgentId();
349
+ // 1st identical response → ok → idle
350
+ await session.sendAndWaitForIdle('Message 1');
351
+ // 2nd identical response → maxRepeatedResponses = 2 → paused
352
+ await session.sendMessage('Message 2');
353
+ await waitForAgentPaused(session, entryAgentId);
354
+ expect(session.state.agents.get(entryAgentId).status).toBe('paused');
355
+ // Resume the agent
356
+ const resumeResult = await session.callPluginMethod('agents.resume', {
357
+ agentId: String(entryAgentId),
358
+ });
359
+ expect(resumeResult.ok).toBe(true);
360
+ // All counters should be reset after resume
361
+ const countersAfterResume = selectPluginState(session.state, 'agentLimits')?.get(entryAgentId);
362
+ expect(countersAfterResume).toBeDefined();
363
+ expect(countersAfterResume.inferenceCount).toBe(0);
364
+ expect(countersAfterResume.toolCallCount).toBe(0);
365
+ expect(countersAfterResume.recentResponseHashes).toEqual([]);
366
+ // Agent resumes with reset counters. Wait for it to settle (no work → idle).
367
+ await session.waitForIdle();
368
+ // Send 2 more messages to hit the limit again
369
+ await session.sendAndWaitForIdle('Message 3');
370
+ await session.sendMessage('Message 4');
371
+ await waitForAgentPaused(session, entryAgentId);
372
+ const resumedEvents = await session.getEventsByType(agentEvents, 'agent_resumed');
373
+ expect(resumedEvents.filter(e => e.agentId === entryAgentId)).toHaveLength(1);
374
+ await harness.shutdown();
375
+ });
376
+ });
377
+ });
378
+ //# sourceMappingURL=limits-guard.integration.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"limits-guard.integration.test.js","sourceRoot":"","sources":["../../../src/plugins/limits-guard/limits-guard.integration.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,240 @@
1
+ import { agentEvents } from '../../core/agents/state.js';
2
+ import { llmEvents } from '../../core/llm/state.js';
3
+ import { definePlugin } from '../../core/plugins/plugin-builder.js';
4
+ import { selectPluginState } from '../../core/sessions/reducer.js';
5
+ import { toolEvents } from '../../core/tools/state.js';
6
+ import { responseFingerprint, toolCallFingerprint } from '../../lib/utils/hash.js';
7
+ import { mailboxEvents } from '../../plugins/mailbox/state.js';
8
+ import { checkLimits, countConsecutiveTailDuplicates, resolveAgentLimits } from './limit-guard.js';
9
+ export const createAgentCounters = () => ({
10
+ inferenceCount: 0,
11
+ toolCallCount: 0,
12
+ spawnedAgentCount: 0,
13
+ messagesSentCount: 0,
14
+ consecutiveToolFailures: {},
15
+ recentToolCallHashes: [],
16
+ recentResponseHashes: [],
17
+ });
18
+ /**
19
+ * Extract agent counters from session state (for external consumers).
20
+ */
21
+ export function selectAgentCounters(sessionState, agentId) {
22
+ return selectPluginState(sessionState, 'agentLimits')?.get(agentId) ?? createAgentCounters();
23
+ }
24
+ // ============================================================================
25
+ // Limit warning event
26
+ // ============================================================================
27
+ import z from 'zod/v4';
28
+ import { agentIdSchema } from '../../core/agents/schema.js';
29
+ import { createEventsFactory } from '../../core/events/types.js';
30
+ export const limitsEvents = createEventsFactory({
31
+ events: {
32
+ limit_warning: z.object({
33
+ agentId: agentIdSchema,
34
+ limitName: z.string(),
35
+ currentValue: z.number(),
36
+ hardLimit: z.number(),
37
+ message: z.string(),
38
+ }),
39
+ },
40
+ });
41
+ // ============================================================================
42
+ // Helper
43
+ // ============================================================================
44
+ /**
45
+ * Resolve tool name from agent state for tool_completed/tool_failed events.
46
+ * Must run BEFORE core reducer (which clears executingToolCall).
47
+ */
48
+ function resolveToolName(sessionState, agentId, toolCallId) {
49
+ const agent = sessionState.agents.get(agentId);
50
+ if (!agent)
51
+ return 'unknown';
52
+ return agent.executingToolCall?.toolName
53
+ ?? agent.pendingToolCalls.find((tc) => tc.id === toolCallId)?.name
54
+ ?? 'unknown';
55
+ }
56
+ export const limitsGuardPlugin = definePlugin('limits-guard')
57
+ .events([agentEvents, llmEvents, toolEvents, mailboxEvents])
58
+ .state({
59
+ key: 'agentLimits',
60
+ initial: () => new Map(),
61
+ reduce: (limits, event, sessionState) => {
62
+ switch (event.type) {
63
+ case 'agent_spawned': {
64
+ const newLimits = new Map(limits);
65
+ newLimits.set(event.agentId, createAgentCounters());
66
+ // Increment parent's spawnedAgentCount
67
+ if (event.parentId) {
68
+ const parentCounters = newLimits.get(event.parentId);
69
+ if (parentCounters) {
70
+ newLimits.set(event.parentId, {
71
+ ...parentCounters,
72
+ spawnedAgentCount: parentCounters.spawnedAgentCount + 1,
73
+ });
74
+ }
75
+ }
76
+ return newLimits;
77
+ }
78
+ case 'mailbox_message': {
79
+ // Increment sender's messagesSentCount if sender is an agent
80
+ const senderAgentId = typeof event.message.from === 'string' && event.message.from !== 'user'
81
+ ? event.message.from
82
+ : null;
83
+ if (!senderAgentId)
84
+ return limits;
85
+ const senderCounters = limits.get(senderAgentId);
86
+ if (!senderCounters)
87
+ return limits;
88
+ const newLimits = new Map(limits);
89
+ newLimits.set(senderAgentId, {
90
+ ...senderCounters,
91
+ messagesSentCount: senderCounters.messagesSentCount + 1,
92
+ });
93
+ return newLimits;
94
+ }
95
+ case 'inference_completed': {
96
+ const counters = limits.get(event.agentId);
97
+ if (!counters)
98
+ return limits;
99
+ const hasToolCalls = event.response.toolCalls.length > 0;
100
+ let newRecentResponseHashes = counters.recentResponseHashes;
101
+ if (!hasToolCalls) {
102
+ const isWaiting = /^\s*WAITING\s*$/.test(event.response.content ?? '');
103
+ if (!isWaiting) {
104
+ const hash = responseFingerprint(event.response.content);
105
+ newRecentResponseHashes = [...newRecentResponseHashes, hash].slice(-10);
106
+ }
107
+ }
108
+ const newLimits = new Map(limits);
109
+ newLimits.set(event.agentId, {
110
+ ...counters,
111
+ inferenceCount: counters.inferenceCount + 1,
112
+ recentResponseHashes: newRecentResponseHashes,
113
+ });
114
+ return newLimits;
115
+ }
116
+ case 'tool_started': {
117
+ const counters = limits.get(event.agentId);
118
+ if (!counters)
119
+ return limits;
120
+ const fingerprint = toolCallFingerprint(event.toolName, event.input);
121
+ const newLimits = new Map(limits);
122
+ newLimits.set(event.agentId, {
123
+ ...counters,
124
+ toolCallCount: counters.toolCallCount + 1,
125
+ recentToolCallHashes: [...counters.recentToolCallHashes, fingerprint].slice(-20),
126
+ });
127
+ return newLimits;
128
+ }
129
+ case 'tool_completed': {
130
+ const counters = limits.get(event.agentId);
131
+ if (!counters)
132
+ return limits;
133
+ // Resolve tool name from agent state (must run before core clears executingToolCall)
134
+ const toolName = resolveToolName(sessionState, event.agentId, event.toolCallId);
135
+ const { [toolName]: _, ...restFailures } = counters.consecutiveToolFailures;
136
+ const newLimits = new Map(limits);
137
+ newLimits.set(event.agentId, {
138
+ ...counters,
139
+ consecutiveToolFailures: restFailures,
140
+ });
141
+ return newLimits;
142
+ }
143
+ case 'tool_failed': {
144
+ const counters = limits.get(event.agentId);
145
+ if (!counters)
146
+ return limits;
147
+ // Resolve tool name from agent state (must run before core clears executingToolCall)
148
+ const toolName = resolveToolName(sessionState, event.agentId, event.toolCallId);
149
+ const currentEntry = counters.consecutiveToolFailures[toolName];
150
+ const newLimits = new Map(limits);
151
+ newLimits.set(event.agentId, {
152
+ ...counters,
153
+ consecutiveToolFailures: {
154
+ ...counters.consecutiveToolFailures,
155
+ [toolName]: { count: (currentEntry?.count ?? 0) + 1, lastError: event.error },
156
+ },
157
+ });
158
+ return newLimits;
159
+ }
160
+ case 'agent_resumed': {
161
+ const counters = limits.get(event.agentId);
162
+ if (!counters)
163
+ return limits;
164
+ const newLimits = new Map(limits);
165
+ newLimits.set(event.agentId, {
166
+ ...counters,
167
+ inferenceCount: 0,
168
+ toolCallCount: 0,
169
+ spawnedAgentCount: 0,
170
+ messagesSentCount: 0,
171
+ consecutiveToolFailures: {},
172
+ recentToolCallHashes: [],
173
+ recentResponseHashes: [],
174
+ });
175
+ return newLimits;
176
+ }
177
+ default:
178
+ return limits;
179
+ }
180
+ },
181
+ })
182
+ .agentConfig()
183
+ .hook('afterInference', async (ctx) => {
184
+ const resolvedLimits = resolveAgentLimits(ctx.pluginAgentConfig?.limits);
185
+ const counters = ctx.pluginState.get(ctx.agentId);
186
+ if (!counters)
187
+ return null;
188
+ // The inference_completed event hasn't been emitted yet, so project
189
+ // what the counters will be after it's processed by the reducer.
190
+ const hasToolCalls = ctx.response.toolCalls.length > 0;
191
+ const projected = {
192
+ ...counters,
193
+ inferenceCount: counters.inferenceCount + 1,
194
+ recentResponseHashes: hasToolCalls
195
+ ? counters.recentResponseHashes
196
+ : [...counters.recentResponseHashes, responseFingerprint(ctx.response.content)].slice(-10),
197
+ };
198
+ const limitCheck = checkLimits(projected, resolvedLimits);
199
+ if (limitCheck.status === 'hard_limit') {
200
+ return { action: 'pause', reason: limitCheck.reason };
201
+ }
202
+ return null;
203
+ })
204
+ .status((ctx) => {
205
+ const resolvedLimits = resolveAgentLimits(ctx.pluginAgentConfig?.limits);
206
+ const counters = ctx.pluginState.get(ctx.agentId);
207
+ if (!counters)
208
+ return null;
209
+ const parts = [];
210
+ // Consecutive failures of the same tool — surface from count=2, well before the
211
+ // hard limit. The agent sees what it tried, how many times, and with what error,
212
+ // so it can change strategy rather than blindly retry.
213
+ for (const [toolName, entry] of Object.entries(counters.consecutiveToolFailures)) {
214
+ if (entry.count >= 2) {
215
+ const lastError = entry.lastError.length > 600
216
+ ? entry.lastError.slice(0, 600) + '… [truncated]'
217
+ : entry.lastError;
218
+ parts.push(`⚠️ Tool "${toolName}" has failed ${entry.count} times in a row (hard limit: ${resolvedLimits.maxConsecutiveToolFailures}). `
219
+ + `Last error: ${lastError}\n`
220
+ + `Do NOT retry the same call again. Diagnose the root cause, change approach, or stop and report what you tried.`);
221
+ }
222
+ }
223
+ // Same tool call with identical arguments repeated — even successful calls
224
+ // that produce no new information are a waste. Warn from 2 repeats.
225
+ const repeatedCalls = countConsecutiveTailDuplicates(counters.recentToolCallHashes);
226
+ if (repeatedCalls >= 2) {
227
+ parts.push(`⚠️ You have called the same tool with identical arguments ${repeatedCalls} times in a row `
228
+ + `(hard limit: ${resolvedLimits.maxRepeatedToolCalls}). `
229
+ + `Either the call is idempotent (stop repeating it) or your approach isn't working (change strategy).`);
230
+ }
231
+ // Existing counter-based soft warnings (maxTurns, maxToolCalls, etc.)
232
+ const limitCheck = checkLimits(counters, resolvedLimits);
233
+ if (limitCheck.status === 'soft_warning') {
234
+ parts.push(`⚠️ Approaching ${limitCheck.limitName} limit: ${limitCheck.currentValue}/${limitCheck.hardLimit}. `
235
+ + `Wrap up your current task or you will be stopped.`);
236
+ }
237
+ return parts.length > 0 ? parts.join('\n\n') : null;
238
+ })
239
+ .build();
240
+ //# sourceMappingURL=plugin.js.map