@pugi/cli 0.1.0-beta.8 → 0.1.0-beta.87

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 (402) hide show
  1. package/CHANGELOG.md +96 -0
  2. package/THIRD_PARTY_NOTICES.md +40 -0
  3. package/assets/pugi-prozr2-mascot.ansi +9 -0
  4. package/bin/run.js +33 -1
  5. package/dist/commands/deploy.js +40 -40
  6. package/dist/commands/flatten.js +191 -0
  7. package/dist/commands/jobs-watch.js +201 -0
  8. package/dist/commands/jobs.js +42 -27
  9. package/dist/commands/smoke.js +133 -0
  10. package/dist/core/agent-progress/cleanup.js +134 -0
  11. package/dist/core/agent-progress/schema.js +144 -0
  12. package/dist/core/agent-progress/writer.js +101 -0
  13. package/dist/core/agents/adaptive-router.js +330 -0
  14. package/dist/core/agents/query-decomposer.js +297 -0
  15. package/dist/core/agents/registry.js +2 -2
  16. package/dist/core/approvals/shortcut-resolver.js +98 -0
  17. package/dist/core/artifact-chain/dispatcher.js +148 -0
  18. package/dist/core/artifact-chain/exporter.js +164 -0
  19. package/dist/core/artifact-chain/state.js +243 -0
  20. package/dist/core/artifact-chain/steps.js +169 -0
  21. package/dist/core/ask-user/question.js +92 -0
  22. package/dist/core/audit/audit-trail.js +275 -0
  23. package/dist/core/auth/ensure-authenticated.js +129 -0
  24. package/dist/core/auth/env-provider.js +238 -0
  25. package/dist/core/auto-open-browser.js +4 -4
  26. package/dist/core/auto-update/channels.js +122 -0
  27. package/dist/core/auto-update/checker.js +241 -0
  28. package/dist/core/auto-update/state.js +235 -0
  29. package/dist/core/bare-mode/index.js +107 -0
  30. package/dist/core/bash/redirect.js +281 -0
  31. package/dist/core/bash-classifier.js +436 -40
  32. package/dist/core/checkpoint/resumer.js +149 -0
  33. package/dist/core/checkpoint/rewinder.js +291 -0
  34. package/dist/core/checkpoints/shadow-git.js +670 -0
  35. package/dist/core/citations/parser.js +109 -0
  36. package/dist/core/classifier/yolo-classifier.js +88 -0
  37. package/dist/core/codegraph/decision-store.js +248 -0
  38. package/dist/core/codegraph/detect-repo.js +459 -0
  39. package/dist/core/codegraph/install.js +134 -0
  40. package/dist/core/codegraph/offer-hook.js +220 -0
  41. package/dist/core/compact/auto-trigger.js +96 -0
  42. package/dist/core/compact/buffer-rewriter.js +115 -0
  43. package/dist/core/compact/summarizer.js +208 -0
  44. package/dist/core/compact/token-counter.js +108 -0
  45. package/dist/core/consensus/anvil-fanout.js +25 -25
  46. package/dist/core/consensus/diff-capture.js +121 -12
  47. package/dist/core/consensus/rubric.js +21 -21
  48. package/dist/core/context/builder.js +6 -6
  49. package/dist/core/context/compaction-events.js +8 -8
  50. package/dist/core/context/compaction.js +31 -31
  51. package/dist/core/context/index.js +15 -8
  52. package/dist/core/context/invariants.js +51 -51
  53. package/dist/core/context/markdown-loader.js +28 -10
  54. package/dist/core/context/markdown-traverse.js +255 -0
  55. package/dist/core/context/pugiignore.js +41 -41
  56. package/dist/core/context/repo-skeleton.js +37 -37
  57. package/dist/core/context/tool-eviction.js +55 -0
  58. package/dist/core/context/watcher.js +32 -32
  59. package/dist/core/context/working-set.js +23 -23
  60. package/dist/core/coordinator/agent-tools.js +77 -0
  61. package/dist/core/coordinator/agent-toolset.js +65 -0
  62. package/dist/core/coordinator/fsm.js +73 -0
  63. package/dist/core/coordinator/mode-fsm.js +70 -0
  64. package/dist/core/cost/rate-card.js +129 -0
  65. package/dist/core/cost/tracker.js +221 -0
  66. package/dist/core/credentials.js +12 -12
  67. package/dist/core/cron/scheduler.js +138 -0
  68. package/dist/core/denial-tracking/index.js +8 -0
  69. package/dist/core/denial-tracking/state.js +264 -0
  70. package/dist/core/diagnostics/probe-runner.js +93 -0
  71. package/dist/core/diagnostics/probes/api.js +46 -0
  72. package/dist/core/diagnostics/probes/auth.js +93 -0
  73. package/dist/core/diagnostics/probes/bare-mode.js +42 -0
  74. package/dist/core/diagnostics/probes/cli-version.js +127 -0
  75. package/dist/core/diagnostics/probes/config.js +72 -0
  76. package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
  77. package/dist/core/diagnostics/probes/disk.js +81 -0
  78. package/dist/core/diagnostics/probes/engine-live.js +46 -0
  79. package/dist/core/diagnostics/probes/git.js +65 -0
  80. package/dist/core/diagnostics/probes/hooks.js +118 -0
  81. package/dist/core/diagnostics/probes/mcp.js +75 -0
  82. package/dist/core/diagnostics/probes/node.js +59 -0
  83. package/dist/core/diagnostics/probes/pnpm.js +36 -0
  84. package/dist/core/diagnostics/probes/pugi-md.js +89 -0
  85. package/dist/core/diagnostics/probes/sandbox.js +40 -0
  86. package/dist/core/diagnostics/probes/session.js +74 -0
  87. package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
  88. package/dist/core/diagnostics/probes/workspace.js +63 -0
  89. package/dist/core/diagnostics/types.js +70 -0
  90. package/dist/core/dispatch/cache-cleanup.js +197 -0
  91. package/dist/core/dispatch/cache-handoff.js +295 -0
  92. package/dist/core/edits/apply-patch-layer-e.js +189 -0
  93. package/dist/core/edits/dispatch.js +293 -7
  94. package/dist/core/edits/format-matrix.js +26 -0
  95. package/dist/core/edits/fuzzy-ladder.js +650 -0
  96. package/dist/core/edits/index.js +3 -1
  97. package/dist/core/edits/journal.js +199 -0
  98. package/dist/core/edits/layer-a-apply.js +15 -15
  99. package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
  100. package/dist/core/edits/layer-b-apply.js +9 -9
  101. package/dist/core/edits/layer-c-apply.js +6 -6
  102. package/dist/core/edits/layer-d-ast.js +557 -14
  103. package/dist/core/edits/marker-parser.js +12 -12
  104. package/dist/core/edits/security-gate.js +27 -27
  105. package/dist/core/edits/verify-hook.js +273 -0
  106. package/dist/core/edits/worktree.js +322 -0
  107. package/dist/core/engine/anvil-client.js +140 -26
  108. package/dist/core/engine/auto-compact.js +179 -0
  109. package/dist/core/engine/budgets.js +186 -0
  110. package/dist/core/engine/context-prefix.js +155 -0
  111. package/dist/core/engine/index.js +1 -1
  112. package/dist/core/engine/intensity.js +158 -0
  113. package/dist/core/engine/intent.js +260 -0
  114. package/dist/core/engine/native-pugi.js +1295 -227
  115. package/dist/core/engine/prompts.js +134 -16
  116. package/dist/core/engine/strip-internal-fields.js +124 -0
  117. package/dist/core/engine/tool-bridge.js +1295 -59
  118. package/dist/core/evaluation/golden-dataset.js +293 -0
  119. package/dist/core/feedback/queue.js +177 -0
  120. package/dist/core/feedback/submitter.js +145 -0
  121. package/dist/core/file-cache.js +113 -1
  122. package/dist/core/flatten/flatten-repo.js +439 -0
  123. package/dist/core/format/osc8-link.js +28 -0
  124. package/dist/core/hook-chains.js +392 -0
  125. package/dist/core/hooks/citation-verify-hook.js +138 -0
  126. package/dist/core/hooks/citation-verify.js +112 -0
  127. package/dist/core/hooks/events.js +44 -0
  128. package/dist/core/hooks/index.js +15 -0
  129. package/dist/core/hooks/registry.js +213 -0
  130. package/dist/core/hooks/runner.js +236 -0
  131. package/dist/core/hooks/v2/event-emitter.js +115 -0
  132. package/dist/core/hooks/v2/executor.js +282 -0
  133. package/dist/core/hooks/v2/index.js +25 -0
  134. package/dist/core/hooks/v2/lifecycle.js +104 -0
  135. package/dist/core/hooks/v2/loader.js +216 -0
  136. package/dist/core/hooks/v2/matcher.js +125 -0
  137. package/dist/core/hooks/v2/trust.js +143 -0
  138. package/dist/core/hooks/v2/types.js +86 -0
  139. package/dist/core/image/renderer.js +71 -0
  140. package/dist/core/init/detector.js +582 -0
  141. package/dist/core/init/template-renderer.js +242 -0
  142. package/dist/core/jobs/registry.js +18 -18
  143. package/dist/core/ledger/results-tsv.js +142 -0
  144. package/dist/core/log-discipline/stdout-redirect.js +51 -0
  145. package/dist/core/lsp/cache.js +105 -0
  146. package/dist/core/lsp/client.js +776 -0
  147. package/dist/core/lsp/language-detect.js +66 -0
  148. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  149. package/dist/core/lsp/symbol-tools.js +372 -0
  150. package/dist/core/mcp/client.js +97 -28
  151. package/dist/core/mcp/http-server.js +553 -0
  152. package/dist/core/mcp/orchestrator-tools.js +662 -0
  153. package/dist/core/mcp/permission.js +190 -0
  154. package/dist/core/mcp/registry.js +39 -17
  155. package/dist/core/mcp/server-tools.js +219 -0
  156. package/dist/core/mcp/server.js +397 -0
  157. package/dist/core/mcp/trust.js +10 -10
  158. package/dist/core/memory/dual-write.js +416 -0
  159. package/dist/core/memory/passive-extract.js +130 -0
  160. package/dist/core/memory/phase1-kinds.js +20 -0
  161. package/dist/core/memory/secret-scanner.js +304 -0
  162. package/dist/core/memory-sync/queue.js +170 -0
  163. package/dist/core/metrics/extract.js +113 -0
  164. package/dist/core/modes/roo-modes.js +68 -0
  165. package/dist/core/onboarding/ensure-initialized.js +133 -0
  166. package/dist/core/onboarding/marker.js +111 -0
  167. package/dist/core/onboarding/telemetry-state.js +108 -0
  168. package/dist/core/output-style/presets.js +176 -0
  169. package/dist/core/output-style/state.js +185 -0
  170. package/dist/core/path-security.js +287 -5
  171. package/dist/core/permission.js +82 -22
  172. package/dist/core/permissions/auto-classifier.js +124 -0
  173. package/dist/core/permissions/bash-parser.js +371 -0
  174. package/dist/core/permissions/circuit-breaker.js +83 -0
  175. package/dist/core/permissions/constrained-edit.js +91 -0
  176. package/dist/core/permissions/gate.js +278 -0
  177. package/dist/core/permissions/index.js +20 -0
  178. package/dist/core/permissions/mode.js +174 -0
  179. package/dist/core/permissions/network-egress.js +137 -0
  180. package/dist/core/permissions/state.js +241 -0
  181. package/dist/core/permissions/tool-class.js +93 -0
  182. package/dist/core/plan-mode/ui-state.js +51 -0
  183. package/dist/core/plans/plan-artifact.js +721 -0
  184. package/dist/core/policy-limits/etag-store.js +122 -0
  185. package/dist/core/prd-check/parser.js +215 -0
  186. package/dist/core/prd-check/reporter.js +127 -0
  187. package/dist/core/prd-check/session-review.js +557 -0
  188. package/dist/core/prd-check/verifiers.js +223 -0
  189. package/dist/core/prompt-cache/client-cache.js +99 -0
  190. package/dist/core/prompts/assembly.js +29 -0
  191. package/dist/core/prompts/registry.js +364 -0
  192. package/dist/core/pugi-md/cc-compat-rules.js +735 -0
  193. package/dist/core/pugi-md/context-injector.js +76 -0
  194. package/dist/core/pugi-md/walk-up.js +207 -0
  195. package/dist/core/python/uv-installer.js +270 -0
  196. package/dist/core/python/uv-resolver.js +83 -0
  197. package/dist/core/rate-limit/narrator.js +146 -0
  198. package/dist/core/recipes/cli-types.js +20 -0
  199. package/dist/core/recipes/loader.js +103 -0
  200. package/dist/core/recipes/runner.js +345 -0
  201. package/dist/core/recipes/schema.js +587 -0
  202. package/dist/core/release-notes/parser.js +241 -0
  203. package/dist/core/release-notes/state.js +116 -0
  204. package/dist/core/repl/ask.js +37 -37
  205. package/dist/core/repl/cancellation.js +26 -26
  206. package/dist/core/repl/cap-warning.js +4 -4
  207. package/dist/core/repl/clipboard-read.js +11 -11
  208. package/dist/core/repl/dispatch-fsm.js +12 -12
  209. package/dist/core/repl/history-search.js +15 -15
  210. package/dist/core/repl/history.js +28 -18
  211. package/dist/core/repl/kill-ring.js +5 -5
  212. package/dist/core/repl/model-pricing.js +135 -0
  213. package/dist/core/repl/privacy-banner.js +22 -22
  214. package/dist/core/repl/session.js +2157 -214
  215. package/dist/core/repl/slash-commands.js +533 -40
  216. package/dist/core/repl/store/index.js +1 -1
  217. package/dist/core/repl/store/jsonl-log.js +22 -22
  218. package/dist/core/repl/store/lockfile.js +10 -10
  219. package/dist/core/repl/store/session-store.js +136 -107
  220. package/dist/core/repl/store/types.js +15 -15
  221. package/dist/core/repl/store/uuid-v7.js +12 -12
  222. package/dist/core/repl/workspace-context.js +43 -21
  223. package/dist/core/repo-map/build.js +125 -0
  224. package/dist/core/repo-map/cache.js +185 -0
  225. package/dist/core/repo-map/extractor.js +254 -0
  226. package/dist/core/repo-map/formatter.js +145 -0
  227. package/dist/core/repo-map/page-rank.js +105 -0
  228. package/dist/core/repo-map/scanner.js +211 -0
  229. package/dist/core/retry-budget/budget.js +284 -0
  230. package/dist/core/retry-budget/index.js +5 -0
  231. package/dist/core/retry-budget/retry-cap.js +74 -0
  232. package/dist/core/routing/lead-worker.js +43 -0
  233. package/dist/core/routing/pre-flight-estimator.js +108 -0
  234. package/dist/core/runs/run-tree.js +103 -0
  235. package/dist/core/security/injection-scanner.js +367 -0
  236. package/dist/core/security/output-filter.js +418 -0
  237. package/dist/core/session/env-file.js +105 -0
  238. package/dist/core/session/section-budgets.js +140 -0
  239. package/dist/core/session.js +92 -0
  240. package/dist/core/settings.js +286 -5
  241. package/dist/core/share/formatter.js +271 -0
  242. package/dist/core/share/redactor.js +221 -0
  243. package/dist/core/share/uploader.js +267 -0
  244. package/dist/core/skills/defaults.js +457 -0
  245. package/dist/core/skills/loader.js +22 -22
  246. package/dist/core/skills/sources.js +27 -27
  247. package/dist/core/smoke/headless-driver.js +174 -0
  248. package/dist/core/smoke/orchestrator.js +194 -0
  249. package/dist/core/smoke/runner.js +238 -0
  250. package/dist/core/smoke/scenario-parser.js +316 -0
  251. package/dist/core/statusline.js +99 -0
  252. package/dist/core/subagents/dispatcher-real.js +600 -0
  253. package/dist/core/subagents/dispatcher.js +132 -43
  254. package/dist/core/subagents/index.js +19 -6
  255. package/dist/core/subagents/isolation-matrix.js +213 -0
  256. package/dist/core/subagents/spawn.js +19 -4
  257. package/dist/core/telemetry/emitter.js +229 -0
  258. package/dist/core/telemetry/queue.js +251 -0
  259. package/dist/core/theme/context.js +91 -0
  260. package/dist/core/theme/presets.js +228 -0
  261. package/dist/core/theme/state.js +181 -0
  262. package/dist/core/todos/invariant.js +10 -0
  263. package/dist/core/todos/state.js +177 -0
  264. package/dist/core/tool-schema/compressor.js +89 -0
  265. package/dist/core/transport/version-interceptor.js +166 -0
  266. package/dist/core/trust.js +2 -2
  267. package/dist/core/tui/thinking-block.js +64 -0
  268. package/dist/core/vim/keymap.js +288 -0
  269. package/dist/core/vim/state.js +92 -0
  270. package/dist/core/watch-markers/marker-watcher.js +133 -0
  271. package/dist/core/worktree-manager/cleanup.js +123 -0
  272. package/dist/core/worktree-manager/manager.js +303 -0
  273. package/dist/index.js +28 -0
  274. package/dist/runtime/bootstrap.js +190 -0
  275. package/dist/runtime/cli.js +4151 -489
  276. package/dist/runtime/commands/agents.js +30 -30
  277. package/dist/runtime/commands/budget.js +5 -5
  278. package/dist/runtime/commands/cancel.js +231 -0
  279. package/dist/runtime/commands/chain.js +489 -0
  280. package/dist/runtime/commands/codegraph-status.js +227 -0
  281. package/dist/runtime/commands/compact.js +297 -0
  282. package/dist/runtime/commands/config.js +32 -32
  283. package/dist/runtime/commands/cost.js +199 -0
  284. package/dist/runtime/commands/delegate.js +244 -13
  285. package/dist/runtime/commands/dispatch.js +126 -0
  286. package/dist/runtime/commands/doctor.js +579 -0
  287. package/dist/runtime/commands/feedback.js +184 -0
  288. package/dist/runtime/commands/hooks.js +184 -0
  289. package/dist/runtime/commands/init.js +254 -0
  290. package/dist/runtime/commands/lsp.js +368 -0
  291. package/dist/runtime/commands/mcp.js +879 -0
  292. package/dist/runtime/commands/memory.js +582 -0
  293. package/dist/runtime/commands/model.js +237 -0
  294. package/dist/runtime/commands/onboarding.js +275 -0
  295. package/dist/runtime/commands/patch.js +128 -0
  296. package/dist/runtime/commands/permissions.js +112 -0
  297. package/dist/runtime/commands/plan.js +143 -0
  298. package/dist/runtime/commands/prd-check.js +285 -0
  299. package/dist/runtime/commands/privacy.js +17 -17
  300. package/dist/runtime/commands/recipe.js +325 -0
  301. package/dist/runtime/commands/redo-blob-store.js +92 -0
  302. package/dist/runtime/commands/redo.js +361 -0
  303. package/dist/runtime/commands/release-notes.js +229 -0
  304. package/dist/runtime/commands/repo-map.js +95 -0
  305. package/dist/runtime/commands/report.js +299 -0
  306. package/dist/runtime/commands/resume.js +118 -0
  307. package/dist/runtime/commands/review-consensus.js +68 -53
  308. package/dist/runtime/commands/rewind.js +333 -0
  309. package/dist/runtime/commands/roster.js +14 -14
  310. package/dist/runtime/commands/sessions.js +163 -0
  311. package/dist/runtime/commands/share.js +316 -0
  312. package/dist/runtime/commands/skills.js +31 -31
  313. package/dist/runtime/commands/status.js +186 -0
  314. package/dist/runtime/commands/stickers.js +82 -0
  315. package/dist/runtime/commands/style.js +194 -0
  316. package/dist/runtime/commands/theme.js +196 -0
  317. package/dist/runtime/commands/undo.js +54 -22
  318. package/dist/runtime/commands/update.js +289 -0
  319. package/dist/runtime/commands/vim.js +140 -0
  320. package/dist/runtime/commands/worktree.js +177 -0
  321. package/dist/runtime/commands/worktrees.js +155 -0
  322. package/dist/runtime/headless-repl.js +195 -0
  323. package/dist/runtime/headless.js +543 -0
  324. package/dist/runtime/load-hooks-or-exit.js +71 -0
  325. package/dist/runtime/plan-decompose.js +531 -0
  326. package/dist/runtime/update-check.js +28 -28
  327. package/dist/runtime/version.js +65 -0
  328. package/dist/skills/bundled/batch.js +617 -0
  329. package/dist/skills/bundled/index.js +45 -0
  330. package/dist/skills/bundled/loop.js +358 -0
  331. package/dist/skills/bundled/remember.js +383 -0
  332. package/dist/skills/bundled/simplify.js +289 -0
  333. package/dist/skills/bundled/skillify.js +373 -0
  334. package/dist/skills/bundled/stuck.js +558 -0
  335. package/dist/skills/bundled/verify.js +439 -0
  336. package/dist/testing/vcr.js +486 -0
  337. package/dist/tools/agent-tool.js +229 -0
  338. package/dist/tools/apply-patch.js +556 -0
  339. package/dist/tools/ask-user-question.js +222 -0
  340. package/dist/tools/ask-user.js +115 -0
  341. package/dist/tools/bash.js +623 -45
  342. package/dist/tools/brief.js +224 -0
  343. package/dist/tools/enter-worktree.js +250 -0
  344. package/dist/tools/exit-worktree.js +147 -0
  345. package/dist/tools/file-tools.js +161 -44
  346. package/dist/tools/lsp-tools.js +189 -0
  347. package/dist/tools/mcp-tool.js +260 -0
  348. package/dist/tools/multi-edit.js +361 -0
  349. package/dist/tools/powershell.js +268 -0
  350. package/dist/tools/registry.js +85 -0
  351. package/dist/tools/skill-tool.js +96 -0
  352. package/dist/tools/sleep.js +99 -0
  353. package/dist/tools/synthetic-output.js +133 -0
  354. package/dist/tools/tasks.js +208 -0
  355. package/dist/tools/todo-write.js +184 -0
  356. package/dist/tools/verify-plan-execution.js +295 -0
  357. package/dist/tools/web-fetch-injection-scanner.js +207 -0
  358. package/dist/tools/web-fetch.js +195 -10
  359. package/dist/tools/web-search.js +458 -0
  360. package/dist/tui/agent-progress-card.js +111 -0
  361. package/dist/tui/agent-tree.js +11 -1
  362. package/dist/tui/ask-modal.js +14 -14
  363. package/dist/tui/ask-user-question-prompt.js +203 -0
  364. package/dist/tui/compact-banner.js +81 -0
  365. package/dist/tui/conversation-pane.js +85 -11
  366. package/dist/tui/cost-table.js +111 -0
  367. package/dist/tui/device-flow.js +2 -2
  368. package/dist/tui/doctor-table.js +46 -0
  369. package/dist/tui/feedback-prompt.js +156 -0
  370. package/dist/tui/input-box.js +247 -32
  371. package/dist/tui/login-picker.js +3 -3
  372. package/dist/tui/markdown-render.js +6 -6
  373. package/dist/tui/onboarding-wizard.js +240 -0
  374. package/dist/tui/permissions-picker.js +86 -0
  375. package/dist/tui/render.js +35 -0
  376. package/dist/tui/repl-render.js +332 -54
  377. package/dist/tui/repl-splash-art.js +16 -16
  378. package/dist/tui/repl-splash-mascot.js +48 -24
  379. package/dist/tui/repl-splash.js +22 -22
  380. package/dist/tui/repl.js +124 -44
  381. package/dist/tui/slash-palette.js +6 -6
  382. package/dist/tui/splash.js +2 -2
  383. package/dist/tui/status-bar.js +109 -31
  384. package/dist/tui/status-table.js +7 -0
  385. package/dist/tui/stickers-art.js +136 -0
  386. package/dist/tui/style-table.js +28 -0
  387. package/dist/tui/theme-table.js +29 -0
  388. package/dist/tui/thinking-spinner.js +123 -0
  389. package/dist/tui/tool-stream-pane.js +53 -4
  390. package/dist/tui/update-banner.js +27 -2
  391. package/dist/tui/vim-input.js +267 -0
  392. package/dist/tui/welcome-banner.js +107 -0
  393. package/dist/tui/welcome-data.js +293 -0
  394. package/dist/tui/workspace-context.js +2 -2
  395. package/docs/examples/codegraph.mcp.json +10 -0
  396. package/package.json +23 -6
  397. package/test/scenarios/codegen-create-file.scenario.txt +13 -0
  398. package/test/scenarios/compact-force.scenario.txt +11 -0
  399. package/test/scenarios/identity.scenario.txt +11 -0
  400. package/test/scenarios/persona-handoff.scenario.txt +11 -0
  401. package/test/scenarios/walkback.scenario.txt +12 -0
  402. package/dist/core/engine/compaction-hook.js +0 -154
