@ag-eco/agentplate-cli 0.13.0

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 (455) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +462 -0
  3. package/agents/ap-co-creation.md +90 -0
  4. package/agents/builder.md +144 -0
  5. package/agents/coordinator.md +377 -0
  6. package/agents/lead.md +435 -0
  7. package/agents/merger.md +164 -0
  8. package/agents/monitor.md +214 -0
  9. package/agents/orchestrator.md +239 -0
  10. package/agents/reviewer.md +140 -0
  11. package/agents/scout.md +125 -0
  12. package/agents/supervisor.md +427 -0
  13. package/package.json +66 -0
  14. package/src/agents/capabilities.test.ts +85 -0
  15. package/src/agents/capabilities.ts +125 -0
  16. package/src/agents/checkpoint.test.ts +88 -0
  17. package/src/agents/checkpoint.ts +101 -0
  18. package/src/agents/copilot-hooks-deployer.test.ts +162 -0
  19. package/src/agents/copilot-hooks-deployer.ts +93 -0
  20. package/src/agents/guard-rules.test.ts +372 -0
  21. package/src/agents/guard-rules.ts +97 -0
  22. package/src/agents/headless-mail-injector.test.ts +709 -0
  23. package/src/agents/headless-mail-injector.ts +377 -0
  24. package/src/agents/headless-prompt.test.ts +102 -0
  25. package/src/agents/headless-prompt.ts +68 -0
  26. package/src/agents/hooks-deployer.test.ts +3119 -0
  27. package/src/agents/hooks-deployer.ts +804 -0
  28. package/src/agents/identity.test.ts +604 -0
  29. package/src/agents/identity.ts +384 -0
  30. package/src/agents/lifecycle.test.ts +196 -0
  31. package/src/agents/lifecycle.ts +183 -0
  32. package/src/agents/mail-poll-detect.test.ts +153 -0
  33. package/src/agents/mail-poll-detect.ts +73 -0
  34. package/src/agents/manifest.test.ts +1026 -0
  35. package/src/agents/manifest.ts +376 -0
  36. package/src/agents/overlay.test.ts +1058 -0
  37. package/src/agents/overlay.ts +490 -0
  38. package/src/agents/scope-detect.test.ts +190 -0
  39. package/src/agents/scope-detect.ts +146 -0
  40. package/src/agents/turn-lock.test.ts +181 -0
  41. package/src/agents/turn-lock.ts +235 -0
  42. package/src/agents/turn-runner-dispatch.test.ts +182 -0
  43. package/src/agents/turn-runner-dispatch.ts +105 -0
  44. package/src/agents/turn-runner.test.ts +2312 -0
  45. package/src/agents/turn-runner.ts +1383 -0
  46. package/src/beads/client.test.ts +217 -0
  47. package/src/beads/client.ts +230 -0
  48. package/src/beads/molecules.test.ts +338 -0
  49. package/src/beads/molecules.ts +198 -0
  50. package/src/commands/agents.test.ts +328 -0
  51. package/src/commands/agents.ts +299 -0
  52. package/src/commands/clean.test.ts +797 -0
  53. package/src/commands/clean.ts +791 -0
  54. package/src/commands/completions.test.ts +348 -0
  55. package/src/commands/completions.ts +981 -0
  56. package/src/commands/coordinator.test.ts +2975 -0
  57. package/src/commands/coordinator.ts +1841 -0
  58. package/src/commands/costs.test.ts +1183 -0
  59. package/src/commands/costs.ts +599 -0
  60. package/src/commands/dashboard.test.ts +954 -0
  61. package/src/commands/dashboard.ts +1212 -0
  62. package/src/commands/discover.test.ts +288 -0
  63. package/src/commands/discover.ts +202 -0
  64. package/src/commands/doctor.test.ts +303 -0
  65. package/src/commands/doctor.ts +311 -0
  66. package/src/commands/ecosystem.test.ts +226 -0
  67. package/src/commands/ecosystem.ts +248 -0
  68. package/src/commands/errors.test.ts +654 -0
  69. package/src/commands/errors.ts +197 -0
  70. package/src/commands/feed.test.ts +709 -0
  71. package/src/commands/feed.ts +260 -0
  72. package/src/commands/group.test.ts +475 -0
  73. package/src/commands/group.ts +546 -0
  74. package/src/commands/hooks.test.ts +458 -0
  75. package/src/commands/hooks.ts +263 -0
  76. package/src/commands/init.test.ts +1011 -0
  77. package/src/commands/init.ts +967 -0
  78. package/src/commands/inspect.test.ts +1239 -0
  79. package/src/commands/inspect.ts +648 -0
  80. package/src/commands/log.test.ts +1913 -0
  81. package/src/commands/log.ts +958 -0
  82. package/src/commands/logs.test.ts +801 -0
  83. package/src/commands/logs.ts +483 -0
  84. package/src/commands/mail.test.ts +1501 -0
  85. package/src/commands/mail.ts +848 -0
  86. package/src/commands/merge.test.ts +864 -0
  87. package/src/commands/merge.ts +381 -0
  88. package/src/commands/metrics.test.ts +458 -0
  89. package/src/commands/metrics.ts +129 -0
  90. package/src/commands/monitor.test.ts +191 -0
  91. package/src/commands/monitor.ts +409 -0
  92. package/src/commands/nudge.test.ts +579 -0
  93. package/src/commands/nudge.ts +646 -0
  94. package/src/commands/orchestrator.ts +42 -0
  95. package/src/commands/prime.test.ts +612 -0
  96. package/src/commands/prime.ts +359 -0
  97. package/src/commands/replay.test.ts +757 -0
  98. package/src/commands/replay.ts +231 -0
  99. package/src/commands/run.test.ts +469 -0
  100. package/src/commands/run.ts +353 -0
  101. package/src/commands/serve/agent-actions.test.ts +210 -0
  102. package/src/commands/serve/agent-actions.ts +192 -0
  103. package/src/commands/serve/build.test.ts +202 -0
  104. package/src/commands/serve/build.ts +206 -0
  105. package/src/commands/serve/coordinator-actions.test.ts +339 -0
  106. package/src/commands/serve/coordinator-actions.ts +410 -0
  107. package/src/commands/serve/dev.test.ts +168 -0
  108. package/src/commands/serve/dev.ts +117 -0
  109. package/src/commands/serve/mail-actions.test.ts +312 -0
  110. package/src/commands/serve/mail-actions.ts +167 -0
  111. package/src/commands/serve/rest.test.ts +1680 -0
  112. package/src/commands/serve/rest.ts +1130 -0
  113. package/src/commands/serve/static.ts +51 -0
  114. package/src/commands/serve/ws.test.ts +361 -0
  115. package/src/commands/serve/ws.ts +332 -0
  116. package/src/commands/serve.test.ts +459 -0
  117. package/src/commands/serve.ts +654 -0
  118. package/src/commands/sling.test.ts +1583 -0
  119. package/src/commands/sling.ts +1351 -0
  120. package/src/commands/spec.test.ts +179 -0
  121. package/src/commands/spec.ts +105 -0
  122. package/src/commands/status.test.ts +614 -0
  123. package/src/commands/status.ts +403 -0
  124. package/src/commands/stop.test.ts +964 -0
  125. package/src/commands/stop.ts +319 -0
  126. package/src/commands/supervisor.test.ts +185 -0
  127. package/src/commands/supervisor.ts +537 -0
  128. package/src/commands/trace.test.ts +762 -0
  129. package/src/commands/trace.ts +205 -0
  130. package/src/commands/update.test.ts +466 -0
  131. package/src/commands/update.ts +263 -0
  132. package/src/commands/upgrade.test.ts +48 -0
  133. package/src/commands/upgrade.ts +240 -0
  134. package/src/commands/watch.test.ts +257 -0
  135. package/src/commands/watch.ts +308 -0
  136. package/src/commands/worktree.test.ts +1297 -0
  137. package/src/commands/worktree.ts +451 -0
  138. package/src/config.test.ts +1535 -0
  139. package/src/config.ts +1064 -0
  140. package/src/doctor/agents.test.ts +523 -0
  141. package/src/doctor/agents.ts +399 -0
  142. package/src/doctor/config-check.test.ts +191 -0
  143. package/src/doctor/config-check.ts +183 -0
  144. package/src/doctor/consistency.test.ts +807 -0
  145. package/src/doctor/consistency.ts +347 -0
  146. package/src/doctor/databases.test.ts +350 -0
  147. package/src/doctor/databases.ts +243 -0
  148. package/src/doctor/dependencies.test.ts +296 -0
  149. package/src/doctor/dependencies.ts +272 -0
  150. package/src/doctor/ecosystem.test.ts +308 -0
  151. package/src/doctor/ecosystem.ts +156 -0
  152. package/src/doctor/logs.test.ts +253 -0
  153. package/src/doctor/logs.ts +295 -0
  154. package/src/doctor/merge-queue.test.ts +315 -0
  155. package/src/doctor/merge-queue.ts +167 -0
  156. package/src/doctor/providers.test.ts +409 -0
  157. package/src/doctor/providers.ts +250 -0
  158. package/src/doctor/serve.test.ts +95 -0
  159. package/src/doctor/serve.ts +86 -0
  160. package/src/doctor/structure.test.ts +423 -0
  161. package/src/doctor/structure.ts +285 -0
  162. package/src/doctor/types.ts +43 -0
  163. package/src/doctor/version.test.ts +241 -0
  164. package/src/doctor/version.ts +132 -0
  165. package/src/doctor/watchdog.test.ts +167 -0
  166. package/src/doctor/watchdog.ts +214 -0
  167. package/src/e2e/init-sling-lifecycle.test.ts +283 -0
  168. package/src/errors.test.ts +350 -0
  169. package/src/errors.ts +217 -0
  170. package/src/events/store.test.ts +660 -0
  171. package/src/events/store.ts +369 -0
  172. package/src/events/tailer.test.ts +719 -0
  173. package/src/events/tailer.ts +332 -0
  174. package/src/events/tool-filter.test.ts +330 -0
  175. package/src/events/tool-filter.ts +126 -0
  176. package/src/index.ts +533 -0
  177. package/src/insights/analyzer.test.ts +466 -0
  178. package/src/insights/analyzer.ts +203 -0
  179. package/src/insights/quality-gates.test.ts +141 -0
  180. package/src/insights/quality-gates.ts +156 -0
  181. package/src/json.test.ts +72 -0
  182. package/src/json.ts +53 -0
  183. package/src/loam/client.test.ts +752 -0
  184. package/src/loam/client.ts +664 -0
  185. package/src/logging/color.test.ts +252 -0
  186. package/src/logging/color.ts +105 -0
  187. package/src/logging/format.test.ts +110 -0
  188. package/src/logging/format.ts +255 -0
  189. package/src/logging/logger.test.ts +814 -0
  190. package/src/logging/logger.ts +266 -0
  191. package/src/logging/reporter.test.ts +259 -0
  192. package/src/logging/reporter.ts +110 -0
  193. package/src/logging/sanitizer.test.ts +190 -0
  194. package/src/logging/sanitizer.ts +57 -0
  195. package/src/logging/theme.ts +140 -0
  196. package/src/mail/broadcast.test.ts +204 -0
  197. package/src/mail/broadcast.ts +92 -0
  198. package/src/mail/client.test.ts +774 -0
  199. package/src/mail/client.ts +236 -0
  200. package/src/mail/store.test.ts +898 -0
  201. package/src/mail/store.ts +425 -0
  202. package/src/merge/lock.test.ts +149 -0
  203. package/src/merge/lock.ts +140 -0
  204. package/src/merge/predict.test.ts +387 -0
  205. package/src/merge/predict.ts +249 -0
  206. package/src/merge/queue.test.ts +426 -0
  207. package/src/merge/queue.ts +246 -0
  208. package/src/merge/resolver.test.ts +1993 -0
  209. package/src/merge/resolver.ts +926 -0
  210. package/src/metrics/pricing.test.ts +258 -0
  211. package/src/metrics/pricing.ts +135 -0
  212. package/src/metrics/store.test.ts +978 -0
  213. package/src/metrics/store.ts +501 -0
  214. package/src/metrics/summary.test.ts +398 -0
  215. package/src/metrics/summary.ts +178 -0
  216. package/src/metrics/transcript.test.ts +483 -0
  217. package/src/metrics/transcript.ts +114 -0
  218. package/src/runtimes/__fixtures__/claude-stream-fixture.ts +22 -0
  219. package/src/runtimes/aider.test.ts +124 -0
  220. package/src/runtimes/aider.ts +147 -0
  221. package/src/runtimes/amp.test.ts +164 -0
  222. package/src/runtimes/amp.ts +154 -0
  223. package/src/runtimes/claude.test.ts +1474 -0
  224. package/src/runtimes/claude.ts +579 -0
  225. package/src/runtimes/codex.test.ts +805 -0
  226. package/src/runtimes/codex.ts +273 -0
  227. package/src/runtimes/connections.test.ts +214 -0
  228. package/src/runtimes/connections.ts +103 -0
  229. package/src/runtimes/copilot.test.ts +707 -0
  230. package/src/runtimes/copilot.ts +316 -0
  231. package/src/runtimes/cursor.test.ts +497 -0
  232. package/src/runtimes/cursor.ts +205 -0
  233. package/src/runtimes/gemini.test.ts +537 -0
  234. package/src/runtimes/gemini.ts +243 -0
  235. package/src/runtimes/goose.test.ts +133 -0
  236. package/src/runtimes/goose.ts +157 -0
  237. package/src/runtimes/headless-connection.test.ts +264 -0
  238. package/src/runtimes/headless-connection.ts +158 -0
  239. package/src/runtimes/opencode.test.ts +325 -0
  240. package/src/runtimes/opencode.ts +188 -0
  241. package/src/runtimes/pi-guards.test.ts +486 -0
  242. package/src/runtimes/pi-guards.ts +367 -0
  243. package/src/runtimes/pi.test.ts +789 -0
  244. package/src/runtimes/pi.ts +305 -0
  245. package/src/runtimes/registry.test.ts +196 -0
  246. package/src/runtimes/registry.ts +99 -0
  247. package/src/runtimes/sapling.test.ts +1267 -0
  248. package/src/runtimes/sapling.ts +710 -0
  249. package/src/runtimes/types.ts +266 -0
  250. package/src/schema-consistency.test.ts +246 -0
  251. package/src/sessions/compat.test.ts +281 -0
  252. package/src/sessions/compat.ts +105 -0
  253. package/src/sessions/store.test.ts +1748 -0
  254. package/src/sessions/store.ts +858 -0
  255. package/src/test-helpers.test.ts +124 -0
  256. package/src/test-helpers.ts +145 -0
  257. package/src/test-setup.test.ts +31 -0
  258. package/src/test-setup.ts +28 -0
  259. package/src/tools/loam/api.ts +368 -0
  260. package/src/tools/loam/cli.ts +278 -0
  261. package/src/tools/loam/commands/add.ts +52 -0
  262. package/src/tools/loam/commands/archive.ts +214 -0
  263. package/src/tools/loam/commands/audit.ts +276 -0
  264. package/src/tools/loam/commands/compact.ts +1062 -0
  265. package/src/tools/loam/commands/completions.ts +79 -0
  266. package/src/tools/loam/commands/config.ts +381 -0
  267. package/src/tools/loam/commands/delete-domain.ts +121 -0
  268. package/src/tools/loam/commands/delete.ts +316 -0
  269. package/src/tools/loam/commands/diff.ts +200 -0
  270. package/src/tools/loam/commands/doctor.ts +1113 -0
  271. package/src/tools/loam/commands/edit.ts +226 -0
  272. package/src/tools/loam/commands/init.ts +31 -0
  273. package/src/tools/loam/commands/learn.ts +179 -0
  274. package/src/tools/loam/commands/move.ts +323 -0
  275. package/src/tools/loam/commands/onboard.ts +374 -0
  276. package/src/tools/loam/commands/outcome.ts +185 -0
  277. package/src/tools/loam/commands/prime.ts +688 -0
  278. package/src/tools/loam/commands/prune.ts +614 -0
  279. package/src/tools/loam/commands/query.ts +218 -0
  280. package/src/tools/loam/commands/rank.ts +180 -0
  281. package/src/tools/loam/commands/ready.ts +189 -0
  282. package/src/tools/loam/commands/record.ts +1210 -0
  283. package/src/tools/loam/commands/restore.ts +166 -0
  284. package/src/tools/loam/commands/search.ts +327 -0
  285. package/src/tools/loam/commands/setup.ts +887 -0
  286. package/src/tools/loam/commands/status.ts +103 -0
  287. package/src/tools/loam/commands/sync.ts +298 -0
  288. package/src/tools/loam/commands/update.ts +19 -0
  289. package/src/tools/loam/commands/upgrade.ts +93 -0
  290. package/src/tools/loam/commands/validate.ts +190 -0
  291. package/src/tools/loam/index.ts +62 -0
  292. package/src/tools/loam/log.ts +127 -0
  293. package/src/tools/loam/registry/builtins.ts +409 -0
  294. package/src/tools/loam/registry/custom.ts +431 -0
  295. package/src/tools/loam/registry/init.ts +55 -0
  296. package/src/tools/loam/registry/template.ts +40 -0
  297. package/src/tools/loam/registry/type-registry.ts +113 -0
  298. package/src/tools/loam/schemas/config-schema.ts +489 -0
  299. package/src/tools/loam/schemas/config.ts +245 -0
  300. package/src/tools/loam/schemas/index.ts +18 -0
  301. package/src/tools/loam/schemas/record-schema.ts +191 -0
  302. package/src/tools/loam/schemas/record.ts +115 -0
  303. package/src/tools/loam/utils/active-work.ts +205 -0
  304. package/src/tools/loam/utils/anchor-validity.ts +80 -0
  305. package/src/tools/loam/utils/archive.ts +146 -0
  306. package/src/tools/loam/utils/audit.ts +667 -0
  307. package/src/tools/loam/utils/bm25.ts +238 -0
  308. package/src/tools/loam/utils/budget.ts +142 -0
  309. package/src/tools/loam/utils/config.ts +344 -0
  310. package/src/tools/loam/utils/dir-anchors.ts +62 -0
  311. package/src/tools/loam/utils/domain-rules.ts +114 -0
  312. package/src/tools/loam/utils/expertise.ts +393 -0
  313. package/src/tools/loam/utils/format-helpers.ts +96 -0
  314. package/src/tools/loam/utils/format.ts +1234 -0
  315. package/src/tools/loam/utils/git-context.ts +50 -0
  316. package/src/tools/loam/utils/git.ts +183 -0
  317. package/src/tools/loam/utils/hooks.ts +299 -0
  318. package/src/tools/loam/utils/index.ts +52 -0
  319. package/src/tools/loam/utils/json-output.ts +13 -0
  320. package/src/tools/loam/utils/lock.ts +76 -0
  321. package/src/tools/loam/utils/markers.ts +48 -0
  322. package/src/tools/loam/utils/numeric-flags.ts +20 -0
  323. package/src/tools/loam/utils/palette.ts +44 -0
  324. package/src/tools/loam/utils/prime-ranking.ts +135 -0
  325. package/src/tools/loam/utils/recipe-discovery.ts +195 -0
  326. package/src/tools/loam/utils/runtime-flags.ts +28 -0
  327. package/src/tools/loam/utils/scoring.ts +94 -0
  328. package/src/tools/loam/utils/version.ts +116 -0
  329. package/src/tools/sprout/commands/block.ts +64 -0
  330. package/src/tools/sprout/commands/blocked.ts +86 -0
  331. package/src/tools/sprout/commands/close.ts +129 -0
  332. package/src/tools/sprout/commands/completions.ts +198 -0
  333. package/src/tools/sprout/commands/config.ts +238 -0
  334. package/src/tools/sprout/commands/create.ts +164 -0
  335. package/src/tools/sprout/commands/dep.ts +148 -0
  336. package/src/tools/sprout/commands/doctor.ts +979 -0
  337. package/src/tools/sprout/commands/init.ts +83 -0
  338. package/src/tools/sprout/commands/label.ts +178 -0
  339. package/src/tools/sprout/commands/list.ts +210 -0
  340. package/src/tools/sprout/commands/migrate.ts +133 -0
  341. package/src/tools/sprout/commands/onboard.ts +207 -0
  342. package/src/tools/sprout/commands/plan-show.ts +278 -0
  343. package/src/tools/sprout/commands/plan.ts +2526 -0
  344. package/src/tools/sprout/commands/prime.ts +399 -0
  345. package/src/tools/sprout/commands/ready.ts +245 -0
  346. package/src/tools/sprout/commands/search.ts +221 -0
  347. package/src/tools/sprout/commands/show.ts +277 -0
  348. package/src/tools/sprout/commands/stats.ts +146 -0
  349. package/src/tools/sprout/commands/sync.ts +134 -0
  350. package/src/tools/sprout/commands/tpl.ts +364 -0
  351. package/src/tools/sprout/commands/unblock.ts +115 -0
  352. package/src/tools/sprout/commands/update.ts +257 -0
  353. package/src/tools/sprout/commands/upgrade.ts +91 -0
  354. package/src/tools/sprout/config-schema.ts +152 -0
  355. package/src/tools/sprout/config.ts +355 -0
  356. package/src/tools/sprout/filter.ts +107 -0
  357. package/src/tools/sprout/format.ts +43 -0
  358. package/src/tools/sprout/id.ts +22 -0
  359. package/src/tools/sprout/index.ts +204 -0
  360. package/src/tools/sprout/log.ts +76 -0
  361. package/src/tools/sprout/markers.ts +22 -0
  362. package/src/tools/sprout/output.ts +121 -0
  363. package/src/tools/sprout/plan-backref.ts +93 -0
  364. package/src/tools/sprout/plan-context.ts +81 -0
  365. package/src/tools/sprout/plan-domain.ts +139 -0
  366. package/src/tools/sprout/plan-lifecycle.ts +65 -0
  367. package/src/tools/sprout/plan-loam.ts +207 -0
  368. package/src/tools/sprout/plan-schema.ts +209 -0
  369. package/src/tools/sprout/sort.ts +31 -0
  370. package/src/tools/sprout/store.ts +172 -0
  371. package/src/tools/sprout/types.ts +118 -0
  372. package/src/tools/sprout/validation.ts +119 -0
  373. package/src/tools/sprout/version.ts +1 -0
  374. package/src/tools/sprout/yaml.ts +387 -0
  375. package/src/tools/trellis/commands/archive.ts +87 -0
  376. package/src/tools/trellis/commands/completions.ts +610 -0
  377. package/src/tools/trellis/commands/config.ts +382 -0
  378. package/src/tools/trellis/commands/create.ts +252 -0
  379. package/src/tools/trellis/commands/diff.ts +150 -0
  380. package/src/tools/trellis/commands/doctor.ts +771 -0
  381. package/src/tools/trellis/commands/emit.ts +365 -0
  382. package/src/tools/trellis/commands/history.ts +83 -0
  383. package/src/tools/trellis/commands/import.ts +198 -0
  384. package/src/tools/trellis/commands/init.ts +81 -0
  385. package/src/tools/trellis/commands/list.ts +103 -0
  386. package/src/tools/trellis/commands/onboard.ts +156 -0
  387. package/src/tools/trellis/commands/pin.ts +172 -0
  388. package/src/tools/trellis/commands/prime.ts +193 -0
  389. package/src/tools/trellis/commands/render.ts +122 -0
  390. package/src/tools/trellis/commands/schema.ts +353 -0
  391. package/src/tools/trellis/commands/show.ts +115 -0
  392. package/src/tools/trellis/commands/stats.ts +65 -0
  393. package/src/tools/trellis/commands/sync.ts +112 -0
  394. package/src/tools/trellis/commands/tree.ts +123 -0
  395. package/src/tools/trellis/commands/update.ts +330 -0
  396. package/src/tools/trellis/commands/upgrade.ts +95 -0
  397. package/src/tools/trellis/commands/validate.ts +166 -0
  398. package/src/tools/trellis/config-schema.ts +81 -0
  399. package/src/tools/trellis/config.ts +108 -0
  400. package/src/tools/trellis/frontmatter.ts +348 -0
  401. package/src/tools/trellis/id.ts +24 -0
  402. package/src/tools/trellis/index.ts +209 -0
  403. package/src/tools/trellis/markers.ts +28 -0
  404. package/src/tools/trellis/output.ts +84 -0
  405. package/src/tools/trellis/render.ts +212 -0
  406. package/src/tools/trellis/store.ts +144 -0
  407. package/src/tools/trellis/types.ts +82 -0
  408. package/src/tools/trellis/validate.ts +199 -0
  409. package/src/tools/trellis/yaml.ts +309 -0
  410. package/src/tracker/beads.test.ts +454 -0
  411. package/src/tracker/beads.ts +56 -0
  412. package/src/tracker/factory.test.ts +90 -0
  413. package/src/tracker/factory.ts +65 -0
  414. package/src/tracker/sprout.test.ts +461 -0
  415. package/src/tracker/sprout.ts +182 -0
  416. package/src/tracker/types.ts +52 -0
  417. package/src/trellis/client.test.ts +107 -0
  418. package/src/trellis/client.ts +179 -0
  419. package/src/types.ts +970 -0
  420. package/src/utils/bin.test.ts +10 -0
  421. package/src/utils/bin.ts +37 -0
  422. package/src/utils/browser.test.ts +49 -0
  423. package/src/utils/browser.ts +48 -0
  424. package/src/utils/fs.test.ts +119 -0
  425. package/src/utils/fs.ts +62 -0
  426. package/src/utils/pid.test.ts +152 -0
  427. package/src/utils/pid.ts +130 -0
  428. package/src/utils/process-scan.test.ts +53 -0
  429. package/src/utils/process-scan.ts +76 -0
  430. package/src/utils/time.test.ts +43 -0
  431. package/src/utils/time.ts +37 -0
  432. package/src/utils/version.test.ts +33 -0
  433. package/src/utils/version.ts +70 -0
  434. package/src/version.ts +5 -0
  435. package/src/watchdog/daemon.test.ts +3721 -0
  436. package/src/watchdog/daemon.ts +1257 -0
  437. package/src/watchdog/health.test.ts +830 -0
  438. package/src/watchdog/health.ts +434 -0
  439. package/src/watchdog/triage.test.ts +205 -0
  440. package/src/watchdog/triage.ts +205 -0
  441. package/src/worktree/manager.test.ts +720 -0
  442. package/src/worktree/manager.ts +405 -0
  443. package/src/worktree/process.test.ts +172 -0
  444. package/src/worktree/process.ts +131 -0
  445. package/src/worktree/tmux.test.ts +1616 -0
  446. package/src/worktree/tmux.ts +721 -0
  447. package/templates/CLAUDE.md.tmpl +100 -0
  448. package/templates/copilot-hooks.json.tmpl +13 -0
  449. package/templates/hooks.json.tmpl +109 -0
  450. package/templates/overlay.md.tmpl +88 -0
  451. package/ui/dist/apple-touch-icon-bdy6teep.png +0 -0
  452. package/ui/dist/chunk-8s31f05k.css +1 -0
  453. package/ui/dist/chunk-vm5rz679.js +300 -0
  454. package/ui/dist/favicon-nzb39vza.svg +4 -0
  455. package/ui/dist/index.html +17 -0
