macro-agent 0.0.10 → 0.0.12

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 (518) hide show
  1. package/.macro-agent/teams/self-driving/prompts/grinder.md +27 -0
  2. package/.macro-agent/teams/self-driving/prompts/judge.md +27 -0
  3. package/.macro-agent/teams/self-driving/prompts/planner.md +33 -0
  4. package/.macro-agent/teams/self-driving/roles/grinder.yaml +17 -0
  5. package/.macro-agent/teams/self-driving/roles/judge.yaml +24 -0
  6. package/.macro-agent/teams/self-driving/roles/planner.yaml +18 -0
  7. package/.macro-agent/teams/self-driving/team.yaml +103 -0
  8. package/.macro-agent/teams/structured/prompts/developer.md +26 -0
  9. package/.macro-agent/teams/structured/prompts/lead.md +25 -0
  10. package/.macro-agent/teams/structured/prompts/reviewer.md +24 -0
  11. package/.macro-agent/teams/structured/roles/developer.yaml +12 -0
  12. package/.macro-agent/teams/structured/roles/lead.yaml +11 -0
  13. package/.macro-agent/teams/structured/roles/reviewer.yaml +19 -0
  14. package/.macro-agent/teams/structured/team.yaml +89 -0
  15. package/.sudocode/issues.jsonl +56 -51
  16. package/.sudocode/specs.jsonl +8 -1
  17. package/CLAUDE.md +121 -30
  18. package/README.md +60 -3
  19. package/dist/acp/macro-agent.d.ts +4 -0
  20. package/dist/acp/macro-agent.d.ts.map +1 -1
  21. package/dist/acp/macro-agent.js +50 -4
  22. package/dist/acp/macro-agent.js.map +1 -1
  23. package/dist/acp/session-mapper.d.ts +20 -1
  24. package/dist/acp/session-mapper.d.ts.map +1 -1
  25. package/dist/acp/session-mapper.js +90 -1
  26. package/dist/acp/session-mapper.js.map +1 -1
  27. package/dist/acp/types.d.ts +24 -1
  28. package/dist/acp/types.d.ts.map +1 -1
  29. package/dist/acp/types.js.map +1 -1
  30. package/dist/agent/agent-manager.d.ts +40 -1
  31. package/dist/agent/agent-manager.d.ts.map +1 -1
  32. package/dist/agent/agent-manager.js +172 -8
  33. package/dist/agent/agent-manager.js.map +1 -1
  34. package/dist/agent/types.d.ts +22 -0
  35. package/dist/agent/types.d.ts.map +1 -1
  36. package/dist/agent/types.js.map +1 -1
  37. package/dist/agent/wake.d.ts +15 -0
  38. package/dist/agent/wake.d.ts.map +1 -1
  39. package/dist/agent/wake.js +15 -0
  40. package/dist/agent/wake.js.map +1 -1
  41. package/dist/agent-detection/command-builder.d.ts +30 -0
  42. package/dist/agent-detection/command-builder.d.ts.map +1 -0
  43. package/dist/agent-detection/command-builder.js +71 -0
  44. package/dist/agent-detection/command-builder.js.map +1 -0
  45. package/dist/agent-detection/detector.d.ts +84 -0
  46. package/dist/agent-detection/detector.d.ts.map +1 -0
  47. package/dist/agent-detection/detector.js +240 -0
  48. package/dist/agent-detection/detector.js.map +1 -0
  49. package/dist/agent-detection/index.d.ts +12 -0
  50. package/dist/agent-detection/index.d.ts.map +1 -0
  51. package/dist/agent-detection/index.js +14 -0
  52. package/dist/agent-detection/index.js.map +1 -0
  53. package/dist/agent-detection/registry.d.ts +53 -0
  54. package/dist/agent-detection/registry.d.ts.map +1 -0
  55. package/dist/agent-detection/registry.js +177 -0
  56. package/dist/agent-detection/registry.js.map +1 -0
  57. package/dist/agent-detection/types.d.ts +121 -0
  58. package/dist/agent-detection/types.d.ts.map +1 -0
  59. package/dist/agent-detection/types.js +20 -0
  60. package/dist/agent-detection/types.js.map +1 -0
  61. package/dist/api/server.d.ts +5 -1
  62. package/dist/api/server.d.ts.map +1 -1
  63. package/dist/api/server.js +362 -0
  64. package/dist/api/server.js.map +1 -1
  65. package/dist/api/types.d.ts +50 -1
  66. package/dist/api/types.d.ts.map +1 -1
  67. package/dist/cli/acp.d.ts +2 -0
  68. package/dist/cli/acp.d.ts.map +1 -1
  69. package/dist/cli/acp.js +8 -1
  70. package/dist/cli/acp.js.map +1 -1
  71. package/dist/cli/index.js +29 -0
  72. package/dist/cli/index.js.map +1 -1
  73. package/dist/cli/mcp.js +38 -0
  74. package/dist/cli/mcp.js.map +1 -1
  75. package/dist/config/index.d.ts +2 -0
  76. package/dist/config/index.d.ts.map +1 -0
  77. package/dist/config/index.js +2 -0
  78. package/dist/config/index.js.map +1 -0
  79. package/dist/config/project-config.d.ts +46 -0
  80. package/dist/config/project-config.d.ts.map +1 -0
  81. package/dist/config/project-config.js +68 -0
  82. package/dist/config/project-config.js.map +1 -0
  83. package/dist/lifecycle/cascade.d.ts +1 -1
  84. package/dist/lifecycle/cascade.d.ts.map +1 -1
  85. package/dist/lifecycle/handlers/index.d.ts +4 -0
  86. package/dist/lifecycle/handlers/index.d.ts.map +1 -1
  87. package/dist/lifecycle/handlers/index.js +2 -0
  88. package/dist/lifecycle/handlers/index.js.map +1 -1
  89. package/dist/lifecycle/handlers/worker.d.ts +4 -0
  90. package/dist/lifecycle/handlers/worker.d.ts.map +1 -1
  91. package/dist/lifecycle/handlers/worker.js +35 -3
  92. package/dist/lifecycle/handlers/worker.js.map +1 -1
  93. package/dist/mail/conversation-map.d.ts +33 -0
  94. package/dist/mail/conversation-map.d.ts.map +1 -0
  95. package/dist/mail/conversation-map.js +61 -0
  96. package/dist/mail/conversation-map.js.map +1 -0
  97. package/dist/mail/index.d.ts +11 -0
  98. package/dist/mail/index.d.ts.map +1 -0
  99. package/dist/mail/index.js +11 -0
  100. package/dist/mail/index.js.map +1 -0
  101. package/dist/mail/mail-service.d.ts +85 -0
  102. package/dist/mail/mail-service.d.ts.map +1 -0
  103. package/dist/mail/mail-service.js +121 -0
  104. package/dist/mail/mail-service.js.map +1 -0
  105. package/dist/mail/stores/eventstore-conversation-store.d.ts +40 -0
  106. package/dist/mail/stores/eventstore-conversation-store.d.ts.map +1 -0
  107. package/dist/mail/stores/eventstore-conversation-store.js +131 -0
  108. package/dist/mail/stores/eventstore-conversation-store.js.map +1 -0
  109. package/dist/mail/stores/eventstore-participant-store.d.ts +43 -0
  110. package/dist/mail/stores/eventstore-participant-store.d.ts.map +1 -0
  111. package/dist/mail/stores/eventstore-participant-store.js +145 -0
  112. package/dist/mail/stores/eventstore-participant-store.js.map +1 -0
  113. package/dist/mail/stores/eventstore-thread-store.d.ts +46 -0
  114. package/dist/mail/stores/eventstore-thread-store.d.ts.map +1 -0
  115. package/dist/mail/stores/eventstore-thread-store.js +118 -0
  116. package/dist/mail/stores/eventstore-thread-store.js.map +1 -0
  117. package/dist/mail/stores/eventstore-turn-store.d.ts +47 -0
  118. package/dist/mail/stores/eventstore-turn-store.d.ts.map +1 -0
  119. package/dist/mail/stores/eventstore-turn-store.js +153 -0
  120. package/dist/mail/stores/eventstore-turn-store.js.map +1 -0
  121. package/dist/mail/stores/index.d.ts +12 -0
  122. package/dist/mail/stores/index.d.ts.map +1 -0
  123. package/dist/mail/stores/index.js +12 -0
  124. package/dist/mail/stores/index.js.map +1 -0
  125. package/dist/mail/stores/types.d.ts +146 -0
  126. package/dist/mail/stores/types.d.ts.map +1 -0
  127. package/dist/mail/stores/types.js +13 -0
  128. package/dist/mail/stores/types.js.map +1 -0
  129. package/dist/mail/turn-recorder.d.ts +30 -0
  130. package/dist/mail/turn-recorder.d.ts.map +1 -0
  131. package/dist/mail/turn-recorder.js +98 -0
  132. package/dist/mail/turn-recorder.js.map +1 -0
  133. package/dist/map/adapter/acp-over-map.d.ts.map +1 -1
  134. package/dist/map/adapter/acp-over-map.js +32 -2
  135. package/dist/map/adapter/acp-over-map.js.map +1 -1
  136. package/dist/map/adapter/event-translator.d.ts.map +1 -1
  137. package/dist/map/adapter/event-translator.js +4 -0
  138. package/dist/map/adapter/event-translator.js.map +1 -1
  139. package/dist/map/adapter/extensions/agent-detection.d.ts +49 -0
  140. package/dist/map/adapter/extensions/agent-detection.d.ts.map +1 -0
  141. package/dist/map/adapter/extensions/agent-detection.js +91 -0
  142. package/dist/map/adapter/extensions/agent-detection.js.map +1 -0
  143. package/dist/map/adapter/extensions/index.d.ts +10 -1
  144. package/dist/map/adapter/extensions/index.d.ts.map +1 -1
  145. package/dist/map/adapter/extensions/index.js +39 -0
  146. package/dist/map/adapter/extensions/index.js.map +1 -1
  147. package/dist/map/adapter/extensions/resume.d.ts +47 -0
  148. package/dist/map/adapter/extensions/resume.d.ts.map +1 -0
  149. package/dist/map/adapter/extensions/resume.js +59 -0
  150. package/dist/map/adapter/extensions/resume.js.map +1 -0
  151. package/dist/map/adapter/extensions/workspace-files.d.ts +42 -0
  152. package/dist/map/adapter/extensions/workspace-files.d.ts.map +1 -0
  153. package/dist/map/adapter/extensions/workspace-files.js +338 -0
  154. package/dist/map/adapter/extensions/workspace-files.js.map +1 -0
  155. package/dist/map/adapter/mail-handler-adapter.d.ts +27 -0
  156. package/dist/map/adapter/mail-handler-adapter.d.ts.map +1 -0
  157. package/dist/map/adapter/mail-handler-adapter.js +292 -0
  158. package/dist/map/adapter/mail-handler-adapter.js.map +1 -0
  159. package/dist/map/adapter/map-adapter.d.ts +34 -10
  160. package/dist/map/adapter/map-adapter.d.ts.map +1 -1
  161. package/dist/map/adapter/map-adapter.js +110 -14
  162. package/dist/map/adapter/map-adapter.js.map +1 -1
  163. package/dist/map/adapter/rpc-handler.d.ts +4 -1
  164. package/dist/map/adapter/rpc-handler.d.ts.map +1 -1
  165. package/dist/map/adapter/rpc-handler.js +6 -0
  166. package/dist/map/adapter/rpc-handler.js.map +1 -1
  167. package/dist/map/index.d.ts +1 -0
  168. package/dist/map/index.d.ts.map +1 -1
  169. package/dist/map/index.js +2 -0
  170. package/dist/map/index.js.map +1 -1
  171. package/dist/map/types.d.ts +3 -1
  172. package/dist/map/types.d.ts.map +1 -1
  173. package/dist/map/types.js.map +1 -1
  174. package/dist/mcp/mcp-server.d.ts +6 -0
  175. package/dist/mcp/mcp-server.d.ts.map +1 -1
  176. package/dist/mcp/mcp-server.js +45 -0
  177. package/dist/mcp/mcp-server.js.map +1 -1
  178. package/dist/mcp/tools/claim_task.d.ts +35 -0
  179. package/dist/mcp/tools/claim_task.d.ts.map +1 -0
  180. package/dist/mcp/tools/claim_task.js +58 -0
  181. package/dist/mcp/tools/claim_task.js.map +1 -0
  182. package/dist/mcp/tools/done.d.ts +15 -2
  183. package/dist/mcp/tools/done.d.ts.map +1 -1
  184. package/dist/mcp/tools/done.js +45 -10
  185. package/dist/mcp/tools/done.js.map +1 -1
  186. package/dist/mcp/tools/list_claimable_tasks.d.ts +38 -0
  187. package/dist/mcp/tools/list_claimable_tasks.d.ts.map +1 -0
  188. package/dist/mcp/tools/list_claimable_tasks.js +63 -0
  189. package/dist/mcp/tools/list_claimable_tasks.js.map +1 -0
  190. package/dist/mcp/tools/unclaim_task.d.ts +31 -0
  191. package/dist/mcp/tools/unclaim_task.d.ts.map +1 -0
  192. package/dist/mcp/tools/unclaim_task.js +47 -0
  193. package/dist/mcp/tools/unclaim_task.js.map +1 -0
  194. package/dist/metrics/index.d.ts +2 -0
  195. package/dist/metrics/index.d.ts.map +1 -0
  196. package/dist/metrics/index.js +2 -0
  197. package/dist/metrics/index.js.map +1 -0
  198. package/dist/metrics/metrics.d.ts +79 -0
  199. package/dist/metrics/metrics.d.ts.map +1 -0
  200. package/dist/metrics/metrics.js +166 -0
  201. package/dist/metrics/metrics.js.map +1 -0
  202. package/dist/roles/capabilities.d.ts +1 -0
  203. package/dist/roles/capabilities.d.ts.map +1 -1
  204. package/dist/roles/capabilities.js +3 -0
  205. package/dist/roles/capabilities.js.map +1 -1
  206. package/dist/roles/types.d.ts +1 -1
  207. package/dist/roles/types.d.ts.map +1 -1
  208. package/dist/router/channels.d.ts +2 -4
  209. package/dist/router/channels.d.ts.map +1 -1
  210. package/dist/router/channels.js.map +1 -1
  211. package/dist/router/message-router.d.ts +85 -9
  212. package/dist/router/message-router.d.ts.map +1 -1
  213. package/dist/router/message-router.js +203 -14
  214. package/dist/router/message-router.js.map +1 -1
  215. package/dist/router/role-resolver.d.ts +10 -1
  216. package/dist/router/role-resolver.d.ts.map +1 -1
  217. package/dist/router/role-resolver.js +15 -1
  218. package/dist/router/role-resolver.js.map +1 -1
  219. package/dist/router/types.d.ts +30 -1
  220. package/dist/router/types.d.ts.map +1 -1
  221. package/dist/router/types.js.map +1 -1
  222. package/dist/server/combined-server.d.ts +6 -0
  223. package/dist/server/combined-server.d.ts.map +1 -1
  224. package/dist/server/combined-server.js +24 -2
  225. package/dist/server/combined-server.js.map +1 -1
  226. package/dist/store/event-store.d.ts +14 -1
  227. package/dist/store/event-store.d.ts.map +1 -1
  228. package/dist/store/event-store.js +456 -4
  229. package/dist/store/event-store.js.map +1 -1
  230. package/dist/store/types/agents.d.ts +1 -1
  231. package/dist/store/types/agents.d.ts.map +1 -1
  232. package/dist/store/types/conversations.d.ts +91 -0
  233. package/dist/store/types/conversations.d.ts.map +1 -0
  234. package/dist/store/types/conversations.js +8 -0
  235. package/dist/store/types/conversations.js.map +1 -0
  236. package/dist/store/types/events.d.ts +1 -1
  237. package/dist/store/types/events.d.ts.map +1 -1
  238. package/dist/store/types/events.js.map +1 -1
  239. package/dist/store/types/index.d.ts +2 -0
  240. package/dist/store/types/index.d.ts.map +1 -1
  241. package/dist/store/types/index.js +2 -0
  242. package/dist/store/types/index.js.map +1 -1
  243. package/dist/store/types/sessions.d.ts +44 -0
  244. package/dist/store/types/sessions.d.ts.map +1 -0
  245. package/dist/store/types/sessions.js +9 -0
  246. package/dist/store/types/sessions.js.map +1 -0
  247. package/dist/store/types/tasks.d.ts +2 -0
  248. package/dist/store/types/tasks.d.ts.map +1 -1
  249. package/dist/task/backend/memory.d.ts +4 -1
  250. package/dist/task/backend/memory.d.ts.map +1 -1
  251. package/dist/task/backend/memory.js +81 -0
  252. package/dist/task/backend/memory.js.map +1 -1
  253. package/dist/task/backend/types.d.ts +30 -0
  254. package/dist/task/backend/types.d.ts.map +1 -1
  255. package/dist/task/backend/types.js.map +1 -1
  256. package/dist/teams/index.d.ts +4 -0
  257. package/dist/teams/index.d.ts.map +1 -0
  258. package/dist/teams/index.js +4 -0
  259. package/dist/teams/index.js.map +1 -0
  260. package/dist/teams/team-loader.d.ts +20 -0
  261. package/dist/teams/team-loader.d.ts.map +1 -0
  262. package/dist/teams/team-loader.js +293 -0
  263. package/dist/teams/team-loader.js.map +1 -0
  264. package/dist/teams/team-runtime.d.ts +139 -0
  265. package/dist/teams/team-runtime.d.ts.map +1 -0
  266. package/dist/teams/team-runtime.js +613 -0
  267. package/dist/teams/team-runtime.js.map +1 -0
  268. package/dist/teams/types.d.ts +266 -0
  269. package/dist/teams/types.d.ts.map +1 -0
  270. package/dist/teams/types.js +20 -0
  271. package/dist/teams/types.js.map +1 -0
  272. package/dist/trigger/router/trigger-router.d.ts +30 -3
  273. package/dist/trigger/router/trigger-router.d.ts.map +1 -1
  274. package/dist/trigger/router/trigger-router.js +30 -3
  275. package/dist/trigger/router/trigger-router.js.map +1 -1
  276. package/dist/trigger/wake/types.d.ts +31 -5
  277. package/dist/trigger/wake/types.d.ts.map +1 -1
  278. package/dist/trigger/wake/types.js +19 -0
  279. package/dist/trigger/wake/types.js.map +1 -1
  280. package/dist/workspace/dataplane-adapter.d.ts +1 -1
  281. package/dist/workspace/dataplane-adapter.d.ts.map +1 -1
  282. package/dist/workspace/dataplane-adapter.js +1 -1
  283. package/dist/workspace/dataplane-adapter.js.map +1 -1
  284. package/dist/workspace/index.d.ts +1 -1
  285. package/dist/workspace/index.d.ts.map +1 -1
  286. package/dist/workspace/strategies/index.d.ts +6 -0
  287. package/dist/workspace/strategies/index.d.ts.map +1 -0
  288. package/dist/workspace/strategies/index.js +5 -0
  289. package/dist/workspace/strategies/index.js.map +1 -0
  290. package/dist/workspace/strategies/optimistic.d.ts +26 -0
  291. package/dist/workspace/strategies/optimistic.d.ts.map +1 -0
  292. package/dist/workspace/strategies/optimistic.js +121 -0
  293. package/dist/workspace/strategies/optimistic.js.map +1 -0
  294. package/dist/workspace/strategies/queue.d.ts +26 -0
  295. package/dist/workspace/strategies/queue.d.ts.map +1 -0
  296. package/dist/workspace/strategies/queue.js +67 -0
  297. package/dist/workspace/strategies/queue.js.map +1 -0
  298. package/dist/workspace/strategies/registry.d.ts +37 -0
  299. package/dist/workspace/strategies/registry.d.ts.map +1 -0
  300. package/dist/workspace/strategies/registry.js +63 -0
  301. package/dist/workspace/strategies/registry.js.map +1 -0
  302. package/dist/workspace/strategies/trunk.d.ts +20 -0
  303. package/dist/workspace/strategies/trunk.d.ts.map +1 -0
  304. package/dist/workspace/strategies/trunk.js +108 -0
  305. package/dist/workspace/strategies/trunk.js.map +1 -0
  306. package/dist/workspace/strategies/types.d.ts +104 -0
  307. package/dist/workspace/strategies/types.d.ts.map +1 -0
  308. package/dist/workspace/strategies/types.js +11 -0
  309. package/dist/workspace/strategies/types.js.map +1 -0
  310. package/dist/workspace/types.d.ts +1 -1
  311. package/dist/workspace/types.d.ts.map +1 -1
  312. package/dist/workspace/workspace-manager.d.ts +1 -1
  313. package/dist/workspace/workspace-manager.d.ts.map +1 -1
  314. package/docs/implementation-details.md +1127 -0
  315. package/docs/implementation-summary.md +448 -0
  316. package/docs/mail-integration.md +608 -0
  317. package/docs/plan-self-driving-support.md +433 -0
  318. package/docs/spec-self-driving-support.md +462 -0
  319. package/docs/team-templates.md +860 -0
  320. package/docs/teams.md +233 -0
  321. package/package.json +5 -3
  322. package/src/acp/__tests__/integration.test.ts +161 -1
  323. package/src/acp/__tests__/macro-agent.test.ts +95 -0
  324. package/src/acp/__tests__/session-persistence.test.ts +276 -0
  325. package/src/acp/macro-agent.ts +79 -7
  326. package/src/acp/session-mapper.ts +108 -1
  327. package/src/acp/types.ts +33 -1
  328. package/src/agent/agent-manager.ts +278 -6
  329. package/src/agent/types.ts +27 -0
  330. package/src/agent/wake.ts +15 -0
  331. package/src/agent-detection/__tests__/command-builder.test.ts +336 -0
  332. package/src/agent-detection/__tests__/detector.test.ts +768 -0
  333. package/src/agent-detection/__tests__/registry.test.ts +254 -0
  334. package/src/agent-detection/command-builder.ts +90 -0
  335. package/src/agent-detection/detector.ts +307 -0
  336. package/src/agent-detection/index.ts +36 -0
  337. package/src/agent-detection/registry.ts +200 -0
  338. package/src/agent-detection/types.ts +184 -0
  339. package/src/api/__tests__/conversation-api.test.ts +468 -0
  340. package/src/api/server.ts +425 -1
  341. package/src/api/types.ts +64 -1
  342. package/src/cli/acp.ts +9 -1
  343. package/src/cli/index.ts +44 -0
  344. package/src/cli/mcp.ts +47 -0
  345. package/src/config/index.ts +9 -0
  346. package/src/config/project-config.ts +107 -0
  347. package/src/lifecycle/cascade.ts +1 -1
  348. package/src/lifecycle/handlers/index.ts +8 -0
  349. package/src/lifecycle/handlers/worker.ts +48 -3
  350. package/src/mail/__tests__/conversation-lifecycle.test.ts +409 -0
  351. package/src/mail/__tests__/eventstore-stores.test.ts +1073 -0
  352. package/src/mail/__tests__/mail-full-agent.e2e.test.ts +575 -0
  353. package/src/mail/__tests__/mail-integration.test.ts +759 -0
  354. package/src/mail/__tests__/mail-map-protocol.e2e.test.ts +1068 -0
  355. package/src/mail/__tests__/mail-service.test.ts +506 -0
  356. package/src/mail/__tests__/turn-recorder.test.ts +328 -0
  357. package/src/mail/conversation-map.ts +107 -0
  358. package/src/mail/index.ts +25 -0
  359. package/src/mail/mail-service.ts +257 -0
  360. package/src/mail/stores/eventstore-conversation-store.ts +146 -0
  361. package/src/mail/stores/eventstore-participant-store.ts +172 -0
  362. package/src/mail/stores/eventstore-thread-store.ts +129 -0
  363. package/src/mail/stores/eventstore-turn-store.ts +173 -0
  364. package/src/mail/stores/index.ts +12 -0
  365. package/src/mail/stores/types.ts +160 -0
  366. package/src/mail/turn-recorder.ts +124 -0
  367. package/src/map/README.md +79 -0
  368. package/src/map/adapter/__tests__/extensions.test.ts +359 -0
  369. package/src/map/adapter/__tests__/map-adapter.test.ts +90 -0
  370. package/src/map/adapter/__tests__/workspace-files.test.ts +673 -0
  371. package/src/map/adapter/acp-over-map.ts +45 -2
  372. package/src/map/adapter/event-translator.ts +4 -0
  373. package/src/map/adapter/extensions/agent-detection.ts +201 -0
  374. package/src/map/adapter/extensions/index.ts +63 -0
  375. package/src/map/adapter/extensions/resume.ts +114 -0
  376. package/src/map/adapter/extensions/workspace-files.ts +449 -0
  377. package/src/map/adapter/mail-handler-adapter.ts +429 -0
  378. package/src/map/adapter/map-adapter.ts +173 -27
  379. package/src/map/adapter/rpc-handler.ts +8 -1
  380. package/src/map/index.ts +3 -0
  381. package/src/map/types.ts +3 -1
  382. package/src/mcp/mcp-server.ts +67 -0
  383. package/src/mcp/tools/claim_task.ts +86 -0
  384. package/src/mcp/tools/done.ts +59 -10
  385. package/src/mcp/tools/list_claimable_tasks.ts +93 -0
  386. package/src/mcp/tools/unclaim_task.ts +71 -0
  387. package/src/metrics/index.ts +9 -0
  388. package/src/metrics/metrics.ts +280 -0
  389. package/src/roles/capabilities.ts +3 -0
  390. package/src/roles/types.ts +2 -1
  391. package/src/router/README.md +120 -0
  392. package/src/router/__tests__/message-router.test.ts +561 -0
  393. package/src/router/channels.ts +3 -4
  394. package/src/router/message-router.ts +308 -22
  395. package/src/router/role-resolver.ts +22 -1
  396. package/src/router/types.ts +36 -1
  397. package/src/server/combined-server.ts +36 -2
  398. package/src/store/README.md +134 -0
  399. package/src/store/event-store.ts +546 -3
  400. package/src/store/types/agents.ts +1 -1
  401. package/src/store/types/conversations.ts +129 -0
  402. package/src/store/types/events.ts +5 -1
  403. package/src/store/types/index.ts +2 -0
  404. package/src/store/types/sessions.ts +53 -0
  405. package/src/store/types/tasks.ts +3 -0
  406. package/src/task/backend/memory.ts +116 -0
  407. package/src/task/backend/types.ts +43 -0
  408. package/src/teams/__tests__/cross-subsystem.integration.test.ts +983 -0
  409. package/src/teams/__tests__/e2e/team-runtime.e2e.test.ts +553 -0
  410. package/src/teams/__tests__/team-system.test.ts +1280 -0
  411. package/src/teams/index.ts +13 -0
  412. package/src/teams/team-loader.ts +434 -0
  413. package/src/teams/team-runtime.ts +727 -0
  414. package/src/teams/types.ts +377 -0
  415. package/src/trigger/router/trigger-router.ts +30 -3
  416. package/src/trigger/wake/types.ts +32 -5
  417. package/src/trigger/wake/wake-manager.ts +2 -2
  418. package/src/workspace/dataplane-adapter.ts +1 -1
  419. package/src/workspace/index.ts +1 -1
  420. package/src/workspace/strategies/index.ts +18 -0
  421. package/src/workspace/strategies/optimistic.ts +136 -0
  422. package/src/workspace/strategies/queue.ts +81 -0
  423. package/src/workspace/strategies/registry.ts +89 -0
  424. package/src/workspace/strategies/trunk.ts +123 -0
  425. package/src/workspace/strategies/types.ts +145 -0
  426. package/src/workspace/types.ts +1 -1
  427. package/src/workspace/workspace-manager.ts +1 -1
  428. package/.claude/settings.local.json +0 -59
  429. package/dist/map/utils/address-translation.d.ts +0 -99
  430. package/dist/map/utils/address-translation.d.ts.map +0 -1
  431. package/dist/map/utils/address-translation.js +0 -285
  432. package/dist/map/utils/address-translation.js.map +0 -1
  433. package/dist/map/utils/index.d.ts +0 -7
  434. package/dist/map/utils/index.d.ts.map +0 -1
  435. package/dist/map/utils/index.js +0 -7
  436. package/dist/map/utils/index.js.map +0 -1
  437. package/openspec/AGENTS.md +0 -456
  438. package/openspec/changes/archive/2025-12-21-add-mvp-foundation/design.md +0 -128
  439. package/openspec/changes/archive/2025-12-21-add-mvp-foundation/proposal.md +0 -49
  440. package/openspec/changes/archive/2025-12-21-add-mvp-foundation/specs/agent-manager/spec.md +0 -150
  441. package/openspec/changes/archive/2025-12-21-add-mvp-foundation/specs/cli-api/spec.md +0 -258
  442. package/openspec/changes/archive/2025-12-21-add-mvp-foundation/specs/event-store/spec.md +0 -160
  443. package/openspec/changes/archive/2025-12-21-add-mvp-foundation/specs/mcp-tools/spec.md +0 -224
  444. package/openspec/changes/archive/2025-12-21-add-mvp-foundation/specs/message-router/spec.md +0 -153
  445. package/openspec/changes/archive/2025-12-21-add-mvp-foundation/specs/task-manager/spec.md +0 -136
  446. package/openspec/changes/archive/2025-12-21-add-mvp-foundation/tasks.md +0 -147
  447. package/openspec/project.md +0 -31
  448. package/openspec/specs/agent-manager/spec.md +0 -154
  449. package/openspec/specs/cli-api/spec.md +0 -262
  450. package/openspec/specs/event-store/spec.md +0 -164
  451. package/openspec/specs/mcp-tools/spec.md +0 -228
  452. package/openspec/specs/message-router/spec.md +0 -157
  453. package/openspec/specs/task-manager/spec.md +0 -140
  454. package/references/acp-factory-ref/CHANGELOG.md +0 -33
  455. package/references/acp-factory-ref/LICENSE +0 -21
  456. package/references/acp-factory-ref/README.md +0 -341
  457. package/references/acp-factory-ref/package-lock.json +0 -3102
  458. package/references/acp-factory-ref/package.json +0 -96
  459. package/references/acp-factory-ref/python/CHANGELOG.md +0 -33
  460. package/references/acp-factory-ref/python/LICENSE +0 -21
  461. package/references/acp-factory-ref/python/Makefile +0 -57
  462. package/references/acp-factory-ref/python/README.md +0 -253
  463. package/references/acp-factory-ref/python/pyproject.toml +0 -73
  464. package/references/acp-factory-ref/python/tests/__init__.py +0 -0
  465. package/references/acp-factory-ref/python/tests/e2e/__init__.py +0 -1
  466. package/references/acp-factory-ref/python/tests/e2e/test_codex_e2e.py +0 -349
  467. package/references/acp-factory-ref/python/tests/e2e/test_gemini_e2e.py +0 -165
  468. package/references/acp-factory-ref/python/tests/e2e/test_opencode_e2e.py +0 -296
  469. package/references/acp-factory-ref/python/tests/test_client_handler.py +0 -543
  470. package/references/acp-factory-ref/python/tests/test_pushable.py +0 -199
  471. package/references/claude-code-acp/.github/workflows/ci.yml +0 -45
  472. package/references/claude-code-acp/.github/workflows/publish.yml +0 -34
  473. package/references/claude-code-acp/.prettierrc.json +0 -4
  474. package/references/claude-code-acp/CHANGELOG.md +0 -249
  475. package/references/claude-code-acp/LICENSE +0 -222
  476. package/references/claude-code-acp/README.md +0 -53
  477. package/references/claude-code-acp/docs/RELEASES.md +0 -24
  478. package/references/claude-code-acp/eslint.config.js +0 -48
  479. package/references/claude-code-acp/package-lock.json +0 -4570
  480. package/references/claude-code-acp/package.json +0 -88
  481. package/references/claude-code-acp/scripts/release.sh +0 -119
  482. package/references/claude-code-acp/src/acp-agent.ts +0 -2065
  483. package/references/claude-code-acp/src/index.ts +0 -26
  484. package/references/claude-code-acp/src/lib.ts +0 -38
  485. package/references/claude-code-acp/src/mcp-server.ts +0 -911
  486. package/references/claude-code-acp/src/settings.ts +0 -522
  487. package/references/claude-code-acp/src/tests/.claude/commands/quick-math.md +0 -5
  488. package/references/claude-code-acp/src/tests/.claude/commands/say-hello.md +0 -6
  489. package/references/claude-code-acp/src/tests/acp-agent-fork.test.ts +0 -479
  490. package/references/claude-code-acp/src/tests/acp-agent.test.ts +0 -1502
  491. package/references/claude-code-acp/src/tests/extract-lines.test.ts +0 -103
  492. package/references/claude-code-acp/src/tests/fork-session.test.ts +0 -335
  493. package/references/claude-code-acp/src/tests/replace-and-calculate-location.test.ts +0 -334
  494. package/references/claude-code-acp/src/tests/settings.test.ts +0 -617
  495. package/references/claude-code-acp/src/tests/skills-options.test.ts +0 -187
  496. package/references/claude-code-acp/src/tests/tools.test.ts +0 -318
  497. package/references/claude-code-acp/src/tests/typescript-declarations.test.ts +0 -558
  498. package/references/claude-code-acp/src/tools.ts +0 -819
  499. package/references/claude-code-acp/src/utils.ts +0 -171
  500. package/references/claude-code-acp/tsconfig.json +0 -18
  501. package/references/claude-code-acp/vitest.config.ts +0 -19
  502. package/references/multi-agent-protocol/.sudocode/issues.jsonl +0 -82
  503. package/references/multi-agent-protocol/.sudocode/specs.jsonl +0 -9
  504. package/references/multi-agent-protocol/LICENSE +0 -21
  505. package/references/multi-agent-protocol/README.md +0 -113
  506. package/references/multi-agent-protocol/docs/00-design-specification.md +0 -460
  507. package/references/multi-agent-protocol/docs/01-open-questions.md +0 -1050
  508. package/references/multi-agent-protocol/docs/02-wire-protocol.md +0 -296
  509. package/references/multi-agent-protocol/docs/03-streaming-semantics.md +0 -252
  510. package/references/multi-agent-protocol/docs/04-error-handling.md +0 -231
  511. package/references/multi-agent-protocol/docs/05-connection-model.md +0 -244
  512. package/references/multi-agent-protocol/docs/06-visibility-permissions.md +0 -243
  513. package/references/multi-agent-protocol/docs/07-federation.md +0 -259
  514. package/references/multi-agent-protocol/docs/08-macro-agent-migration.md +0 -253
  515. package/references/multi-agent-protocol/package-lock.json +0 -3239
  516. package/references/multi-agent-protocol/package.json +0 -56
  517. package/references/multi-agent-protocol/schema/meta.json +0 -337
  518. package/references/multi-agent-protocol/schema/schema.json +0 -1828