@@ -1,40 +1,40 @@
1
1
  /**
2
- * Tier 0 repo skeleton builder - α6.5 Phase 1 (three-tier context).
2
+ * Tier 0 repo skeleton builder - Phase 1 (three-tier context).
3
3
  *
4
4
  * The skeleton is the ~5KB "always loaded" tier in the three-tier
5
5
  * model. Goal: give the agent enough structural awareness to navigate
6
6
  * an unfamiliar repo on the first turn without uploading the whole
7
7
  * tree. We capture:
8
8
  *
9
- * - cwd + current branch (best-effort, no exec)
10
- * - detected package manager (lockfile heuristic)
11
- * - primary languages (file-extension histogram, top 5)
12
- * - ASCII directory tree (depth <= MAX_TREE_DEPTH, collapses busy
13
- * dirs to a "(N dirs, M files)" line)
14
- * - package.json projection (name/version/scripts/deps)
15
- * - first MAX_README_LINES of README.md
9
+ * - cwd + current branch (best-effort, no exec)
10
+ * - detected package manager (lockfile heuristic)
11
+ * - primary languages (file-extension histogram, top 5)
12
+ * - ASCII directory tree (depth <= MAX_TREE_DEPTH, collapses busy
13
+ * dirs to a "(N dirs, M files)" line)
14
+ * - package.json projection (name/version/scripts/deps)
15
+ * - first MAX_README_LINES of README.md
16
16
  *
17
17
  * Hard constraints:
18
18
  *
19
- * 1. **5KB cap on the rendered string**. We render in priority
20
- * order (headers + meta first, then tree, then package.json, then
21
- * README) and truncate the tail with a "..." marker rather than
22
- * omitting fields. The walker's bound is independent: it stops
23
- * visiting dirs after a hard `MAX_WALK_NODES` even when the
24
- * skeleton would still fit.
19
+ * 1. **5KB cap on the rendered string**. We render in priority
20
+ * order (headers + meta first, then tree, then package.json, then
21
+ * README) and truncate the tail with a "..." marker rather than
22
+ * omitting fields. The walker's bound is independent: it stops
23
+ * visiting dirs after a hard `MAX_WALK_NODES` even when the
24
+ * skeleton would still fit.
25
25
  *
26
- * 2. **Ignore-aware**: every fs read is filtered through `PugiIgnore`
27
- * so `.env`, `*.pem`, `node_modules/`, etc. stay out. The walker
28
- * uses the same matcher so `node_modules/` is never expanded.
26
+ * 2. **Ignore-aware**: every fs read is filtered through `PugiIgnore`
27
+ * so `.env`, `*.pem`, `node_modules/`, etc. stay out. The walker
28
+ * uses the same matcher so `node_modules/` is never expanded.
29
29
  *
30
- * 3. **No exec**: we read `.git/HEAD` for the branch, never spawn
31
- * `git`. The skeleton is built on session bootstrap; spawning
32
- * git on every launch is slow + makes the CLI flaky on hosts
33
- * without git installed.
30
+ * 3. **No exec**: we read `.git/HEAD` for the branch, never spawn
31
+ * `git`. The skeleton is built on session bootstrap; spawning
32
+ * git on every launch is slow + makes the CLI flaky on hosts
33
+ * without git installed.
34
34
  *
35
- * 4. **Best-effort**: every FS read is wrapped in try/catch.
36
- * Missing README / missing package.json / unreadable file -> the
37
- * field is omitted but the skeleton still builds.
35
+ * 4. **Best-effort**: every FS read is wrapped in try/catch.
36
+ * Missing README / missing package.json / unreadable file -> the
37
+ * field is omitted but the skeleton still builds.
38
38
  */