@@ -0,0 +1,807 @@
1
+ import { afterEach, beforeEach, describe, expect, mock, test } from "bun:test";
2
+ import { existsSync, mkdirSync, mkdtempSync, rmSync, writeFileSync } from "node:fs";
3
+ import { tmpdir } from "node:os";
4
+ import { join } from "node:path";
5
+ import { createSessionStore } from "../sessions/store.ts";
6
+ import type { AgentplateConfig } from "../types.ts";
7
+ import type { ConsistencyCheckDeps } from "./consistency.ts";
8
+ import { checkConsistency } from "./consistency.ts";
9
+
10
+ /**
11
+ * Mock tmux functions using dependency injection instead of mock.module().
12
+ * This avoids test isolation issues from module-level mocking.
13
+ */
14
+ const mockListSessions = mock(() => Promise.resolve([] as Array<{ name: string; pid: number }>));
15
+ const mockIsProcessAlive = mock((_pid: number) => true);
16
+
17
+ /**
18
+ * Create a minimal temp git repo for worktree tests.
19
+ */
20
+ function createTempGitRepo(): string {
21
+ const dir = mkdtempSync(join(tmpdir(), "agentplate-test-"));
22
+ const git = (args: string[]) => {
23
+ const proc = Bun.spawnSync(["git", ...args], { cwd: dir, stdout: "ignore", stderr: "pipe" });
24
+ if (proc.exitCode !== 0) {
25
+ throw new Error(`git ${args.join(" ")} failed: ${proc.stderr.toString()}`);
26
+ }
27
+ };
28
+
29
+ git(["init"]);
30
+ git(["config", "user.email", "test@test.com"]);
31
+ git(["config", "user.name", "Test User"]);
32
+ git(["config", "commit.gpgsign", "false"]);
33
+ writeFileSync(join(dir, "README.md"), "# Test Repo\n");
34
+ git(["add", "."]);
35
+ git(["commit", "-m", "Initial commit"]);
36
+
37
+ return dir;
38
+ }
39
+
40
+ /**
41
+ * Create a git worktree at the given path.
42
+ */
43
+ function createWorktree(repoRoot: string, worktreePath: string, branchName: string): void {
44
+ const proc = Bun.spawnSync(["git", "worktree", "add", "-b", branchName, worktreePath, "HEAD"], {
45
+ cwd: repoRoot,
46
+ stdout: "ignore",
47
+ stderr: "pipe",
48
+ });
49
+ if (proc.exitCode !== 0) {
50
+ throw new Error(`Failed to create worktree: ${proc.stderr.toString()}`);
51
+ }
52
+ }
53
+
54
+ describe("checkConsistency", () => {
55
+ let repoRoot: string;
56
+ let agentplateDir: string;
57
+ let config: AgentplateConfig;
58
+ let mockDeps: ConsistencyCheckDeps;
59
+
60
+ beforeEach(() => {
61
+ repoRoot = createTempGitRepo();
62
+ agentplateDir = join(repoRoot, ".agentplate");
63
+ mkdirSync(agentplateDir, { recursive: true });
64
+ mkdirSync(join(agentplateDir, "worktrees"), { recursive: true });
65
+
66
+ config = {
67
+ project: {
68
+ name: "testproject",
69
+ root: repoRoot,
70
+ canonicalBranch: "main",
71
+ },
72
+ agents: {
73
+ manifestPath: join(agentplateDir, "agent-manifest.json"),
74
+ baseDir: join(repoRoot, "agents"),
75
+ maxConcurrent: 5,
76
+ staggerDelayMs: 100,
77
+ maxDepth: 2,
78
+ maxSessionsPerRun: 0,
79
+ maxAgentsPerLead: 5,
80
+ },
81
+ worktrees: {
82
+ baseDir: join(agentplateDir, "worktrees"),
83
+ },
84
+ taskTracker: {
85
+ backend: "auto",
86
+ enabled: false,
87
+ },
88
+ loam: {
89
+ enabled: false,
90
+ domains: [],
91
+ primeFormat: "markdown",
92
+ },
93
+ merge: {
94
+ aiResolveEnabled: false,
95
+ reimagineEnabled: false,
96
+ },
97
+ providers: {
98
+ anthropic: { type: "native" },
99
+ },
100
+ watchdog: {
101
+ tier0Enabled: false,
102
+ tier0IntervalMs: 30000,
103
+ tier1Enabled: false,
104
+ tier2Enabled: false,
105
+ staleThresholdMs: 60000,
106
+ zombieThresholdMs: 300000,
107
+ nudgeIntervalMs: 30000,
108
+ },
109
+ models: {},
110
+ logging: {
111
+ verbose: false,
112
+ redactSecrets: true,
113
+ },
114
+ };
115
+
116
+ // Reset mocks and create deps object
117
+ mockListSessions.mockReset();
118
+ mockIsProcessAlive.mockReset();
119
+ mockListSessions.mockResolvedValue([]);
120
+ mockIsProcessAlive.mockReturnValue(true);
121
+
122
+ mockDeps = {
123
+ listSessions: mockListSessions,
124
+ isProcessAlive: mockIsProcessAlive,
125
+ };
126
+ });
127
+
128
+ afterEach(() => {
129
+ if (existsSync(repoRoot)) {
130
+ rmSync(repoRoot, { recursive: true, force: true });
131
+ }
132
+ });
133
+
134
+ test("returns all pass when no sessions exist", async () => {
135
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
136
+
137
+ expect(checks.length).toBeGreaterThan(0);
138
+ const passChecks = checks.filter((c) => c.status === "pass");
139
+ expect(passChecks.length).toBeGreaterThan(0);
140
+
141
+ const failChecks = checks.filter((c) => c.status === "fail");
142
+ expect(failChecks.length).toBe(0);
143
+ });
144
+
145
+ test("detects orphaned worktrees", async () => {
146
+ // Create a worktree but don't add it to SessionStore
147
+ const worktreePath = join(agentplateDir, "worktrees", "orphan-agent");
148
+ createWorktree(repoRoot, worktreePath, "agentplate/orphan-agent/test-123");
149
+
150
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
151
+
152
+ const orphanCheck = checks.find((c) => c.name === "orphaned-worktrees");
153
+ expect(orphanCheck).toBeDefined();
154
+ expect(orphanCheck?.status).toBe("warn");
155
+ expect(orphanCheck?.message).toContain("1 orphaned worktree");
156
+ expect(orphanCheck?.details?.length).toBe(1);
157
+ expect(orphanCheck?.fixable).toBe(true);
158
+ });
159
+
160
+ test("detects orphaned tmux sessions", async () => {
161
+ // Mock a tmux session that isn't in SessionStore
162
+ mockListSessions.mockResolvedValue([{ name: "agentplate-testproject-orphan", pid: 9999 }]);
163
+
164
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
165
+
166
+ const orphanCheck = checks.find((c) => c.name === "orphaned-tmux");
167
+ expect(orphanCheck).toBeDefined();
168
+ expect(orphanCheck?.status).toBe("warn");
169
+ expect(orphanCheck?.message).toContain("1 orphaned tmux session");
170
+ expect(orphanCheck?.fixable).toBe(true);
171
+ });
172
+
173
+ test("ignores tmux sessions from other projects", async () => {
174
+ // Mock tmux sessions from different projects
175
+ mockListSessions.mockResolvedValue([
176
+ { name: "agentplate-otherproject-agent1", pid: 9999 },
177
+ { name: "my-custom-session", pid: 8888 },
178
+ ]);
179
+
180
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
181
+
182
+ const orphanCheck = checks.find((c) => c.name === "orphaned-tmux");
183
+ expect(orphanCheck).toBeDefined();
184
+ expect(orphanCheck?.status).toBe("pass");
185
+ expect(orphanCheck?.message).toContain("No orphaned tmux sessions");
186
+ });
187
+
188
+ test("detects dead PIDs in SessionStore", async () => {
189
+ // Create a session with a PID that's marked as dead
190
+ const dbPath = join(agentplateDir, "sessions.db");
191
+ const store = createSessionStore(dbPath);
192
+
193
+ store.upsert({
194
+ id: "session-1",
195
+ agentName: "dead-agent",
196
+ capability: "builder",
197
+ worktreePath: join(agentplateDir, "worktrees", "dead-agent"),
198
+ branchName: "agentplate/dead-agent/test-123",
199
+ taskId: "test-123",
200
+ tmuxSession: "agentplate-testproject-dead-agent",
201
+ state: "working",
202
+ pid: 99999, // Non-existent PID
203
+ parentAgent: null,
204
+ depth: 0,
205
+ runId: null,
206
+ startedAt: new Date().toISOString(),
207
+ lastActivity: new Date().toISOString(),
208
+ escalationLevel: 0,
209
+ stalledSince: null,
210
+ transcriptPath: null,
211
+ });
212
+ store.close();
213
+
214
+ // Mock that this PID is not alive
215
+ mockIsProcessAlive.mockReturnValue(false);
216
+
217
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
218
+
219
+ const deadPidCheck = checks.find((c) => c.name === "dead-pids");
220
+ expect(deadPidCheck).toBeDefined();
221
+ expect(deadPidCheck?.status).toBe("warn");
222
+ expect(deadPidCheck?.message).toContain("1 session(s) with dead PIDs");
223
+ expect(deadPidCheck?.fixable).toBe(true);
224
+ });
225
+
226
+ test("passes when all PIDs are alive", async () => {
227
+ const dbPath = join(agentplateDir, "sessions.db");
228
+ const store = createSessionStore(dbPath);
229
+
230
+ store.upsert({
231
+ id: "session-1",
232
+ agentName: "live-agent",
233
+ capability: "builder",
234
+ worktreePath: join(agentplateDir, "worktrees", "live-agent"),
235
+ branchName: "agentplate/live-agent/test-123",
236
+ taskId: "test-123",
237
+ tmuxSession: "agentplate-testproject-live-agent",
238
+ state: "working",
239
+ pid: 12345,
240
+ parentAgent: null,
241
+ depth: 0,
242
+ runId: null,
243
+ startedAt: new Date().toISOString(),
244
+ lastActivity: new Date().toISOString(),
245
+ escalationLevel: 0,
246
+ stalledSince: null,
247
+ transcriptPath: null,
248
+ });
249
+ store.close();
250
+
251
+ // Mock that this PID is alive
252
+ mockIsProcessAlive.mockReturnValue(true);
253
+
254
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
255
+
256
+ const deadPidCheck = checks.find((c) => c.name === "dead-pids");
257
+ expect(deadPidCheck).toBeDefined();
258
+ expect(deadPidCheck?.status).toBe("pass");
259
+ });
260
+
261
+ test("detects missing worktrees for SessionStore entries", async () => {
262
+ const dbPath = join(agentplateDir, "sessions.db");
263
+ const store = createSessionStore(dbPath);
264
+
265
+ const missingWorktreePath = join(agentplateDir, "worktrees", "missing-agent");
266
+ store.upsert({
267
+ id: "session-1",
268
+ agentName: "missing-agent",
269
+ capability: "builder",
270
+ worktreePath: missingWorktreePath,
271
+ branchName: "agentplate/missing-agent/test-123",
272
+ taskId: "test-123",
273
+ tmuxSession: "agentplate-testproject-missing-agent",
274
+ state: "working",
275
+ pid: null,
276
+ parentAgent: null,
277
+ depth: 0,
278
+ runId: null,
279
+ startedAt: new Date().toISOString(),
280
+ lastActivity: new Date().toISOString(),
281
+ escalationLevel: 0,
282
+ stalledSince: null,
283
+ transcriptPath: null,
284
+ });
285
+ store.close();
286
+
287
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
288
+
289
+ const missingCheck = checks.find((c) => c.name === "missing-worktrees");
290
+ expect(missingCheck).toBeDefined();
291
+ expect(missingCheck?.status).toBe("warn");
292
+ expect(missingCheck?.message).toContain("1 session(s) with missing worktrees");
293
+ expect(missingCheck?.fixable).toBe(true);
294
+ });
295
+
296
+ test("detects missing tmux sessions for SessionStore entries", async () => {
297
+ const dbPath = join(agentplateDir, "sessions.db");
298
+ const store = createSessionStore(dbPath);
299
+
300
+ const worktreePath = join(agentplateDir, "worktrees", "agent-without-tmux");
301
+ createWorktree(repoRoot, worktreePath, "agentplate/agent-without-tmux/test-123");
302
+
303
+ store.upsert({
304
+ id: "session-1",
305
+ agentName: "agent-without-tmux",
306
+ capability: "builder",
307
+ worktreePath,
308
+ branchName: "agentplate/agent-without-tmux/test-123",
309
+ taskId: "test-123",
310
+ tmuxSession: "agentplate-testproject-agent-without-tmux",
311
+ state: "working",
312
+ pid: null,
313
+ parentAgent: null,
314
+ depth: 0,
315
+ runId: null,
316
+ startedAt: new Date().toISOString(),
317
+ lastActivity: new Date().toISOString(),
318
+ escalationLevel: 0,
319
+ stalledSince: null,
320
+ transcriptPath: null,
321
+ });
322
+ store.close();
323
+
324
+ // Mock empty tmux sessions list
325
+ mockListSessions.mockResolvedValue([]);
326
+
327
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
328
+
329
+ const missingCheck = checks.find((c) => c.name === "missing-tmux");
330
+ expect(missingCheck).toBeDefined();
331
+ expect(missingCheck?.status).toBe("warn");
332
+ expect(missingCheck?.message).toContain("1 session(s) with missing tmux sessions");
333
+ expect(missingCheck?.fixable).toBe(true);
334
+ });
335
+
336
+ test("passes when everything is consistent", async () => {
337
+ const dbPath = join(agentplateDir, "sessions.db");
338
+ const store = createSessionStore(dbPath);
339
+
340
+ const worktreePath = join(agentplateDir, "worktrees", "consistent-agent");
341
+ createWorktree(repoRoot, worktreePath, "agentplate/consistent-agent/test-123");
342
+
343
+ store.upsert({
344
+ id: "session-1",
345
+ agentName: "consistent-agent",
346
+ capability: "builder",
347
+ worktreePath,
348
+ branchName: "agentplate/consistent-agent/test-123",
349
+ taskId: "test-123",
350
+ tmuxSession: "agentplate-testproject-consistent-agent",
351
+ state: "working",
352
+ pid: 12345,
353
+ parentAgent: null,
354
+ depth: 0,
355
+ runId: null,
356
+ startedAt: new Date().toISOString(),
357
+ lastActivity: new Date().toISOString(),
358
+ escalationLevel: 0,
359
+ stalledSince: null,
360
+ transcriptPath: null,
361
+ });
362
+ store.close();
363
+
364
+ // Mock matching tmux session
365
+ mockListSessions.mockResolvedValue([
366
+ { name: "agentplate-testproject-consistent-agent", pid: 12345 },
367
+ ]);
368
+
369
+ // Mock PID as alive
370
+ mockIsProcessAlive.mockReturnValue(true);
371
+
372
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
373
+
374
+ const warnOrFail = checks.filter((c) => c.status === "warn" || c.status === "fail");
375
+ expect(warnOrFail.length).toBe(0);
376
+ });
377
+
378
+ test("ignores completed sessions whose tmux, pid, and worktree are gone", async () => {
379
+ const dbPath = join(agentplateDir, "sessions.db");
380
+ const store = createSessionStore(dbPath);
381
+
382
+ store.upsert({
383
+ id: "session-1",
384
+ agentName: "completed-agent",
385
+ capability: "builder",
386
+ worktreePath: join(agentplateDir, "worktrees", "completed-agent"),
387
+ branchName: "agentplate/completed-agent/test-123",
388
+ taskId: "test-123",
389
+ tmuxSession: "agentplate-testproject-completed-agent",
390
+ state: "completed",
391
+ pid: 99999,
392
+ parentAgent: null,
393
+ depth: 0,
394
+ runId: null,
395
+ startedAt: new Date().toISOString(),
396
+ lastActivity: new Date().toISOString(),
397
+ escalationLevel: 0,
398
+ stalledSince: null,
399
+ transcriptPath: null,
400
+ });
401
+ store.close();
402
+
403
+ mockIsProcessAlive.mockReturnValue(false);
404
+ mockListSessions.mockResolvedValue([]);
405
+
406
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
407
+
408
+ expect(checks.find((c) => c.name === "dead-pids")?.status).toBe("pass");
409
+ expect(checks.find((c) => c.name === "missing-worktrees")?.status).toBe("pass");
410
+ expect(checks.find((c) => c.name === "missing-tmux")?.status).toBe("pass");
411
+ });
412
+
413
+ test("orphan-spawns: terminal state with live pid is flagged", async () => {
414
+ const dbPath = join(agentplateDir, "sessions.db");
415
+ const store = createSessionStore(dbPath);
416
+
417
+ store.upsert({
418
+ id: "session-1",
419
+ agentName: "orphaned-agent",
420
+ capability: "builder",
421
+ worktreePath: join(agentplateDir, "worktrees", "orphaned-agent"),
422
+ branchName: "agentplate/orphaned-agent/test-123",
423
+ taskId: "test-123",
424
+ tmuxSession: "",
425
+ state: "completed",
426
+ pid: 4242,
427
+ parentAgent: null,
428
+ depth: 0,
429
+ runId: null,
430
+ startedAt: new Date().toISOString(),
431
+ lastActivity: new Date().toISOString(),
432
+ escalationLevel: 0,
433
+ stalledSince: null,
434
+ transcriptPath: null,
435
+ });
436
+ store.close();
437
+
438
+ mockIsProcessAlive.mockReturnValue(true);
439
+ mockListSessions.mockResolvedValue([]);
440
+
441
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
442
+
443
+ const orphanCheck = checks.find((c) => c.name === "orphan-spawns");
444
+ expect(orphanCheck).toBeDefined();
445
+ expect(orphanCheck?.status).toBe("warn");
446
+ expect(orphanCheck?.message).toContain("1 orphaned spawn");
447
+ expect(orphanCheck?.details?.[0]).toContain("orphaned-agent");
448
+ expect(orphanCheck?.fixable).toBe(true);
449
+ });
450
+
451
+ test("orphan-spawns: tmux dead but pid alive is flagged", async () => {
452
+ const dbPath = join(agentplateDir, "sessions.db");
453
+ const store = createSessionStore(dbPath);
454
+
455
+ store.upsert({
456
+ id: "session-1",
457
+ agentName: "tmux-dead-agent",
458
+ capability: "builder",
459
+ worktreePath: join(agentplateDir, "worktrees", "tmux-dead-agent"),
460
+ branchName: "agentplate/tmux-dead-agent/test-123",
461
+ taskId: "test-123",
462
+ tmuxSession: "agentplate-testproject-tmux-dead-agent",
463
+ state: "working",
464
+ pid: 4242,
465
+ parentAgent: null,
466
+ depth: 0,
467
+ runId: null,
468
+ startedAt: new Date().toISOString(),
469
+ lastActivity: new Date().toISOString(),
470
+ escalationLevel: 0,
471
+ stalledSince: null,
472
+ transcriptPath: null,
473
+ });
474
+ store.close();
475
+
476
+ mockIsProcessAlive.mockReturnValue(true);
477
+ // tmux server reports no matching session
478
+ mockListSessions.mockResolvedValue([]);
479
+
480
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
481
+
482
+ const orphanCheck = checks.find((c) => c.name === "orphan-spawns");
483
+ expect(orphanCheck?.status).toBe("warn");
484
+ expect(orphanCheck?.details?.[0]).toContain("tmux session");
485
+ });
486
+
487
+ test("orphan-spawns: passes when terminal-state pid is dead", async () => {
488
+ const dbPath = join(agentplateDir, "sessions.db");
489
+ const store = createSessionStore(dbPath);
490
+
491
+ store.upsert({
492
+ id: "session-1",
493
+ agentName: "clean-completed",
494
+ capability: "builder",
495
+ worktreePath: join(agentplateDir, "worktrees", "clean-completed"),
496
+ branchName: "agentplate/clean-completed/test-123",
497
+ taskId: "test-123",
498
+ tmuxSession: "",
499
+ state: "completed",
500
+ pid: 4242,
501
+ parentAgent: null,
502
+ depth: 0,
503
+ runId: null,
504
+ startedAt: new Date().toISOString(),
505
+ lastActivity: new Date().toISOString(),
506
+ escalationLevel: 0,
507
+ stalledSince: null,
508
+ transcriptPath: null,
509
+ });
510
+ store.close();
511
+
512
+ mockIsProcessAlive.mockReturnValue(false);
513
+
514
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
515
+
516
+ expect(checks.find((c) => c.name === "orphan-spawns")?.status).toBe("pass");
517
+ });
518
+
519
+ test("handles tmux not installed gracefully", async () => {
520
+ // Mock tmux listing to throw an error
521
+ mockListSessions.mockRejectedValue(new Error("tmux: command not found"));
522
+
523
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
524
+
525
+ const tmuxCheck = checks.find((c) => c.name === "tmux-listing");
526
+ expect(tmuxCheck).toBeDefined();
527
+ expect(tmuxCheck?.status).toBe("warn");
528
+ expect(tmuxCheck?.message).toContain("Failed to list tmux sessions");
529
+ });
530
+
531
+ test("fails early if git worktree list fails", async () => {
532
+ // Use a non-existent repo root to trigger worktree listing failure
533
+ const badConfig = { ...config, project: { ...config.project, root: "/nonexistent" } };
534
+
535
+ const checks = await checkConsistency(badConfig, agentplateDir, mockDeps);
536
+
537
+ expect(checks.length).toBe(1);
538
+ expect(checks[0]?.name).toBe("worktree-listing");
539
+ expect(checks[0]?.status).toBe("fail");
540
+ });
541
+
542
+ test("fails early if SessionStore cannot be opened", async () => {
543
+ // Use a bad agentplate directory path
544
+ const badAgentplateDir = "/nonexistent/.agentplate";
545
+
546
+ const checks = await checkConsistency(config, badAgentplateDir, mockDeps);
547
+
548
+ const storeCheck = checks.find((c) => c.name === "sessionstore-open");
549
+ expect(storeCheck).toBeDefined();
550
+ expect(storeCheck?.status).toBe("fail");
551
+ });
552
+
553
+ test("reviewer-coverage: leads without reviewers emits warn", async () => {
554
+ const dbPath = join(agentplateDir, "sessions.db");
555
+ const store = createSessionStore(dbPath);
556
+
557
+ // Add 2 builder sessions under lead-1, no reviewers
558
+ store.upsert({
559
+ id: "session-1",
560
+ agentName: "builder-1",
561
+ capability: "builder",
562
+ worktreePath: join(agentplateDir, "worktrees", "builder-1"),
563
+ branchName: "agentplate/builder-1/test-123",
564
+ taskId: "test-123",
565
+ tmuxSession: "agentplate-testproject-builder-1",
566
+ state: "working",
567
+ pid: null,
568
+ parentAgent: "lead-1",
569
+ depth: 1,
570
+ runId: null,
571
+ startedAt: new Date().toISOString(),
572
+ lastActivity: new Date().toISOString(),
573
+ escalationLevel: 0,
574
+ stalledSince: null,
575
+ transcriptPath: null,
576
+ });
577
+
578
+ store.upsert({
579
+ id: "session-2",
580
+ agentName: "builder-2",
581
+ capability: "builder",
582
+ worktreePath: join(agentplateDir, "worktrees", "builder-2"),
583
+ branchName: "agentplate/builder-2/test-456",
584
+ taskId: "test-456",
585
+ tmuxSession: "agentplate-testproject-builder-2",
586
+ state: "working",
587
+ pid: null,
588
+ parentAgent: "lead-1",
589
+ depth: 1,
590
+ runId: null,
591
+ startedAt: new Date().toISOString(),
592
+ lastActivity: new Date().toISOString(),
593
+ escalationLevel: 0,
594
+ stalledSince: null,
595
+ transcriptPath: null,
596
+ });
597
+ store.close();
598
+
599
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
600
+
601
+ const reviewerCheck = checks.find((c) => c.name === "reviewer-coverage");
602
+ expect(reviewerCheck).toBeDefined();
603
+ expect(reviewerCheck?.status).toBe("warn");
604
+ expect(reviewerCheck?.message).toContain("without any reviewers");
605
+ expect(reviewerCheck?.details).toBeDefined();
606
+ expect(reviewerCheck?.details?.length).toBeGreaterThan(0);
607
+ });
608
+
609
+ test("reviewer-coverage: partial reviewer coverage emits warn", async () => {
610
+ const dbPath = join(agentplateDir, "sessions.db");
611
+ const store = createSessionStore(dbPath);
612
+
613
+ // Add 3 builders and 1 reviewer under same parent
614
+ for (let i = 1; i <= 3; i++) {
615
+ store.upsert({
616
+ id: `session-builder-${i}`,
617
+ agentName: `builder-${i}`,
618
+ capability: "builder",
619
+ worktreePath: join(agentplateDir, "worktrees", `builder-${i}`),
620
+ branchName: `agentplate/builder-${i}/test-${i}`,
621
+ taskId: `test-${i}`,
622
+ tmuxSession: `agentplate-testproject-builder-${i}`,
623
+ state: "working",
624
+ pid: null,
625
+ parentAgent: "lead-1",
626
+ depth: 1,
627
+ runId: null,
628
+ startedAt: new Date().toISOString(),
629
+ lastActivity: new Date().toISOString(),
630
+ escalationLevel: 0,
631
+ stalledSince: null,
632
+ transcriptPath: null,
633
+ });
634
+ }
635
+
636
+ store.upsert({
637
+ id: "session-reviewer-1",
638
+ agentName: "reviewer-1",
639
+ capability: "reviewer",
640
+ worktreePath: join(agentplateDir, "worktrees", "reviewer-1"),
641
+ branchName: "agentplate/reviewer-1/test-r1",
642
+ taskId: "test-r1",
643
+ tmuxSession: "agentplate-testproject-reviewer-1",
644
+ state: "working",
645
+ pid: null,
646
+ parentAgent: "lead-1",
647
+ depth: 1,
648
+ runId: null,
649
+ startedAt: new Date().toISOString(),
650
+ lastActivity: new Date().toISOString(),
651
+ escalationLevel: 0,
652
+ stalledSince: null,
653
+ transcriptPath: null,
654
+ });
655
+ store.close();
656
+
657
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
658
+
659
+ const reviewerCheck = checks.find((c) => c.name === "reviewer-coverage");
660
+ expect(reviewerCheck).toBeDefined();
661
+ expect(reviewerCheck?.status).toBe("warn");
662
+ expect(reviewerCheck?.message).toContain("partial reviewer coverage");
663
+ });
664
+
665
+ test("reviewer-coverage: full reviewer coverage emits pass", async () => {
666
+ const dbPath = join(agentplateDir, "sessions.db");
667
+ const store = createSessionStore(dbPath);
668
+
669
+ // Add 2 builders and 2 reviewers under same parent
670
+ for (let i = 1; i <= 2; i++) {
671
+ store.upsert({
672
+ id: `session-builder-${i}`,
673
+ agentName: `builder-${i}`,
674
+ capability: "builder",
675
+ worktreePath: join(agentplateDir, "worktrees", `builder-${i}`),
676
+ branchName: `agentplate/builder-${i}/test-${i}`,
677
+ taskId: `test-${i}`,
678
+ tmuxSession: `agentplate-testproject-builder-${i}`,
679
+ state: "working",
680
+ pid: null,
681
+ parentAgent: "lead-1",
682
+ depth: 1,
683
+ runId: null,
684
+ startedAt: new Date().toISOString(),
685
+ lastActivity: new Date().toISOString(),
686
+ escalationLevel: 0,
687
+ stalledSince: null,
688
+ transcriptPath: null,
689
+ });
690
+
691
+ store.upsert({
692
+ id: `session-reviewer-${i}`,
693
+ agentName: `reviewer-${i}`,
694
+ capability: "reviewer",
695
+ worktreePath: join(agentplateDir, "worktrees", `reviewer-${i}`),
696
+ branchName: `agentplate/reviewer-${i}/test-r${i}`,
697
+ taskId: `test-r${i}`,
698
+ tmuxSession: `agentplate-testproject-reviewer-${i}`,
699
+ state: "working",
700
+ pid: null,
701
+ parentAgent: "lead-1",
702
+ depth: 1,
703
+ runId: null,
704
+ startedAt: new Date().toISOString(),
705
+ lastActivity: new Date().toISOString(),
706
+ escalationLevel: 0,
707
+ stalledSince: null,
708
+ transcriptPath: null,
709
+ });
710
+ }
711
+ store.close();
712
+
713
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
714
+
715
+ const reviewerCheck = checks.find((c) => c.name === "reviewer-coverage");
716
+ expect(reviewerCheck).toBeDefined();
717
+ expect(reviewerCheck?.status).toBe("pass");
718
+ });
719
+
720
+ test("reviewer-coverage: no builder sessions emits pass", async () => {
721
+ // Don't create any sessions at all
722
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
723
+
724
+ const reviewerCheck = checks.find((c) => c.name === "reviewer-coverage");
725
+ expect(reviewerCheck).toBeDefined();
726
+ expect(reviewerCheck?.status).toBe("pass");
727
+ expect(reviewerCheck?.message).toContain("No builder sessions found");
728
+ });
729
+
730
+ test("reviewer-coverage: multiple leads mixed coverage", async () => {
731
+ const dbPath = join(agentplateDir, "sessions.db");
732
+ const store = createSessionStore(dbPath);
733
+
734
+ // Lead-1 has builders + reviewers (good)
735
+ store.upsert({
736
+ id: "session-builder-1",
737
+ agentName: "builder-1",
738
+ capability: "builder",
739
+ worktreePath: join(agentplateDir, "worktrees", "builder-1"),
740
+ branchName: "agentplate/builder-1/test-1",
741
+ taskId: "test-1",
742
+ tmuxSession: "agentplate-testproject-builder-1",
743
+ state: "working",
744
+ pid: null,
745
+ parentAgent: "lead-1",
746
+ depth: 1,
747
+ runId: null,
748
+ startedAt: new Date().toISOString(),
749
+ lastActivity: new Date().toISOString(),
750
+ escalationLevel: 0,
751
+ stalledSince: null,
752
+ transcriptPath: null,
753
+ });
754
+
755
+ store.upsert({
756
+ id: "session-reviewer-1",
757
+ agentName: "reviewer-1",
758
+ capability: "reviewer",
759
+ worktreePath: join(agentplateDir, "worktrees", "reviewer-1"),
760
+ branchName: "agentplate/reviewer-1/test-r1",
761
+ taskId: "test-r1",
762
+ tmuxSession: "agentplate-testproject-reviewer-1",
763
+ state: "working",
764
+ pid: null,
765
+ parentAgent: "lead-1",
766
+ depth: 1,
767
+ runId: null,
768
+ startedAt: new Date().toISOString(),
769
+ lastActivity: new Date().toISOString(),
770
+ escalationLevel: 0,
771
+ stalledSince: null,
772
+ transcriptPath: null,
773
+ });
774
+
775
+ // Lead-2 has builders only (bad)
776
+ store.upsert({
777
+ id: "session-builder-2",
778
+ agentName: "builder-2",
779
+ capability: "builder",
780
+ worktreePath: join(agentplateDir, "worktrees", "builder-2"),
781
+ branchName: "agentplate/builder-2/test-2",
782
+ taskId: "test-2",
783
+ tmuxSession: "agentplate-testproject-builder-2",
784
+ state: "working",
785
+ pid: null,
786
+ parentAgent: "lead-2",
787
+ depth: 1,
788
+ runId: null,
789
+ startedAt: new Date().toISOString(),
790
+ lastActivity: new Date().toISOString(),
791
+ escalationLevel: 0,
792
+ stalledSince: null,
793
+ transcriptPath: null,
794
+ });
795
+ store.close();
796
+
797
+ const checks = await checkConsistency(config, agentplateDir, mockDeps);
798
+
799
+ const reviewerCheck = checks.find((c) => c.name === "reviewer-coverage");
800
+ expect(reviewerCheck).toBeDefined();
801
+ expect(reviewerCheck?.status).toBe("warn");
802
+ expect(reviewerCheck?.details).toBeDefined();
803
+ // Should contain lead-2 in the details
804
+ const detailsStr = reviewerCheck?.details?.join(" ");
805
+ expect(detailsStr).toContain("lead-2");
806
+ });
807
+ });