@@ -0,0 +1,254 @@
1
+ /**
2
+ * Agent Registry Tests
3
+ */
4
+
5
+ import { describe, it, expect } from "vitest";
6
+
7
+ import {
8
+ AgentRegistry,
9
+ createAgentRegistry,
10
+ BUILTIN_AGENTS,
11
+ } from "../registry.js";
12
+ import type { CLIAgentDefinition } from "../types.js";
13
+
14
+ describe("AgentRegistry", () => {
15
+ // ===========================================================================
16
+ // Built-in Agents
17
+ // ===========================================================================
18
+
19
+ describe("BUILTIN_AGENTS", () => {
20
+ it("contains all 6 known agents", () => {
21
+ expect(BUILTIN_AGENTS).toHaveLength(6);
22
+ });
23
+
24
+ it("includes claude-code", () => {
25
+ const agent = BUILTIN_AGENTS.find((a) => a.id === "claude-code");
26
+ expect(agent).toBeDefined();
27
+ expect(agent!.binary).toBe("claude");
28
+ expect(agent!.vendor).toBe("Anthropic");
29
+ });
30
+
31
+ it("includes codex", () => {
32
+ const agent = BUILTIN_AGENTS.find((a) => a.id === "codex");
33
+ expect(agent).toBeDefined();
34
+ expect(agent!.binary).toBe("codex");
35
+ expect(agent!.vendor).toBe("OpenAI");
36
+ });
37
+
38
+ it("includes gemini-cli", () => {
39
+ const agent = BUILTIN_AGENTS.find((a) => a.id === "gemini-cli");
40
+ expect(agent).toBeDefined();
41
+ expect(agent!.binary).toBe("gemini");
42
+ expect(agent!.vendor).toBe("Google");
43
+ });
44
+
45
+ it("includes opencode", () => {
46
+ const agent = BUILTIN_AGENTS.find((a) => a.id === "opencode");
47
+ expect(agent).toBeDefined();
48
+ expect(agent!.binary).toBe("opencode");
49
+ expect(agent!.vendor).toBe("Anomaly");
50
+ });
51
+
52
+ it("includes aider", () => {
53
+ const agent = BUILTIN_AGENTS.find((a) => a.id === "aider");
54
+ expect(agent).toBeDefined();
55
+ expect(agent!.binary).toBe("aider");
56
+ expect(agent!.vendor).toBe("Aider");
57
+ });
58
+
59
+ it("includes goose", () => {
60
+ const agent = BUILTIN_AGENTS.find((a) => a.id === "goose");
61
+ expect(agent).toBeDefined();
62
+ expect(agent!.binary).toBe("goose");
63
+ expect(agent!.vendor).toBe("Block");
64
+ });
65
+
66
+ it("all agents have required fields", () => {
67
+ for (const agent of BUILTIN_AGENTS) {
68
+ expect(agent.id).toBeTruthy();
69
+ expect(agent.name).toBeTruthy();
70
+ expect(agent.description).toBeTruthy();
71
+ expect(agent.binary).toBeTruthy();
72
+ expect(agent.versionArgs).toBeDefined();
73
+ expect(agent.versionArgs.length).toBeGreaterThan(0);
74
+ expect(agent.headless).toBeDefined();
75
+ expect(agent.headless.promptFlag).toBeDefined();
76
+ expect(agent.vendor).toBeTruthy();
77
+ }
78
+ });
79
+ });
80
+
81
+ // ===========================================================================
82
+ // Registry Operations
83
+ // ===========================================================================
84
+
85
+ describe("constructor", () => {
86
+ it("creates registry with built-in agents", () => {
87
+ const registry = new AgentRegistry();
88
+ expect(registry.size).toBe(6);
89
+ });
90
+
91
+ it("creates registry with additional custom agents", () => {
92
+ const custom: CLIAgentDefinition = {
93
+ id: "custom-agent",
94
+ name: "Custom Agent",
95
+ description: "A custom agent",
96
+ binary: "custom",
97
+ versionArgs: ["--version"],
98
+ headless: { promptFlag: "--prompt" },
99
+ vendor: "Custom",
100
+ };
101
+ const registry = new AgentRegistry([custom]);
102
+ expect(registry.size).toBe(7);
103
+ expect(registry.get("custom-agent")).toEqual(custom);
104
+ });
105
+
106
+ it("custom agents can override built-in agents", () => {
107
+ const override: CLIAgentDefinition = {
108
+ id: "claude-code",
109
+ name: "Custom Claude",
110
+ description: "Overridden",
111
+ binary: "custom-claude",
112
+ versionArgs: ["--version"],
113
+ headless: { promptFlag: "-p" },
114
+ vendor: "Custom",
115
+ };
116
+ const registry = new AgentRegistry([override]);
117
+ expect(registry.get("claude-code")!.binary).toBe("custom-claude");
118
+ });
119
+ });
120
+
121
+ describe("get()", () => {
122
+ it("returns agent definition by ID", () => {
123
+ const registry = new AgentRegistry();
124
+ const agent = registry.get("claude-code");
125
+ expect(agent).toBeDefined();
126
+ expect(agent!.id).toBe("claude-code");
127
+ });
128
+
129
+ it("returns undefined for unknown ID", () => {
130
+ const registry = new AgentRegistry();
131
+ expect(registry.get("nonexistent")).toBeUndefined();
132
+ });
133
+ });
134
+
135
+ describe("has()", () => {
136
+ it("returns true for existing agent", () => {
137
+ const registry = new AgentRegistry();
138
+ expect(registry.has("claude-code")).toBe(true);
139
+ });
140
+
141
+ it("returns false for unknown agent", () => {
142
+ const registry = new AgentRegistry();
143
+ expect(registry.has("nonexistent")).toBe(false);
144
+ });
145
+ });
146
+
147
+ describe("register()", () => {
148
+ it("adds a new agent definition", () => {
149
+ const registry = new AgentRegistry();
150
+ const custom: CLIAgentDefinition = {
151
+ id: "new-agent",
152
+ name: "New Agent",
153
+ description: "Newly added",
154
+ binary: "new",
155
+ versionArgs: ["--version"],
156
+ headless: { promptFlag: "--prompt" },
157
+ vendor: "New",
158
+ };
159
+ registry.register(custom);
160
+ expect(registry.has("new-agent")).toBe(true);
161
+ expect(registry.get("new-agent")).toEqual(custom);
162
+ });
163
+
164
+ it("overrides existing agent definition", () => {
165
+ const registry = new AgentRegistry();
166
+ const override: CLIAgentDefinition = {
167
+ id: "claude-code",
168
+ name: "Custom Claude",
169
+ description: "Overridden",
170
+ binary: "custom",
171
+ versionArgs: ["--version"],
172
+ headless: { promptFlag: "-p" },
173
+ vendor: "Custom",
174
+ };
175
+ registry.register(override);
176
+ expect(registry.get("claude-code")!.name).toBe("Custom Claude");
177
+ });
178
+ });
179
+
180
+ describe("remove()", () => {
181
+ it("removes an existing agent definition", () => {
182
+ const registry = new AgentRegistry();
183
+ expect(registry.remove("claude-code")).toBe(true);
184
+ expect(registry.has("claude-code")).toBe(false);
185
+ expect(registry.size).toBe(5);
186
+ });
187
+
188
+ it("returns false for non-existent agent", () => {
189
+ const registry = new AgentRegistry();
190
+ expect(registry.remove("nonexistent")).toBe(false);
191
+ });
192
+ });
193
+
194
+ describe("list()", () => {
195
+ it("returns all registered agents", () => {
196
+ const registry = new AgentRegistry();
197
+ const agents = registry.list();
198
+ expect(agents).toHaveLength(6);
199
+ const ids = agents.map((a) => a.id);
200
+ expect(ids).toContain("claude-code");
201
+ expect(ids).toContain("codex");
202
+ expect(ids).toContain("gemini-cli");
203
+ expect(ids).toContain("opencode");
204
+ expect(ids).toContain("aider");
205
+ expect(ids).toContain("goose");
206
+ });
207
+ });
208
+
209
+ describe("listEnabled()", () => {
210
+ it("returns all agents when no disabled list", () => {
211
+ const registry = new AgentRegistry();
212
+ expect(registry.listEnabled()).toHaveLength(6);
213
+ });
214
+
215
+ it("returns all agents when disabled list is empty", () => {
216
+ const registry = new AgentRegistry();
217
+ expect(registry.listEnabled([])).toHaveLength(6);
218
+ });
219
+
220
+ it("excludes disabled agents", () => {
221
+ const registry = new AgentRegistry();
222
+ const enabled = registry.listEnabled(["goose", "aider"]);
223
+ expect(enabled).toHaveLength(4);
224
+ const ids = enabled.map((a) => a.id);
225
+ expect(ids).not.toContain("goose");
226
+ expect(ids).not.toContain("aider");
227
+ });
228
+ });
229
+
230
+ // ===========================================================================
231
+ // Factory
232
+ // ===========================================================================
233
+
234
+ describe("createAgentRegistry()", () => {
235
+ it("creates registry with defaults", () => {
236
+ const registry = createAgentRegistry();
237
+ expect(registry.size).toBe(6);
238
+ });
239
+
240
+ it("creates registry with additional agents", () => {
241
+ const custom: CLIAgentDefinition = {
242
+ id: "extra",
243
+ name: "Extra",
244
+ description: "Extra agent",
245
+ binary: "extra",
246
+ versionArgs: ["--version"],
247
+ headless: { promptFlag: "--prompt" },
248
+ vendor: "Extra",
249
+ };
250
+ const registry = createAgentRegistry([custom]);
251
+ expect(registry.size).toBe(7);
252
+ });
253
+ });
254
+ });
@@ -0,0 +1,90 @@
1
+ /**
2
+ * Command Builder
3
+ *
4
+ * Constructs headless invocation commands for CLI coding agents
5
+ * based on their definitions.
6
+ */
7
+
8
+ import type {
9
+ CLIAgentDefinition,
10
+ SpawnCommand,
11
+ SpawnCommandOptions,
12
+ } from "./types.js";
13
+ import { AgentDetectionError } from "./types.js";
14
+
15
+ /**
16
+ * Build a headless spawn command for a CLI agent.
17
+ *
18
+ * Constructs the command and arguments array based on the agent definition's
19
+ * headless configuration, model flag, and working directory flag.
20
+ *
21
+ * @param definition - The CLI agent definition
22
+ * @param task - The prompt/task to pass to the agent
23
+ * @param options - Optional model and cwd overrides
24
+ * @returns The command and args array ready for child_process.spawn()
25
+ *
26
+ * @example
27
+ * ```ts
28
+ * const cmd = buildSpawnCommand(claudeCode, "Fix the auth bug", { model: "claude-sonnet-4-5" });
29
+ * // { command: "claude", args: ["-p", "Fix the auth bug", "--output-format", "stream-json", "--model", "claude-sonnet-4-5"] }
30
+ * ```
31
+ */
32
+ export function buildSpawnCommand(
33
+ definition: CLIAgentDefinition,
34
+ task: string,
35
+ options?: SpawnCommandOptions
36
+ ): SpawnCommand {
37
+ if (!task) {
38
+ throw new AgentDetectionError(
39
+ "Task prompt is required",
40
+ "DETECTION_FAILED"
41
+ );
42
+ }
43
+
44
+ const args: string[] = [];
45
+
46
+ // 1. Subcommand (e.g., "exec", "run")
47
+ if (definition.headless.subcommand) {
48
+ args.push(definition.headless.subcommand);
49
+ }
50
+
51
+ // 2. Default flags (e.g., "--full-auto", "--yes")
52
+ if (definition.headless.defaultFlags) {
53
+ args.push(...definition.headless.defaultFlags);
54
+ }
55
+
56
+ // 3. Model flag
57
+ if (options?.model && definition.modelFlag) {
58
+ args.push(definition.modelFlag, options.model);
59
+ }
60
+
61
+ // 4. Working directory flag
62
+ if (options?.cwd && definition.cwdFlag) {
63
+ args.push(definition.cwdFlag, options.cwd);
64
+ }
65
+
66
+ // 5. Prompt (flag-based or positional)
67
+ if (definition.headless.promptFlag) {
68
+ args.push(definition.headless.promptFlag, task);
69
+ } else {
70
+ args.push(task);
71
+ }
72
+
73
+ return { command: definition.binary, args };
74
+ }
75
+
76
+ /**
77
+ * Format a spawn command as a single shell string (for display/logging).
78
+ */
79
+ export function formatSpawnCommand(spawnCommand: SpawnCommand): string {
80
+ const parts = [spawnCommand.command];
81
+ for (const arg of spawnCommand.args) {
82
+ // Quote arguments that contain spaces
83
+ if (arg.includes(" ")) {
84
+ parts.push(`"${arg}"`);
85
+ } else {
86
+ parts.push(arg);
87
+ }
88
+ }
89
+ return parts.join(" ");
90
+ }
@@ -0,0 +1,307 @@
1
+ /**
2
+ * CLI Agent Detector
3
+ *
4
+ * Detects installed CLI coding agents by checking PATH availability
5
+ * and querying version information. Results are cached with configurable TTL.
6
+ */
7
+
8
+ import { execFile } from "node:child_process";
9
+ import { promisify } from "node:util";
10
+
11
+ import type {
12
+ CLIAgentDefinition,
13
+ DetectedAgent,
14
+ DetectionResult,
15
+ AgentDetectionConfig,
16
+ } from "./types.js";
17
+ import { AgentDetectionError } from "./types.js";
18
+ import { AgentRegistry, createAgentRegistry } from "./registry.js";
19
+
20
+ const execFileAsync = promisify(execFile);
21
+
22
+ // =============================================================================
23
+ // Constants
24
+ // =============================================================================
25
+
26
+ const DEFAULT_CACHE_TTL_MS = 60_000;
27
+ const DEFAULT_WHICH_TIMEOUT_MS = 5_000;
28
+ const DEFAULT_VERSION_TIMEOUT_MS = 10_000;
29
+
30
+ /** Regex to parse version strings like "1.2.3", "0.82.1-beta", etc. */
31
+ const VERSION_REGEX = /\d+\.\d+[\.\d]*/;
32
+
33
+ // =============================================================================
34
+ // Agent Detector
35
+ // =============================================================================
36
+
37
+ /**
38
+ * Detects installed CLI coding agents on the system.
39
+ *
40
+ * Runs binary lookups and version checks in parallel, caches results,
41
+ * and provides query methods for available agents.
42
+ */
43
+ export class AgentDetector {
44
+ private readonly registry: AgentRegistry;
45
+ private readonly cacheTtlMs: number;
46
+ private readonly whichTimeoutMs: number;
47
+ private readonly versionTimeoutMs: number;
48
+ private readonly disabledAgents: Set<string>;
49
+
50
+ private cachedResult: DetectionResult | null = null;
51
+ private detectionInProgress: Promise<DetectionResult> | null = null;
52
+
53
+ constructor(config?: AgentDetectionConfig) {
54
+ this.cacheTtlMs = config?.cacheTtlMs ?? DEFAULT_CACHE_TTL_MS;
55
+ this.whichTimeoutMs = config?.whichTimeoutMs ?? DEFAULT_WHICH_TIMEOUT_MS;
56
+ this.versionTimeoutMs =
57
+ config?.versionTimeoutMs ?? DEFAULT_VERSION_TIMEOUT_MS;
58
+ this.disabledAgents = new Set(config?.disabledAgents ?? []);
59
+
60
+ this.registry = createAgentRegistry(config?.additionalAgents);
61
+ }
62
+
63
+ /**
64
+ * Get the underlying agent registry.
65
+ */
66
+ getRegistry(): AgentRegistry {
67
+ return this.registry;
68
+ }
69
+
70
+ /**
71
+ * Detect all installed agents.
72
+ *
73
+ * Returns cached results if available and not expired.
74
+ * If a detection is already in progress, returns its promise.
75
+ */
76
+ async detect(options?: { refresh?: boolean }): Promise<DetectionResult> {
77
+ // Return cached result if valid
78
+ if (!options?.refresh && this.cachedResult && !this.isCacheExpired()) {
79
+ return this.cachedResult;
80
+ }
81
+
82
+ // If detection is already running, return the in-progress promise
83
+ if (this.detectionInProgress) {
84
+ return this.detectionInProgress;
85
+ }
86
+
87
+ // Start a new detection scan
88
+ this.detectionInProgress = this.runDetection();
89
+
90
+ try {
91
+ const result = await this.detectionInProgress;
92
+ this.cachedResult = result;
93
+ return result;
94
+ } finally {
95
+ this.detectionInProgress = null;
96
+ }
97
+ }
98
+
99
+ /**
100
+ * Get available agents (installed only by default).
101
+ */
102
+ async getAvailableAgents(options?: {
103
+ refresh?: boolean;
104
+ includeNotInstalled?: boolean;
105
+ }): Promise<DetectionResult> {
106
+ const result = await this.detect({ refresh: options?.refresh });
107
+
108
+ if (options?.includeNotInstalled) {
109
+ return result;
110
+ }
111
+
112
+ return {
113
+ ...result,
114
+ agents: result.agents.filter((a) => a.installed),
115
+ };
116
+ }
117
+
118
+ /**
119
+ * Get a specific detected agent by ID.
120
+ */
121
+ async getAgent(id: string): Promise<DetectedAgent | undefined> {
122
+ const result = await this.detect();
123
+ return result.agents.find((a) => a.id === id);
124
+ }
125
+
126
+ /**
127
+ * Check if a specific agent is installed.
128
+ */
129
+ async isInstalled(id: string): Promise<boolean> {
130
+ const agent = await this.getAgent(id);
131
+ return agent?.installed ?? false;
132
+ }
133
+
134
+ /**
135
+ * Get the definition for an agent, throwing if not found.
136
+ */
137
+ getDefinition(id: string): CLIAgentDefinition {
138
+ const def = this.registry.get(id);
139
+ if (!def) {
140
+ throw new AgentDetectionError(
141
+ `Unknown agent backend: ${id}`,
142
+ "UNKNOWN_AGENT",
143
+ id
144
+ );
145
+ }
146
+ return def;
147
+ }
148
+
149
+ /**
150
+ * Invalidate the detection cache.
151
+ */
152
+ invalidateCache(): void {
153
+ this.cachedResult = null;
154
+ }
155
+
156
+ /**
157
+ * Get partial results if detection is in progress, or cached results.
158
+ * Returns null if no results are available yet.
159
+ */
160
+ getCachedResult(): DetectionResult | null {
161
+ return this.cachedResult;
162
+ }
163
+
164
+ /**
165
+ * Whether a detection scan is currently running.
166
+ */
167
+ isDetecting(): boolean {
168
+ return this.detectionInProgress !== null;
169
+ }
170
+
171
+ // ===========================================================================
172
+ // Private
173
+ // ===========================================================================
174
+
175
+ private isCacheExpired(): boolean {
176
+ if (!this.cachedResult) return true;
177
+ const maxAge = Math.max(...this.cachedResult.agents.map((a) => a.detectedAt));
178
+ return Date.now() - maxAge > this.cacheTtlMs;
179
+ }
180
+
181
+ private async runDetection(): Promise<DetectionResult> {
182
+ const definitions = this.registry.listEnabled(
183
+ Array.from(this.disabledAgents)
184
+ );
185
+ const startTime = Date.now();
186
+
187
+ const results = await Promise.allSettled(
188
+ definitions.map((def) => this.detectAgent(def))
189
+ );
190
+
191
+ const agents: DetectedAgent[] = results.map((result, i) => {
192
+ if (result.status === "fulfilled") {
193
+ return result.value;
194
+ }
195
+ // On failure, return a not-installed entry
196
+ return {
197
+ id: definitions[i].id,
198
+ name: definitions[i].name,
199
+ installed: false,
200
+ definition: definitions[i],
201
+ detectedAt: Date.now(),
202
+ };
203
+ });
204
+
205
+ return {
206
+ agents,
207
+ scanned: definitions.length,
208
+ durationMs: Date.now() - startTime,
209
+ };
210
+ }
211
+
212
+ private async detectAgent(
213
+ definition: CLIAgentDefinition
214
+ ): Promise<DetectedAgent> {
215
+ const detectedAt = Date.now();
216
+
217
+ // Step 1: Check if binary exists on PATH
218
+ const binaryPath = await this.findBinary(definition.binary);
219
+
220
+ if (!binaryPath) {
221
+ return {
222
+ id: definition.id,
223
+ name: definition.name,
224
+ installed: false,
225
+ definition,
226
+ detectedAt,
227
+ };
228
+ }
229
+
230
+ // Step 2: Get version string
231
+ const version = await this.getVersion(definition);
232
+
233
+ return {
234
+ id: definition.id,
235
+ name: definition.name,
236
+ installed: true,
237
+ version: version ?? undefined,
238
+ path: binaryPath,
239
+ definition,
240
+ detectedAt,
241
+ };
242
+ }
243
+
244
+ private async findBinary(binary: string): Promise<string | null> {
245
+ try {
246
+ const { stdout } = await execFileAsync("which", [binary], {
247
+ timeout: this.whichTimeoutMs,
248
+ });
249
+ const path = stdout.trim();
250
+ return path || null;
251
+ } catch {
252
+ // Binary not found or which failed
253
+ try {
254
+ // Fallback to command -v
255
+ const { stdout } = await execFileAsync(
256
+ "/bin/sh",
257
+ ["-c", `command -v ${binary}`],
258
+ { timeout: this.whichTimeoutMs }
259
+ );
260
+ const path = stdout.trim();
261
+ return path || null;
262
+ } catch {
263
+ return null;
264
+ }
265
+ }
266
+ }
267
+
268
+ private async getVersion(
269
+ definition: CLIAgentDefinition
270
+ ): Promise<string | null> {
271
+ try {
272
+ const { stdout, stderr } = await execFileAsync(
273
+ definition.binary,
274
+ definition.versionArgs,
275
+ { timeout: this.versionTimeoutMs }
276
+ );
277
+
278
+ // Some tools output version to stderr
279
+ const output = stdout || stderr;
280
+ return parseVersion(output);
281
+ } catch {
282
+ return null;
283
+ }
284
+ }
285
+ }
286
+
287
+ // =============================================================================
288
+ // Helpers
289
+ // =============================================================================
290
+
291
+ /**
292
+ * Parse a version string from command output.
293
+ * Extracts the first semver-like pattern (e.g., "1.2.3").
294
+ */
295
+ export function parseVersion(output: string): string | null {
296
+ const match = output.match(VERSION_REGEX);
297
+ return match ? match[0] : null;
298
+ }
299
+
300
+ /**
301
+ * Create a new AgentDetector with the given configuration.
302
+ */
303
+ export function createAgentDetector(
304
+ config?: AgentDetectionConfig
305
+ ): AgentDetector {
306
+ return new AgentDetector(config);
307
+ }
@@ -0,0 +1,36 @@
1
+ /**
2
+ * CLI Agent Auto-Detection Module
3
+ *
4
+ * Detects installed CLI coding agents, exposes them via a registry,
5
+ * and provides command construction for headless invocation.
6
+ */
7
+
8
+ // Types
9
+ export type {
10
+ CLIAgentDefinition,
11
+ HeadlessConfig,
12
+ DetectedAgent,
13
+ DetectionResult,
14
+ SpawnCommand,
15
+ SpawnCommandOptions,
16
+ AgentDetectionConfig,
17
+ AgentDetectionErrorCode,
18
+ } from "./types.js";
19
+ export { AgentDetectionError } from "./types.js";
20
+
21
+ // Registry
22
+ export {
23
+ AgentRegistry,
24
+ createAgentRegistry,
25
+ BUILTIN_AGENTS,
26
+ } from "./registry.js";
27
+
28
+ // Detector
29
+ export {
30
+ AgentDetector,
31
+ createAgentDetector,
32
+ parseVersion,
33
+ } from "./detector.js";
34
+
35
+ // Command builder
36
+ export { buildSpawnCommand, formatSpawnCommand } from "./command-builder.js";