39
39
  import { existsSync, readdirSync, readFileSync, statSync } from 'node:fs';
40
40
  import { basename, join, resolve } from 'node:path';
@@ -64,7 +64,7 @@ export function buildRepoSkeleton(cwd, options) {
64
64
  // contracts, customer-specific data, NDA'd boilerplate) keeps those
65
65
  // files out of the agent context. Without this gate, the direct
66
66
  // skeleton reads would BYPASS the matcher even though the walker
67
- // honours it. triple-review P1 (PR #380).
67
+ // honours it. triple-review P1 (PR).
68
68
  const pkg = readPackageJson(normalised, ignore);
69
69
  const packageManager = detectPackageManager(normalised, ignore);
70
70
  const walk = walkTree(normalised, ignore);
@@ -93,10 +93,10 @@ export function buildRepoSkeleton(cwd, options) {
93
93
  *
94
94
  * Section order (highest priority first):
95
95
  *
96
- * 1. Meta header (cwd / branch / package manager / languages)
97
- * 2. Directory tree
98
- * 3. package.json projection (name / version / scripts / deps count)
99
- * 4. README excerpt
96
+ * 1. Meta header (cwd / branch / package manager / languages)
97
+ * 2. Directory tree
98
+ * 3. package.json projection (name / version / scripts / deps count)
99
+ * 4. README excerpt
100
100
  *
101
101
  * Each lower-priority section is appended only when there is room for
102
102
  * its FULL body. Partial sections would mislead the model (e.g. a
@@ -132,7 +132,7 @@ export function renderSkeleton(skeleton) {
132
132
  return out;
133
133
  }
134
134
  /* ------------------------------------------------------------------ */
135
- /* Section renderers */
135
+ /* Section renderers */
136
136
  /* ------------------------------------------------------------------ */
137
137
  function renderHeader(skeleton) {
138
138
  const lines = [];
@@ -184,7 +184,7 @@ function renderReadmeSection(skeleton) {
184
184
  return `\n## README (first ${MAX_README_LINES} lines)\n${skeleton.readmeExcerpt}`;
185
185
  }
186
186
  /* ------------------------------------------------------------------ */
187
- /* Signal readers */
187
+ /* Signal readers */
188
188
  /* ------------------------------------------------------------------ */
189
189
  /**
190
190
  * Read the current branch from `.git/HEAD` WITHOUT spawning git. The
@@ -264,7 +264,7 @@ export function readPackageJson(cwd, ignore) {
264
264
  // package.json to .pugiignore (private template projects, customer
265
265
  // boilerplate under NDA) must not have it read into the agent
266
266
  // context even though the walker would have skipped it.
267
- // triple-review P1 (PR #380). The matcher is optional so direct callers
267
+ // triple-review P1 (PR). The matcher is optional so direct callers
268
268
  // (e.g. unit tests of readPackageJson in isolation) keep working.
269
269
  if (ignore?.isIgnored(pkgPath))
270
270
  return undefined;
@@ -298,7 +298,7 @@ export function readReadme(cwd, overridePath, ignore) {
298
298
  for (const candidate of candidates) {
299
299
  // Skip candidates the operator has explicitly ignored. A workspace
300
300
  // README that documents private contracts / customer data must
301
- // never land in the rendered skeleton. triple-review P1 (PR #380).
301
+ // never land in the rendered skeleton. triple-review P1 (PR).
302
302
  if (ignore?.isIgnored(candidate))
303
303
  continue;
304
304
  const raw = safeRead(candidate);
@@ -383,7 +383,7 @@ function walkTree(cwd, ignore) {
383
383
  }
384
384
  /**
385
385
  * Render a `DirNode` as a UTF-8 ASCII-tree string. Lines use the
386
- * Claude Code convention: `├──` / `└──` glyphs, two-space indents per
386
+ * the upstream tool convention: `├──` / `└──` glyphs, two-space indents per
387
387
  * level. Collapsed dirs land as `name/ (N dirs, M files)`.
388
388
  */
389
389
  function renderTree(root) {
@@ -408,12 +408,12 @@ function appendChildren(node, prefix, lines) {
408
408
  const isLast = i === children.length - 1;
409
409
  const glyph = isLast ? '└── ' : '├── ';
410
410
  lines.push(`${prefix}${glyph}${child.name}/${counts(child)}`);
411
- const nextPrefix = `${prefix}${isLast ? ' ' : '│ '}`;
411
+ const nextPrefix = `${prefix}${isLast ? ' ' : '│ '}`;
412
412
  appendChildren(child, nextPrefix, lines);
413
413
  }
414
414
  }
415
415
  /* ------------------------------------------------------------------ */
416
- /* Language histogram */
416
+ /* Language histogram */
417
417
  /* ------------------------------------------------------------------ */
418
418
  /**
419
419
  * Map a filename to a human-readable language label. Returns null for
@@ -495,7 +495,7 @@ export function topLanguages(histogram, topN) {
495
495
  return entries.slice(0, topN).map(([lang]) => lang);
496
496
  }
497
497
  /* ------------------------------------------------------------------ */
498
- /* Small helpers */
498
+ /* Small helpers */
499
499
  /* ------------------------------------------------------------------ */
500
500
  function safeRead(path) {
501
501
  // Stat first to keep the regular-file guard (Codex R2 P2): a FIFO or
@@ -0,0 +1,55 @@
1
+ export function estimateTokens(text) {
2
+ return Math.ceil(text.length / 4);
3
+ }
4
+ export function evictByLRU(entries, targetTokens) {
5
+ const sorted = [...entries].sort((a, b) => a.usedAt.localeCompare(b.usedAt));
6
+ const total = sorted.reduce((sum, e) => sum + e.tokensEst, 0);
7
+ if (total <= targetTokens) {
8
+ return {
9
+ evicted: [],
10
+ kept: sorted.map((e) => e.id),
11
+ freedTokens: 0,
12
+ };
13
+ }
14
+ const evicted = [];
15
+ const kept = [];
16
+ let freedTokens = 0;
17
+ let remaining = total;
18
+ for (const entry of sorted) {
19
+ if (remaining > targetTokens) {
20
+ evicted.push(entry.id);
21
+ freedTokens += entry.tokensEst;
22
+ remaining -= entry.tokensEst;
23
+ }
24
+ else {
25
+ kept.push(entry.id);
26
+ }
27
+ }
28
+ return { evicted, kept, freedTokens };
29
+ }
30
+ export function evictByContentClass(entries, rules, now = new Date()) {
31
+ const ruleByTool = new Map(rules.map((r) => [r.tool, r]));
32
+ const nowMs = now.getTime();
33
+ const evicted = [];
34
+ const kept = [];
35
+ let freedTokens = 0;
36
+ for (const entry of entries) {
37
+ const rule = ruleByTool.get(entry.tool);
38
+ if (!rule) {
39
+ kept.push(entry.id);
40
+ continue;
41
+ }
42
+ const ageMs = nowMs - new Date(entry.ts).getTime();
43
+ const tooOld = ageMs > rule.maxAgeMs;
44
+ const tooBig = entry.tokensEst > rule.maxTokens;
45
+ if (tooOld || tooBig) {
46
+ evicted.push(entry.id);
47
+ freedTokens += entry.tokensEst;
48
+ }
49
+ else {
50
+ kept.push(entry.id);
51
+ }
52
+ }
53
+ return { evicted, kept, freedTokens };
54
+ }
55
+ //# sourceMappingURL=tool-eviction.js.map
@@ -1,5 +1,5 @@
1
1
  /**
2
- * chokidar-based filewatch - α6.5 Phase 1 (three-tier context).
2
+ * chokidar-based filewatch - Phase 1 (three-tier context).
3
3
  *
4
4
  * The REPL needs a live signal when files change so the operator's
5
5
  * agent can re-read a stale file, and so the status bar can surface a
@@ -8,35 +8,35 @@
8
8
  *
9
9
  * Design choices:
10
10
  *
11
- * 1. **Ignore-aware**: every watched path is filtered through the
12
- * same `PugiIgnore` matcher the skeleton walker uses. We pass the
13
- * matcher into chokidar's `ignored` option so `node_modules/`,
14
- * `dist/`, `.git/`, and secret files are never watched at all -
15
- * this matters because chokidar's resource bound is the number
16
- * of watched paths, not the number of total files in the repo.
11
+ * 1. **Ignore-aware**: every watched path is filtered through the
12
+ * same `PugiIgnore` matcher the skeleton walker uses. We pass the
13
+ * matcher into chokidar's `ignored` option so `node_modules/`,
14
+ * `dist/`, `.git/`, and secret files are never watched at all -
15
+ * this matters because chokidar's resource bound is the number
16
+ * of watched paths, not the number of total files in the repo.
17
17
  *
18
- * 2. **Throttle**: chokidar can fire dozens of `change` events when
19
- * a code formatter rewrites a directory. We batch events in a
20
- * `THROTTLE_WINDOW_MS` window and emit a single aggregated event
21
- * so the REPL system line / status bar do not flicker.
18
+ * 2. **Throttle**: chokidar can fire dozens of `change` events when
19
+ * a code formatter rewrites a directory. We batch events in a
20
+ * `THROTTLE_WINDOW_MS` window and emit a single aggregated event
21
+ * so the REPL system line / status bar do not flicker.
22
22
  *
23
- * 3. **Watch cap**: per spec, the watcher caps watched paths at
24
- * `MAX_WATCHED_PATHS`. Once the count crosses the cap we close
25
- * the watcher and fall back to "no live updates this session" -
26
- * better to silently lose the badge than to consume thousands of
27
- * file descriptors. The fallback path emits one warning event so
28
- * the operator knows live updates are off.
23
+ * 3. **Watch cap**: per spec, the watcher caps watched paths at
24
+ * `MAX_WATCHED_PATHS`. Once the count crosses the cap we close
25
+ * the watcher and fall back to "no live updates this session" -
26
+ * better to silently lose the badge than to consume thousands of
27
+ * file descriptors. The fallback path emits one warning event so
28
+ * the operator knows live updates are off.
29
29
  *
30
- * 4. **Lifecycle**: `start()` is async (chokidar's `ready` event)
31
- * so the caller can `await` the initial scan. `close()` is also
32
- * async (chokidar's close is async). The watcher is reusable
33
- * across multiple `start` / `close` pairs, but Phase 1 expects
34
- * one-watcher-per-REPL-session.
30
+ * 4. **Lifecycle**: `start()` is async (chokidar's `ready` event)
31
+ * so the caller can `await` the initial scan. `close()` is also
32
+ * async (chokidar's close is async). The watcher is reusable
33
+ * across multiple `start` / `close` pairs, but Phase 1 expects
34
+ * one-watcher-per-REPL-session.
35
35
  *
36
- * 5. **Testability**: the chokidar handle is injectable via the
37
- * `watchFactory` option. Production wires `chokidar.watch`;
38
- * tests pass a fake emitter that lets the spec drive `add` /
39
- * `change` / `unlink` events synchronously.
36
+ * 5. **Testability**: the chokidar handle is injectable via the
37
+ * `watchFactory` option. Production wires `chokidar.watch`;
38
+ * tests pass a fake emitter that lets the spec drive `add` /
39
+ * `change` / `unlink` events synchronously.
40
40
  */
41
41
  import { EventEmitter } from 'node:events';
42
42
  import chokidar from 'chokidar';
@@ -50,7 +50,7 @@ export const THROTTLE_WINDOW_MS = 250;
50
50
  * giant subtree post-ready can blow past the cap without ever
51
51
  * re-evaluating. Sample every N adds so the cap fires within a
52
52
  * bounded window even on post-ready growth. Cheap: one Map
53
- * iteration per N adds. triple-review P2 (PR #380).
53
+ * iteration per N adds. triple-review P2 (PR).
54
54
  */
55
55
  export const CAP_CHECK_ADD_INTERVAL = 1_000;
56
56
  /**
@@ -76,7 +76,7 @@ export class PugiWatcher extends EventEmitter {
76
76
  /**
77
77
  * Counter of `add` events received since the last cap-check sample.
78
78
  * When it crosses CAP_CHECK_ADD_INTERVAL we schedule another check.
79
- * Resets on every cap-check run. triple-review P2 (PR #380).
79
+ * Resets on every cap-check run. triple-review P2 (PR).
80
80
  */
81
81
  addsSinceLastCapCheck = 0;
82
82
  constructor(options) {
@@ -97,7 +97,7 @@ export class PugiWatcher extends EventEmitter {
97
97
  // the event so emit() does not crash the CLI; any real consumer
98
98
  // that subscribes later receives the events as normal because
99
99
  // EventEmitter delivers to every registered listener, not just one.
100
- // Codex P1 (PR #380).
100
+ // Codex P1 (PR).
101
101
  this.on('error', () => {
102
102
  /* defensive no-op so emit('error') never throws */
103
103
  });
@@ -162,7 +162,7 @@ export class PugiWatcher extends EventEmitter {
162
162
  // post-ready burst (e.g. `git checkout` of a giant subtree)
163
163
  // does not silently blow past the watched-paths cap. The
164
164
  // ready-time check is preserved; this is the missing follow-up
165
- // sampler the comment promised. triple-review P2 (PR #380).
165
+ // sampler the comment promised. triple-review P2 (PR).
166
166
  this.addsSinceLastCapCheck += 1;
167
167
  if (this.addsSinceLastCapCheck >= CAP_CHECK_ADD_INTERVAL) {
168
168
  this.addsSinceLastCapCheck = 0;
@@ -225,7 +225,7 @@ export class PugiWatcher extends EventEmitter {
225
225
  return this.buffer.length;
226
226
  }
227
227
  /* ------------------------------------------------------------ */
228
- /* Internals */
228
+ /* Internals */
229
229
  /* ------------------------------------------------------------ */
230
230
  queue(kind, path) {
231
231
  if (this.closed)
@@ -291,7 +291,7 @@ export class PugiWatcher extends EventEmitter {
291
291
  }
292
292
  }
293
293
  /* ------------------------------------------------------------------ */
294
- /* Helpers */
294
+ /* Helpers */
295
295
  /* ------------------------------------------------------------------ */
296
296
  /**
297
297
  * Convert an absolute path to a repo-relative POSIX path. chokidar
@@ -1,41 +1,41 @@
1
1
  /**
2
- * Tier 1 working-set tracker - α6.5 Phase 1 (three-tier context).
2
+ * Tier 1 working-set tracker - Phase 1 (three-tier context).
3
3
  *
4
4
  * The model context window is finite. Tier 0 (repo skeleton, ~5KB) is
5
- * always loaded; Tier 2 (RAG, Anvil-side, deferred to α6.5b) answers
5
+ * always loaded; Tier 2 (RAG, Anvil-side, deferred to ) answers
6
6
  * queries on demand. Tier 1 - the working set - lives between them: the
7
7
  * subset of files the agent has touched THIS session, bounded so a
8
8
  * long REPL session does not unbounded-grow the system prompt.
9
9
  *
10
10
  * Design choices:
11
11
  *
12
- * 1. **LRU semantics**: the most-recently-touched file stays in the
13
- * window; the oldest is evicted when the cap is hit. We use a
14
- * `Map` so insertion order doubles as recency order (delete +
15
- * re-set bubbles the entry to the tail).
12
+ * 1. **LRU semantics**: the most-recently-touched file stays in the
13
+ * window; the oldest is evicted when the cap is hit. We use a
14
+ * `Map` so insertion order doubles as recency order (delete +
15
+ * re-set bubbles the entry to the tail).
16
16
  *
17
- * 2. **Source tracking**: every `track()` call records the source -
18
- * `read` (agent fetched the file), `edit` (agent modified it),
19
- * `write` (agent created it). The source seeds future heuristics
20
- * (e.g. always pin edited files near the cap, never evict files
21
- * written this session) but Phase 1 keeps the policy simple:
22
- * pure LRU.
17
+ * 2. **Source tracking**: every `track()` call records the source -
18
+ * `read` (agent fetched the file), `edit` (agent modified it),
19
+ * `write` (agent created it). The source seeds future heuristics
20
+ * (e.g. always pin edited files near the cap, never evict files
21
+ * written this session) but Phase 1 keeps the policy simple:
22
+ * pure LRU.
23
23
  *
24
- * 3. **No persistence**: working set is in-memory only. A REPL
25
- * restart rebuilds from scratch (and from `/resume` SessionStore
26
- * replay if/when we wire that). Per α6.5 spec: "node:sqlite NOT
27
- * needed for this sprint."
24
+ * 3. **No persistence**: working set is in-memory only. A REPL
25
+ * restart rebuilds from scratch (and from `/resume` SessionStore
26
+ * replay if/when we wire that). Per spec: "node:sqlite NOT
27
+ * needed for this sprint."
28
28
  *
29
- * 4. **Path normalisation**: we resolve every path to an absolute
30
- * form on insert so two `track()` calls with `./foo.ts` and
31
- * `/abs/cwd/foo.ts` collapse to one entry.
29
+ * 4. **Path normalisation**: we resolve every path to an absolute
30
+ * form on insert so two `track()` calls with `./foo.ts` and
31
+ * `/abs/cwd/foo.ts` collapse to one entry.
32
32
  *
33
- * 5. **Defensive cap**: `MAX_WORKING_SET = 50` per spec. We hold the
34
- * cap as a configurable option so tests can dial it down to
35
- * exercise eviction without 51 fixture files.
33
+ * 5. **Defensive cap**: `MAX_WORKING_SET = 50` per spec. We hold the
34
+ * cap as a configurable option so tests can dial it down to
35
+ * exercise eviction without 51 fixture files.
36
36
  */
37
37
  import { resolve } from 'node:path';
38
- /** Spec-mandated default cap. Mirrors the doc string in the α6.5 sprint plan. */
38
+ /** Spec-mandated default cap. Mirrors the doc string in the sprint plan. */
39
39
  export const DEFAULT_WORKING_SET_CAPACITY = 50;
40
40
  export class WorkingSet {
41
41
  entries = new Map();
@@ -0,0 +1,77 @@
1
+ function snapshot(task) {
2
+ return { ...task };
3
+ }
4
+ function isTerminal(status) {
5
+ return status === 'completed' || status === 'cancelled' || status === 'failed';
6
+ }
7
+ export function createAgentCoordinator(options) {
8
+ const tasks = new Map();
9
+ return {
10
+ async createTask(agentType, prompt) {
11
+ const { taskId } = await options.spawn(agentType, prompt);
12
+ const task = {
13
+ id: taskId,
14
+ agentType,
15
+ prompt,
16
+ status: 'queued',
17
+ createdAt: Date.now(),
18
+ };
19
+ tasks.set(task.id, task);
20
+ task.status = 'running';
21
+ task.startedAt = Date.now();
22
+ return snapshot(task);
23
+ },
24
+ async sendMessage(taskId, message) {
25
+ const task = tasks.get(taskId);
26
+ if (!task) {
27
+ throw new Error(`agent task ${taskId} not found`);
28
+ }
29
+ if (task.status !== 'running') {
30
+ throw new Error(`agent task ${taskId} is not running (status=${task.status})`);
31
+ }
32
+ await options.send(taskId, message);
33
+ },
34
+ async stopTask(taskId) {
35
+ const task = tasks.get(taskId);
36
+ if (!task) {
37
+ throw new Error(`agent task ${taskId} not found`);
38
+ }
39
+ if (isTerminal(task.status)) {
40
+ return;
41
+ }
42
+ await options.stop(taskId);
43
+ task.status = 'cancelled';
44
+ task.completedAt = Date.now();
45
+ },
46
+ listTasks() {
47
+ return Array.from(tasks.values())
48
+ .sort((a, b) => b.createdAt - a.createdAt)
49
+ .map(snapshot);
50
+ },
51
+ getTask(taskId) {
52
+ const task = tasks.get(taskId);
53
+ return task ? snapshot(task) : null;
54
+ },
55
+ markCompleted(taskId, output) {
56
+ const task = tasks.get(taskId);
57
+ if (!task) {
58
+ return;
59
+ }
60
+ task.status = 'completed';
61
+ task.completedAt = Date.now();
62
+ task.output = output;
63
+ delete task.error;
64
+ },
65
+ markFailed(taskId, error) {
66
+ const task = tasks.get(taskId);
67
+ if (!task) {
68
+ return;
69
+ }
70
+ task.status = 'failed';
71
+ task.completedAt = Date.now();
72
+ task.error = error;
73
+ delete task.output;
74
+ },
75
+ };
76
+ }
77
+ //# sourceMappingURL=agent-tools.js.map
@@ -0,0 +1,65 @@
1
+ // apps/pugi-cli/src/core/coordinator/agent-toolset.ts
2
+ import { randomUUID } from 'node:crypto';
3
+ export class AgentCoordinator {
4
+ agents = new Map();
5
+ spawnAgent(subagentType, prompt) {
6
+ const handle = {
7
+ id: randomUUID(),
8
+ subagentType,
9
+ prompt,
10
+ status: 'spawned',
11
+ spawnedAt: new Date().toISOString(),
12
+ };
13
+ this.agents.set(handle.id, handle);
14
+ return handle;
15
+ }
16
+ sendMessage(agentId, _message) {
17
+ const agent = this.agents.get(agentId);
18
+ if (!agent) {
19
+ return { ok: false, reason: `agent ${agentId} not found` };
20
+ }
21
+ if (agent.status !== 'spawned' && agent.status !== 'running') {
22
+ return { ok: false, reason: `agent ${agentId} not accepting messages (status=${agent.status})` };
23
+ }
24
+ return { ok: true };
25
+ }
26
+ taskStop(agentId) {
27
+ const agent = this.agents.get(agentId);
28
+ if (!agent) {
29
+ return { ok: false, reason: `agent ${agentId} not found` };
30
+ }
31
+ agent.status = 'stopped';
32
+ agent.completedAt = new Date().toISOString();
33
+ return { ok: true };
34
+ }
35
+ markCompleted(agentId, result) {
36
+ const agent = this.agents.get(agentId);
37
+ if (!agent) {
38
+ return;
39
+ }
40
+ agent.status = 'completed';
41
+ agent.completedAt = new Date().toISOString();
42
+ agent.result = result;
43
+ }
44
+ markFailed(agentId, reason) {
45
+ const agent = this.agents.get(agentId);
46
+ if (!agent) {
47
+ return;
48
+ }
49
+ agent.status = 'failed';
50
+ agent.completedAt = new Date().toISOString();
51
+ agent.result = reason;
52
+ }
53
+ listAgents(filter) {
54
+ const all = Array.from(this.agents.values());
55
+ if (!filter?.status) {
56
+ return all;
57
+ }
58
+ const wanted = filter.status;
59
+ return all.filter((a) => a.status === wanted);
60
+ }
61
+ getAgent(id) {
62
+ return this.agents.get(id);
63
+ }
64
+ }
65
+ //# sourceMappingURL=agent-toolset.js.map
@@ -0,0 +1,73 @@
1
+ export class CoordinatorFSM {
2
+ state;
3
+ constructor() {
4
+ this.state = {
5
+ phase: 'research',
6
+ inflightOps: [],
7
+ completedPhases: [],
8
+ };
9
+ }
10
+ getState() {
11
+ // Return a shallow copy to prevent direct mutation of the internal state.
12
+ return {
13
+ ...this.state,
14
+ inflightOps: [...this.state.inflightOps],
15
+ completedPhases: [...this.state.completedPhases],
16
+ };
17
+ }
18
+ canStart(op) {
19
+ switch (this.state.phase) {
20
+ case 'research':
21
+ return op === 'read';
22
+ case 'synth':
23
+ return op === 'read';
24
+ case 'impl':
25
+ if (op === 'write') {
26
+ // 'write' is only allowed when no other operations are in flight.
27
+ return this.state.inflightOps.length === 0;
28
+ }
29
+ // 'read' and 'exec' are allowed as long as no 'write' is in flight.
30
+ return !this.state.inflightOps.includes('write');
31
+ case 'done':
32
+ return false;
33
+ default:
34
+ return false;
35
+ }
36
+ }
37
+ start(op) {
38
+ if (!this.canStart(op)) {
39
+ throw new Error(`Operation '${op}' cannot be started in phase '${this.state.phase}' with inflight ops: [${this.state.inflightOps.join(', ')}]`);
40
+ }
41
+ this.state.inflightOps.push(op);
42
+ }
43
+ finish(op) {
44
+ const opIndex = this.state.inflightOps.indexOf(op);
45
+ if (opIndex === -1) {
46
+ throw new Error(`Cannot finish operation '${op}' as it is not in flight. Inflight ops: [${this.state.inflightOps.join(', ')}]`);
47
+ }
48
+ this.state.inflightOps.splice(opIndex, 1);
49
+ }
50
+ advance() {
51
+ if (this.state.inflightOps.length > 0) {
52
+ throw new Error(`Cannot advance phase with operations in flight: [${this.state.inflightOps.join(', ')}]`);
53
+ }
54
+ const currentPhase = this.state.phase;
55
+ let nextPhase;
56
+ switch (currentPhase) {
57
+ case 'research':
58
+ nextPhase = 'synth';
59
+ break;
60
+ case 'synth':
61
+ nextPhase = 'impl';
62
+ break;
63
+ case 'impl':
64
+ nextPhase = 'done';
65
+ break;
66
+ case 'done':
67
+ throw new Error("Cannot advance from final phase 'done'.");
68
+ }
69
+ this.state.completedPhases.push(currentPhase);
70
+ this.state.phase = nextPhase;
71
+ }
72
+ }
73
+ //# sourceMappingURL=fsm.js.map