@pugi/cli 0.1.0-beta.10 → 0.1.0-beta.101

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 (464) hide show
  1. package/CHANGELOG.md +132 -0
  2. package/LICENSE +1 -1
  3. package/README.md +55 -11
  4. package/assets/pugi-prozr2-mascot.ansi +9 -0
  5. package/bin/run.js +33 -1
  6. package/dist/commands/deploy.js +40 -40
  7. package/dist/commands/flatten.js +191 -0
  8. package/dist/commands/jobs-watch.js +201 -0
  9. package/dist/commands/jobs.js +42 -27
  10. package/dist/commands/retro.js +210 -0
  11. package/dist/commands/smoke.js +133 -0
  12. package/dist/core/agent-progress/cleanup.js +134 -0
  13. package/dist/core/agent-progress/schema.js +144 -0
  14. package/dist/core/agent-progress/writer.js +101 -0
  15. package/dist/core/agents/adaptive-router.js +330 -0
  16. package/dist/core/agents/query-decomposer.js +297 -0
  17. package/dist/core/agents/registry.js +3 -3
  18. package/dist/core/approvals/shortcut-resolver.js +98 -0
  19. package/dist/core/artifact-chain/dispatcher.js +148 -0
  20. package/dist/core/artifact-chain/exporter.js +164 -0
  21. package/dist/core/artifact-chain/state.js +243 -0
  22. package/dist/core/artifact-chain/steps.js +169 -0
  23. package/dist/core/ask-user/question.js +92 -0
  24. package/dist/core/audit/audit-trail.js +275 -0
  25. package/dist/core/auth/ensure-authenticated.js +129 -0
  26. package/dist/core/auth/env-provider.js +238 -0
  27. package/dist/core/auto-open-browser.js +4 -4
  28. package/dist/core/auto-update/channels.js +122 -0
  29. package/dist/core/auto-update/checker.js +241 -0
  30. package/dist/core/auto-update/state.js +235 -0
  31. package/dist/core/bare-mode/index.js +107 -0
  32. package/dist/core/bash/redirect.js +281 -0
  33. package/dist/core/bash-classifier.js +436 -40
  34. package/dist/core/checkpoint/resumer.js +149 -0
  35. package/dist/core/checkpoint/rewinder.js +291 -0
  36. package/dist/core/checkpoints/shadow-git.js +670 -0
  37. package/dist/core/citations/parser.js +109 -0
  38. package/dist/core/classifier/yolo-classifier.js +88 -0
  39. package/dist/core/codegraph/db.js +506 -0
  40. package/dist/core/codegraph/decision-store.js +248 -0
  41. package/dist/core/codegraph/detect-repo.js +459 -0
  42. package/dist/core/codegraph/install.js +134 -0
  43. package/dist/core/codegraph/offer-hook.js +220 -0
  44. package/dist/core/codegraph/parser.js +598 -0
  45. package/dist/core/codegraph/queries/go.scm +57 -0
  46. package/dist/core/codegraph/queries/javascript.scm +56 -0
  47. package/dist/core/codegraph/queries/python.scm +55 -0
  48. package/dist/core/codegraph/queries/rust.scm +63 -0
  49. package/dist/core/codegraph/queries/typescript.scm +91 -0
  50. package/dist/core/codegraph/reindex.js +218 -0
  51. package/dist/core/codegraph/resolve-edges.js +107 -0
  52. package/dist/core/codegraph/types.js +34 -0
  53. package/dist/core/codegraph/watcher.js +440 -0
  54. package/dist/core/compact/auto-trigger.js +96 -0
  55. package/dist/core/compact/buffer-rewriter.js +115 -0
  56. package/dist/core/compact/summarizer.js +208 -0
  57. package/dist/core/compact/token-counter.js +108 -0
  58. package/dist/core/consensus/anvil-fanout.js +25 -25
  59. package/dist/core/consensus/diff-capture.js +121 -12
  60. package/dist/core/consensus/rubric.js +21 -21
  61. package/dist/core/context/builder.js +6 -6
  62. package/dist/core/context/compaction-events.js +8 -8
  63. package/dist/core/context/compaction.js +31 -31
  64. package/dist/core/context/index.js +15 -8
  65. package/dist/core/context/invariants.js +51 -51
  66. package/dist/core/context/markdown-loader.js +28 -10
  67. package/dist/core/context/markdown-traverse.js +255 -0
  68. package/dist/core/context/pugiignore.js +41 -41
  69. package/dist/core/context/repo-skeleton.js +37 -37
  70. package/dist/core/context/tool-eviction.js +55 -0
  71. package/dist/core/context/watcher.js +32 -32
  72. package/dist/core/context/working-set.js +23 -23
  73. package/dist/core/coordinator/agent-tools.js +77 -0
  74. package/dist/core/coordinator/agent-toolset.js +65 -0
  75. package/dist/core/coordinator/fsm.js +73 -0
  76. package/dist/core/coordinator/mode-fsm.js +70 -0
  77. package/dist/core/cost/rate-card.js +129 -0
  78. package/dist/core/cost/tracker.js +221 -0
  79. package/dist/core/credentials.js +13 -13
  80. package/dist/core/cron/scheduler.js +138 -0
  81. package/dist/core/denial-tracking/index.js +8 -0
  82. package/dist/core/denial-tracking/state.js +264 -0
  83. package/dist/core/diagnostics/probe-runner.js +93 -0
  84. package/dist/core/diagnostics/probes/api.js +46 -0
  85. package/dist/core/diagnostics/probes/auth.js +93 -0
  86. package/dist/core/diagnostics/probes/bare-mode.js +42 -0
  87. package/dist/core/diagnostics/probes/cli-version.js +127 -0
  88. package/dist/core/diagnostics/probes/config.js +72 -0
  89. package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
  90. package/dist/core/diagnostics/probes/disk.js +81 -0
  91. package/dist/core/diagnostics/probes/engine-live.js +46 -0
  92. package/dist/core/diagnostics/probes/git.js +65 -0
  93. package/dist/core/diagnostics/probes/hooks.js +118 -0
  94. package/dist/core/diagnostics/probes/mcp.js +75 -0
  95. package/dist/core/diagnostics/probes/node.js +59 -0
  96. package/dist/core/diagnostics/probes/pnpm.js +36 -0
  97. package/dist/core/diagnostics/probes/pugi-md.js +89 -0
  98. package/dist/core/diagnostics/probes/sandbox.js +67 -0
  99. package/dist/core/diagnostics/probes/session.js +74 -0
  100. package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
  101. package/dist/core/diagnostics/probes/workspace.js +63 -0
  102. package/dist/core/diagnostics/types.js +70 -0
  103. package/dist/core/dispatch/cache-cleanup.js +197 -0
  104. package/dist/core/dispatch/cache-handoff.js +295 -0
  105. package/dist/core/edits/apply-patch-layer-e.js +189 -0
  106. package/dist/core/edits/dispatch.js +333 -7
  107. package/dist/core/edits/format-detector.js +260 -0
  108. package/dist/core/edits/format-matrix.js +26 -0
  109. package/dist/core/edits/fuzzy-ladder.js +650 -0
  110. package/dist/core/edits/index.js +5 -1
  111. package/dist/core/edits/journal.js +199 -0
  112. package/dist/core/edits/layer-a-apply.js +15 -15
  113. package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
  114. package/dist/core/edits/layer-b-apply.js +9 -9
  115. package/dist/core/edits/layer-c-apply.js +6 -6
  116. package/dist/core/edits/layer-d-ast.js +557 -14
  117. package/dist/core/edits/marker-parser.js +12 -12
  118. package/dist/core/edits/security-gate.js +27 -27
  119. package/dist/core/edits/verify-hook.js +273 -0
  120. package/dist/core/edits/worktree.js +29 -29
  121. package/dist/core/engine/anvil-client.js +214 -26
  122. package/dist/core/engine/auto-compact.js +247 -0
  123. package/dist/core/engine/budgets.js +220 -0
  124. package/dist/core/engine/compact-llm-summarizer.js +124 -0
  125. package/dist/core/engine/context-prefix.js +155 -0
  126. package/dist/core/engine/index.js +1 -1
  127. package/dist/core/engine/intensity.js +163 -0
  128. package/dist/core/engine/intent.js +260 -0
  129. package/dist/core/engine/native-pugi.js +1559 -227
  130. package/dist/core/engine/prompts.js +219 -19
  131. package/dist/core/engine/strip-internal-fields.js +124 -0
  132. package/dist/core/engine/tool-bridge.js +1887 -59
  133. package/dist/core/engine/verification-patterns.js +195 -0
  134. package/dist/core/eval/v1/ledger.js +83 -0
  135. package/dist/core/eval/v1/runner.js +280 -0
  136. package/dist/core/eval/v1/scoring.js +68 -0
  137. package/dist/core/eval/v1/task-loader.js +191 -0
  138. package/dist/core/eval/v1/types.js +14 -0
  139. package/dist/core/eval/v1/verifier.js +176 -0
  140. package/dist/core/eval/v1/yaml-parser.js +250 -0
  141. package/dist/core/evaluation/golden-dataset.js +293 -0
  142. package/dist/core/feedback/queue.js +177 -0
  143. package/dist/core/feedback/submitter.js +145 -0
  144. package/dist/core/file-cache.js +113 -1
  145. package/dist/core/flatten/flatten-repo.js +439 -0
  146. package/dist/core/format/osc8-link.js +28 -0
  147. package/dist/core/hook-chains.js +392 -0
  148. package/dist/core/hooks/citation-verify-hook.js +138 -0
  149. package/dist/core/hooks/citation-verify.js +112 -0
  150. package/dist/core/hooks/events.js +46 -0
  151. package/dist/core/hooks/index.js +15 -0
  152. package/dist/core/hooks/registry.js +216 -0
  153. package/dist/core/hooks/runner.js +236 -0
  154. package/dist/core/hooks/v2/event-emitter.js +115 -0
  155. package/dist/core/hooks/v2/executor.js +282 -0
  156. package/dist/core/hooks/v2/index.js +25 -0
  157. package/dist/core/hooks/v2/lifecycle.js +104 -0
  158. package/dist/core/hooks/v2/loader.js +216 -0
  159. package/dist/core/hooks/v2/matcher.js +125 -0
  160. package/dist/core/hooks/v2/trust.js +143 -0
  161. package/dist/core/hooks/v2/types.js +86 -0
  162. package/dist/core/hooks/worktree-events.js +158 -0
  163. package/dist/core/image/renderer.js +71 -0
  164. package/dist/core/init/detector.js +582 -0
  165. package/dist/core/init/template-renderer.js +242 -0
  166. package/dist/core/jobs/registry.js +18 -18
  167. package/dist/core/ledger/results-tsv.js +142 -0
  168. package/dist/core/log-discipline/stdout-redirect.js +51 -0
  169. package/dist/core/lsp/cache.js +105 -0
  170. package/dist/core/lsp/client.js +551 -41
  171. package/dist/core/lsp/language-detect.js +66 -0
  172. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  173. package/dist/core/lsp/server-detect.js +173 -0
  174. package/dist/core/lsp/symbol-cache.js +162 -0
  175. package/dist/core/lsp/symbol-tools.js +664 -0
  176. package/dist/core/mcp/client.js +97 -28
  177. package/dist/core/mcp/http-server.js +553 -0
  178. package/dist/core/mcp/orchestrator-config.js +192 -0
  179. package/dist/core/mcp/orchestrator-tools.js +806 -0
  180. package/dist/core/mcp/permission.js +190 -0
  181. package/dist/core/mcp/registry.js +39 -17
  182. package/dist/core/mcp/server-tools.js +219 -0
  183. package/dist/core/mcp/server.js +397 -0
  184. package/dist/core/mcp/trust.js +10 -10
  185. package/dist/core/memory/dual-write.js +416 -0
  186. package/dist/core/memory/passive-extract.js +130 -0
  187. package/dist/core/memory/phase1-kinds.js +20 -0
  188. package/dist/core/memory/secret-scanner.js +304 -0
  189. package/dist/core/memory-sync/queue.js +170 -0
  190. package/dist/core/metrics/extract.js +113 -0
  191. package/dist/core/modes/roo-modes.js +68 -0
  192. package/dist/core/notes/notes-paths.js +113 -0
  193. package/dist/core/notes/notes-recorder.js +140 -0
  194. package/dist/core/notes/notes-writer.js +53 -0
  195. package/dist/core/notes/renderers.js +0 -0
  196. package/dist/core/notes/slug.js +105 -0
  197. package/dist/core/onboarding/ensure-initialized.js +133 -0
  198. package/dist/core/onboarding/marker.js +111 -0
  199. package/dist/core/onboarding/telemetry-state.js +108 -0
  200. package/dist/core/output-style/presets.js +176 -0
  201. package/dist/core/output-style/state.js +185 -0
  202. package/dist/core/path-security.js +287 -5
  203. package/dist/core/permission.js +82 -22
  204. package/dist/core/permissions/auto-classifier.js +124 -0
  205. package/dist/core/permissions/bash-parser.js +371 -0
  206. package/dist/core/permissions/circuit-breaker.js +83 -0
  207. package/dist/core/permissions/constrained-edit.js +91 -0
  208. package/dist/core/permissions/gate.js +278 -0
  209. package/dist/core/permissions/index.js +20 -0
  210. package/dist/core/permissions/mode.js +174 -0
  211. package/dist/core/permissions/network-egress.js +137 -0
  212. package/dist/core/permissions/state.js +241 -0
  213. package/dist/core/permissions/tool-class.js +107 -0
  214. package/dist/core/plan-mode/ui-state.js +51 -0
  215. package/dist/core/plans/plan-artifact.js +721 -0
  216. package/dist/core/policy-limits/etag-store.js +122 -0
  217. package/dist/core/prd-check/parser.js +215 -0
  218. package/dist/core/prd-check/reporter.js +127 -0
  219. package/dist/core/prd-check/session-review.js +557 -0
  220. package/dist/core/prd-check/verifiers.js +223 -0
  221. package/dist/core/prompt-cache/client-cache.js +99 -0
  222. package/dist/core/prompts/assembly.js +29 -0
  223. package/dist/core/prompts/registry.js +364 -0
  224. package/dist/core/pugi-gitignore.js +52 -0
  225. package/dist/core/pugi-md/cc-compat-rules.js +735 -0
  226. package/dist/core/pugi-md/context-injector.js +76 -0
  227. package/dist/core/pugi-md/walk-up.js +207 -0
  228. package/dist/core/python/uv-installer.js +270 -0
  229. package/dist/core/python/uv-resolver.js +83 -0
  230. package/dist/core/rate-limit/narrator.js +146 -0
  231. package/dist/core/recipes/cli-types.js +20 -0
  232. package/dist/core/recipes/loader.js +103 -0
  233. package/dist/core/recipes/runner.js +345 -0
  234. package/dist/core/recipes/schema.js +587 -0
  235. package/dist/core/release-notes/parser.js +241 -0
  236. package/dist/core/release-notes/state.js +116 -0
  237. package/dist/core/repl/ask.js +37 -37
  238. package/dist/core/repl/cancellation.js +26 -26
  239. package/dist/core/repl/cap-warning.js +4 -4
  240. package/dist/core/repl/clipboard-read.js +11 -11
  241. package/dist/core/repl/dispatch-fsm.js +12 -12
  242. package/dist/core/repl/engine-bridge.js +303 -0
  243. package/dist/core/repl/history-search.js +15 -15
  244. package/dist/core/repl/history.js +28 -18
  245. package/dist/core/repl/kill-ring.js +5 -5
  246. package/dist/core/repl/model-pricing.js +135 -0
  247. package/dist/core/repl/privacy-banner.js +22 -22
  248. package/dist/core/repl/session.js +2690 -229
  249. package/dist/core/repl/slash-commands.js +540 -41
  250. package/dist/core/repl/store/index.js +1 -1
  251. package/dist/core/repl/store/jsonl-log.js +22 -22
  252. package/dist/core/repl/store/lockfile.js +10 -10
  253. package/dist/core/repl/store/session-store.js +136 -107
  254. package/dist/core/repl/store/types.js +15 -15
  255. package/dist/core/repl/store/uuid-v7.js +12 -12
  256. package/dist/core/repl/tool-route.js +382 -0
  257. package/dist/core/repl/workspace-context.js +43 -21
  258. package/dist/core/repo-map/build.js +125 -0
  259. package/dist/core/repo-map/cache.js +185 -0
  260. package/dist/core/repo-map/extractor.js +254 -0
  261. package/dist/core/repo-map/formatter.js +145 -0
  262. package/dist/core/repo-map/page-rank.js +105 -0
  263. package/dist/core/repo-map/scanner.js +211 -0
  264. package/dist/core/retro/git-collector.js +251 -0
  265. package/dist/core/retro/health-card.js +25 -0
  266. package/dist/core/retro/metrics.js +342 -0
  267. package/dist/core/retro/narrative.js +249 -0
  268. package/dist/core/retro/plane-collector.js +274 -0
  269. package/dist/core/retro/pr-issue-link.js +65 -0
  270. package/dist/core/retro/types.js +16 -0
  271. package/dist/core/retry-budget/budget.js +284 -0
  272. package/dist/core/retry-budget/index.js +5 -0
  273. package/dist/core/retry-budget/retry-cap.js +74 -0
  274. package/dist/core/routing/lead-worker.js +43 -0
  275. package/dist/core/routing/pre-flight-estimator.js +108 -0
  276. package/dist/core/runs/run-tree.js +103 -0
  277. package/dist/core/sandboxing/adapter.js +43 -0
  278. package/dist/core/sandboxing/bubblewrap.js +209 -0
  279. package/dist/core/sandboxing/index.js +78 -0
  280. package/dist/core/sandboxing/none.js +19 -0
  281. package/dist/core/sandboxing/policy.js +97 -0
  282. package/dist/core/sandboxing/seatbelt.js +231 -0
  283. package/dist/core/security/injection-scanner.js +367 -0
  284. package/dist/core/security/output-filter.js +418 -0
  285. package/dist/core/session/env-file.js +105 -0
  286. package/dist/core/session/section-budgets.js +140 -0
  287. package/dist/core/session.js +119 -0
  288. package/dist/core/settings.js +402 -5
  289. package/dist/core/share/formatter.js +271 -0
  290. package/dist/core/share/redactor.js +221 -0
  291. package/dist/core/share/uploader.js +267 -0
  292. package/dist/core/skills/defaults.js +30 -30
  293. package/dist/core/skills/loader.js +22 -22
  294. package/dist/core/skills/sources.js +27 -27
  295. package/dist/core/smoke/headless-driver.js +174 -0
  296. package/dist/core/smoke/orchestrator.js +194 -0
  297. package/dist/core/smoke/runner.js +238 -0
  298. package/dist/core/smoke/scenario-parser.js +316 -0
  299. package/dist/core/statusline.js +99 -0
  300. package/dist/core/subagents/dispatcher-real.js +600 -0
  301. package/dist/core/subagents/dispatcher.js +146 -52
  302. package/dist/core/subagents/index.js +19 -6
  303. package/dist/core/subagents/isolation-matrix.js +213 -0
  304. package/dist/core/subagents/spawn.js +19 -4
  305. package/dist/core/telemetry/emitter.js +229 -0
  306. package/dist/core/telemetry/queue.js +251 -0
  307. package/dist/core/theme/context.js +91 -0
  308. package/dist/core/theme/presets.js +228 -0
  309. package/dist/core/theme/state.js +181 -0
  310. package/dist/core/todos/invariant.js +10 -0
  311. package/dist/core/todos/state.js +177 -0
  312. package/dist/core/tool-schema/compressor.js +89 -0
  313. package/dist/core/transport/version-interceptor.js +166 -0
  314. package/dist/core/trust.js +2 -2
  315. package/dist/core/tui/thinking-block.js +64 -0
  316. package/dist/core/vim/keymap.js +288 -0
  317. package/dist/core/vim/state.js +92 -0
  318. package/dist/core/watch-markers/marker-watcher.js +133 -0
  319. package/dist/core/worktree/include-parser.js +249 -0
  320. package/dist/core/worktree-manager/cleanup.js +123 -0
  321. package/dist/core/worktree-manager/manager.js +303 -0
  322. package/dist/index.js +36 -0
  323. package/dist/runtime/bootstrap.js +190 -0
  324. package/dist/runtime/cli.js +4403 -561
  325. package/dist/runtime/commands/agents.js +31 -31
  326. package/dist/runtime/commands/budget.js +5 -5
  327. package/dist/runtime/commands/cancel.js +231 -0
  328. package/dist/runtime/commands/chain.js +489 -0
  329. package/dist/runtime/commands/codegraph-status.js +227 -0
  330. package/dist/runtime/commands/compact.js +297 -0
  331. package/dist/runtime/commands/config.js +74 -40
  332. package/dist/runtime/commands/cost.js +199 -0
  333. package/dist/runtime/commands/delegate.js +27 -4
  334. package/dist/runtime/commands/dispatch.js +126 -0
  335. package/dist/runtime/commands/doctor.js +579 -0
  336. package/dist/runtime/commands/eval-v1.js +266 -0
  337. package/dist/runtime/commands/feedback.js +184 -0
  338. package/dist/runtime/commands/hooks.js +187 -0
  339. package/dist/runtime/commands/index-cmd.js +459 -0
  340. package/dist/runtime/commands/init.js +254 -0
  341. package/dist/runtime/commands/lsp.js +200 -38
  342. package/dist/runtime/commands/mcp.js +935 -0
  343. package/dist/runtime/commands/memory.js +582 -0
  344. package/dist/runtime/commands/model.js +237 -0
  345. package/dist/runtime/commands/onboarding.js +275 -0
  346. package/dist/runtime/commands/patch.js +12 -12
  347. package/dist/runtime/commands/permissions.js +112 -0
  348. package/dist/runtime/commands/plan.js +143 -0
  349. package/dist/runtime/commands/prd-check.js +285 -0
  350. package/dist/runtime/commands/privacy.js +17 -17
  351. package/dist/runtime/commands/recipe.js +325 -0
  352. package/dist/runtime/commands/redo-blob-store.js +92 -0
  353. package/dist/runtime/commands/redo.js +361 -0
  354. package/dist/runtime/commands/release-notes.js +229 -0
  355. package/dist/runtime/commands/repo-map.js +95 -0
  356. package/dist/runtime/commands/report.js +299 -0
  357. package/dist/runtime/commands/resume.js +118 -0
  358. package/dist/runtime/commands/review-consensus.js +68 -53
  359. package/dist/runtime/commands/rewind.js +333 -0
  360. package/dist/runtime/commands/roster.js +14 -14
  361. package/dist/runtime/commands/servers-cli.js +182 -0
  362. package/dist/runtime/commands/servers.js +236 -0
  363. package/dist/runtime/commands/sessions.js +163 -0
  364. package/dist/runtime/commands/share.js +316 -0
  365. package/dist/runtime/commands/skills.js +31 -31
  366. package/dist/runtime/commands/status.js +186 -0
  367. package/dist/runtime/commands/stickers.js +82 -0
  368. package/dist/runtime/commands/style.js +194 -0
  369. package/dist/runtime/commands/theme.js +196 -0
  370. package/dist/runtime/commands/undo.js +54 -22
  371. package/dist/runtime/commands/update.js +289 -0
  372. package/dist/runtime/commands/vim.js +140 -0
  373. package/dist/runtime/commands/worktree.js +8 -8
  374. package/dist/runtime/commands/worktrees.js +155 -0
  375. package/dist/runtime/deprecation-warning.js +69 -0
  376. package/dist/runtime/engine-exit-code.js +50 -0
  377. package/dist/runtime/headless-repl.js +195 -0
  378. package/dist/runtime/headless.js +548 -0
  379. package/dist/runtime/load-hooks-or-exit.js +71 -0
  380. package/dist/runtime/plan-decompose.js +22 -22
  381. package/dist/runtime/sigint-guard.js +272 -0
  382. package/dist/runtime/stream-renderer.js +195 -0
  383. package/dist/runtime/update-check.js +28 -28
  384. package/dist/runtime/version.js +65 -0
  385. package/dist/runtime/worktree-bootstrap.js +579 -0
  386. package/dist/skills/bundled/batch.js +617 -0
  387. package/dist/skills/bundled/index.js +45 -0
  388. package/dist/skills/bundled/loop.js +358 -0
  389. package/dist/skills/bundled/remember.js +383 -0
  390. package/dist/skills/bundled/simplify.js +289 -0
  391. package/dist/skills/bundled/skillify.js +373 -0
  392. package/dist/skills/bundled/stuck.js +558 -0
  393. package/dist/skills/bundled/verify.js +439 -0
  394. package/dist/testing/vcr.js +486 -0
  395. package/dist/tools/agent-tool.js +229 -0
  396. package/dist/tools/apply-patch.js +89 -28
  397. package/dist/tools/ask-user-question.js +337 -0
  398. package/dist/tools/ask-user.js +115 -0
  399. package/dist/tools/bash.js +811 -49
  400. package/dist/tools/brief.js +224 -0
  401. package/dist/tools/cron.js +433 -0
  402. package/dist/tools/enter-worktree.js +250 -0
  403. package/dist/tools/exit-worktree.js +147 -0
  404. package/dist/tools/file-tools.js +161 -44
  405. package/dist/tools/http-request.js +336 -0
  406. package/dist/tools/lsp-tools.js +377 -1
  407. package/dist/tools/mcp-tool.js +260 -0
  408. package/dist/tools/multi-edit.js +361 -0
  409. package/dist/tools/powershell.js +268 -0
  410. package/dist/tools/registry.js +120 -5
  411. package/dist/tools/server-tools.js +892 -0
  412. package/dist/tools/skill-tool.js +96 -0
  413. package/dist/tools/sleep.js +99 -0
  414. package/dist/tools/synthetic-output.js +133 -0
  415. package/dist/tools/tasks.js +208 -0
  416. package/dist/tools/todo-write.js +184 -0
  417. package/dist/tools/verify-plan-execution.js +295 -0
  418. package/dist/tools/web-fetch-injection-scanner.js +207 -0
  419. package/dist/tools/web-fetch.js +195 -10
  420. package/dist/tools/web-search.js +458 -0
  421. package/dist/tui/agent-progress-card.js +111 -0
  422. package/dist/tui/agent-tree.js +22 -1
  423. package/dist/tui/ask-modal.js +14 -14
  424. package/dist/tui/ask-user-question-chips.js +315 -0
  425. package/dist/tui/ask-user-question-prompt.js +203 -0
  426. package/dist/tui/compact-banner.js +81 -0
  427. package/dist/tui/conversation-pane.js +85 -11
  428. package/dist/tui/cost-table.js +111 -0
  429. package/dist/tui/device-flow.js +2 -2
  430. package/dist/tui/doctor-table.js +46 -0
  431. package/dist/tui/feedback-prompt.js +156 -0
  432. package/dist/tui/input-box.js +247 -32
  433. package/dist/tui/login-picker.js +3 -3
  434. package/dist/tui/markdown-render.js +6 -6
  435. package/dist/tui/multi-file-diff-approval.js +375 -0
  436. package/dist/tui/onboarding-wizard.js +240 -0
  437. package/dist/tui/permissions-picker.js +86 -0
  438. package/dist/tui/render.js +36 -1
  439. package/dist/tui/repl-render.js +239 -25
  440. package/dist/tui/repl-splash-art.js +16 -16
  441. package/dist/tui/repl-splash-mascot.js +48 -24
  442. package/dist/tui/repl-splash.js +22 -22
  443. package/dist/tui/repl.js +125 -45
  444. package/dist/tui/slash-palette.js +6 -6
  445. package/dist/tui/splash.js +2 -2
  446. package/dist/tui/status-bar.js +109 -31
  447. package/dist/tui/status-table.js +7 -0
  448. package/dist/tui/stickers-art.js +136 -0
  449. package/dist/tui/style-table.js +28 -0
  450. package/dist/tui/theme-table.js +29 -0
  451. package/dist/tui/thinking-spinner.js +123 -0
  452. package/dist/tui/tool-stream-pane.js +53 -4
  453. package/dist/tui/update-banner.js +27 -2
  454. package/dist/tui/vim-input.js +267 -0
  455. package/dist/tui/welcome-banner.js +107 -0
  456. package/dist/tui/welcome-data.js +293 -0
  457. package/dist/tui/workspace-context.js +2 -2
  458. package/package.json +29 -6
  459. package/test/scenarios/codegen-create-file.scenario.txt +13 -0
  460. package/test/scenarios/compact-force.scenario.txt +12 -0
  461. package/test/scenarios/identity.scenario.txt +11 -0
  462. package/test/scenarios/persona-handoff.scenario.txt +12 -0
  463. package/test/scenarios/walkback.scenario.txt +12 -0
  464. package/dist/core/engine/compaction-hook.js +0 -154
@@ -0,0 +1,266 @@
1
+ /**
2
+ * `pugi eval-v1` - run the frozen 20-task benchmark harness (#120).
3
+ *
4
+ * Surface:
5
+ *
6
+ * pugi eval-v1 --list # list task ids + briefs
7
+ * pugi eval-v1 --task 01-snake-html # run one task
8
+ * pugi eval-v1 --task 01,02,03 # run a subset
9
+ * pugi eval-v1 --all # run the full 20
10
+ * pugi eval-v1 --all --model qwen3-coder # pin the engine model
11
+ * pugi eval-v1 --all --json # JSON envelope output
12
+ * pugi eval-v1 --all --no-ledger # skip results.tsv append
13
+ *
14
+ * The handler stays thin: it resolves the tasks directory + pugi
15
+ * binary, calls `runHarness`, appends every result к the ledger
16
+ * (unless `--no-ledger`), and renders either a text table or a JSON
17
+ * envelope.
18
+ *
19
+ * Aggregate scoring (mean across all per-task scores) drives the CI
20
+ * regression gate in `.github/workflows/eval-v1.yml`. The gate is
21
+ * "aggregatePugiScore must not drop more than 10% vs the last week
22
+ * median" - the workflow computes the median itself from the
23
+ * committed ledger; the CLI's only job is к produce a fresh row.
24
+ */
25
+ import { execFileSync } from 'node:child_process';
26
+ import { existsSync } from 'node:fs';
27
+ import { dirname, resolve } from 'node:path';
28
+ import { fileURLToPath } from 'node:url';
29
+ import { defaultLedgerPath, defaultTasksDir, loadAllTasks, } from '../../core/eval/v1/task-loader.js';
30
+ import { aggregateScore } from '../../core/eval/v1/scoring.js';
31
+ import { appendLedgerRows } from '../../core/eval/v1/ledger.js';
32
+ import { runHarness } from '../../core/eval/v1/runner.js';
33
+ function parseFlags(args) {
34
+ const out = {
35
+ list: false,
36
+ all: false,
37
+ task: [],
38
+ model: undefined,
39
+ ledger: true,
40
+ tasksDir: undefined,
41
+ ledgerPath: undefined,
42
+ help: false,
43
+ };
44
+ for (let i = 0; i < args.length; i += 1) {
45
+ const arg = args[i] ?? '';
46
+ if (arg === '--list')
47
+ out.list = true;
48
+ else if (arg === '--all')
49
+ out.all = true;
50
+ else if (arg === '--no-ledger')
51
+ out.ledger = false;
52
+ else if (arg === '--help' || arg === '-h')
53
+ out.help = true;
54
+ else if (arg === '--task') {
55
+ const next = args[i + 1];
56
+ if (!next || next.startsWith('--')) {
57
+ throw new Error('--task requires a comma-separated list of ids');
58
+ }
59
+ out.task.push(...next.split(',').map((s) => s.trim()).filter(Boolean));
60
+ i += 1;
61
+ }
62
+ else if (arg.startsWith('--task=')) {
63
+ out.task.push(...arg.slice('--task='.length).split(',').map((s) => s.trim()).filter(Boolean));
64
+ }
65
+ else if (arg === '--model') {
66
+ const next = args[i + 1];
67
+ if (!next)
68
+ throw new Error('--model requires a value');
69
+ out.model = next;
70
+ i += 1;
71
+ }
72
+ else if (arg.startsWith('--model=')) {
73
+ out.model = arg.slice('--model='.length);
74
+ }
75
+ else if (arg === '--tasks-dir') {
76
+ const next = args[i + 1];
77
+ if (!next)
78
+ throw new Error('--tasks-dir requires a path');
79
+ out.tasksDir = next;
80
+ i += 1;
81
+ }
82
+ else if (arg.startsWith('--tasks-dir=')) {
83
+ out.tasksDir = arg.slice('--tasks-dir='.length);
84
+ }
85
+ else if (arg === '--ledger') {
86
+ const next = args[i + 1];
87
+ if (!next)
88
+ throw new Error('--ledger requires a path');
89
+ out.ledgerPath = next;
90
+ i += 1;
91
+ }
92
+ else if (arg.startsWith('--ledger=')) {
93
+ out.ledgerPath = arg.slice('--ledger='.length);
94
+ }
95
+ else if (arg === '--json') {
96
+ // accepted at top-level; ignore here
97
+ }
98
+ else {
99
+ throw new Error(`unknown eval-v1 flag: ${arg}`);
100
+ }
101
+ }
102
+ return out;
103
+ }
104
+ function resolvePackageRoot(override) {
105
+ if (override)
106
+ return resolve(override);
107
+ // The compiled file lives at `dist/runtime/commands/eval-v1.js`;
108
+ // climb three directories to reach the package root in both dev
109
+ // (`src/runtime/commands/eval-v1.ts`) and dist builds.
110
+ const here = dirname(fileURLToPath(import.meta.url));
111
+ return resolve(here, '..', '..', '..');
112
+ }
113
+ function resolvePugiBin(override, packageRoot) {
114
+ if (override)
115
+ return resolve(override);
116
+ const local = resolve(packageRoot, 'bin', 'run.js');
117
+ if (existsSync(local))
118
+ return local;
119
+ // Fall back к PATH lookup; spawn will pick it up.
120
+ return 'pugi';
121
+ }
122
+ function captureGitSha(packageRoot) {
123
+ try {
124
+ const out = execFileSync('git', ['rev-parse', '--short', 'HEAD'], {
125
+ cwd: packageRoot,
126
+ encoding: 'utf8',
127
+ stdio: ['ignore', 'pipe', 'ignore'],
128
+ });
129
+ const trimmed = out.trim();
130
+ return trimmed === '' ? '(unknown)' : trimmed;
131
+ }
132
+ catch {
133
+ return '(unknown)';
134
+ }
135
+ }
136
+ function renderTable(report) {
137
+ const headers = ['id', 'difficulty', 'status', 'score', 'tokens', 'wall_ms', 'verifs'];
138
+ const rows = report.results.map((r) => [
139
+ r.taskId,
140
+ findDifficulty(r.taskId, report) ?? '?',
141
+ r.status,
142
+ r.pugiScore.toFixed(2),
143
+ String(r.tokensUsed),
144
+ String(r.wallClockMs),
145
+ `${r.verifications.filter((v) => v.passed).length}/${r.verifications.length}`,
146
+ ]);
147
+ const widths = headers.map((h, i) => Math.max(h.length, ...rows.map((r) => (r[i] ?? '').length)));
148
+ const renderRow = (cells) => cells.map((c, i) => c.padEnd(widths[i] ?? 0)).join(' ');
149
+ const lines = [renderRow(headers), renderRow(widths.map((w) => '-'.repeat(w)))];
150
+ for (const row of rows)
151
+ lines.push(renderRow(row));
152
+ lines.push('');
153
+ lines.push(`aggregate pugi_score: ${report.aggregatePugiScore.toFixed(2)} (mean of ${report.results.length})`);
154
+ lines.push(`model: ${report.model}`);
155
+ lines.push(`git_sha: ${report.gitSha}`);
156
+ return lines.join('\n');
157
+ }
158
+ const difficultyMap = new WeakMap();
159
+ function findDifficulty(taskId, report) {
160
+ const cached = difficultyMap.get(report);
161
+ return cached?.get(taskId);
162
+ }
163
+ function indexDifficulty(report, specs) {
164
+ const map = new Map();
165
+ for (const s of specs)
166
+ map.set(s.id, s.difficulty);
167
+ difficultyMap.set(report, map);
168
+ }
169
+ export async function runEvalV1Command(ctx) {
170
+ let flags;
171
+ try {
172
+ flags = parseFlags(ctx.args);
173
+ }
174
+ catch (err) {
175
+ ctx.writeOutput({ command: 'eval-v1', status: 'invalid_args', reason: err.message }, `pugi eval-v1: ${err.message}`);
176
+ return 2;
177
+ }
178
+ if (flags.help) {
179
+ const help = [
180
+ 'pugi eval-v1 - frozen 20-task benchmark harness (#120)',
181
+ '',
182
+ 'usage:',
183
+ ' pugi eval-v1 --list',
184
+ ' pugi eval-v1 --task <id[,id...]>',
185
+ ' pugi eval-v1 --all [--model <slug>] [--no-ledger] [--json]',
186
+ ].join('\n');
187
+ ctx.writeOutput({ command: 'eval-v1', status: 'help', help }, help);
188
+ return 0;
189
+ }
190
+ const packageRoot = resolvePackageRoot(ctx.packageRoot);
191
+ const tasksDir = flags.tasksDir ?? defaultTasksDir(packageRoot);
192
+ const ledgerPath = flags.ledgerPath ?? defaultLedgerPath(packageRoot);
193
+ const log = ctx.log ?? ((line) => process.stderr.write(`${line}\n`));
194
+ let loaded;
195
+ try {
196
+ loaded = loadAllTasks(tasksDir);
197
+ }
198
+ catch (err) {
199
+ ctx.writeOutput({
200
+ command: 'eval-v1',
201
+ status: 'tasks_load_failed',
202
+ reason: err.message,
203
+ }, `pugi eval-v1: ${err.message}`);
204
+ return 2;
205
+ }
206
+ const specs = loaded.map((l) => l.spec);
207
+ if (flags.list) {
208
+ const indexLines = specs.map((s) => `${s.id} [${s.command}/${s.difficulty}] ${s.brief}`);
209
+ const out = indexLines.join('\n');
210
+ ctx.writeOutput({ command: 'eval-v1', status: 'listed', tasks: specs.map((s) => ({
211
+ id: s.id, command: s.command, difficulty: s.difficulty, brief: s.brief,
212
+ })) }, out);
213
+ return 0;
214
+ }
215
+ if (!flags.all && flags.task.length === 0) {
216
+ const reason = 'pass --all OR --task <id[,id...]> OR --list';
217
+ ctx.writeOutput({ command: 'eval-v1', status: 'invalid_args', reason }, `pugi eval-v1: ${reason}`);
218
+ return 2;
219
+ }
220
+ const only = flags.all ? undefined : flags.task;
221
+ if (only) {
222
+ const known = new Set(specs.map((s) => s.id));
223
+ const unknown = only.filter((id) => !known.has(id));
224
+ if (unknown.length) {
225
+ const reason = `unknown task id(s): ${unknown.join(', ')}`;
226
+ ctx.writeOutput({ command: 'eval-v1', status: 'invalid_args', reason }, `pugi eval-v1: ${reason}`);
227
+ return 2;
228
+ }
229
+ }
230
+ const pugiBin = resolvePugiBin(ctx.pugiBin, packageRoot);
231
+ const startedAt = new Date().toISOString();
232
+ const gitSha = captureGitSha(packageRoot);
233
+ const results = await runHarness({
234
+ specs,
235
+ options: {
236
+ pugiBin,
237
+ ...(flags.model !== undefined ? { model: flags.model } : {}),
238
+ ...(only !== undefined ? { only } : {}),
239
+ ...(ctx.runner ? { runner: ctx.runner } : {}),
240
+ ...(ctx.env !== undefined ? { env: ctx.env } : {}),
241
+ onTaskStart: (s) => log(`[eval-v1] ${s.id} starting (${s.command}/${s.difficulty})`),
242
+ onTaskFinish: (r) => log(`[eval-v1] ${r.taskId} ${r.status} score=${r.pugiScore.toFixed(2)} tokens=${r.tokensUsed} wall=${r.wallClockMs}ms`),
243
+ },
244
+ });
245
+ const report = {
246
+ schemaVersion: 1,
247
+ startedAt,
248
+ model: flags.model ?? '(default)',
249
+ results,
250
+ aggregatePugiScore: aggregateScore(results),
251
+ gitSha,
252
+ };
253
+ indexDifficulty(report, specs);
254
+ if (flags.ledger && results.length > 0) {
255
+ appendLedgerRows(ledgerPath, results.map((r) => ({
256
+ timestamp: startedAt,
257
+ gitSha,
258
+ model: report.model,
259
+ result: r,
260
+ })));
261
+ }
262
+ ctx.writeOutput(report, renderTable(report));
263
+ const anyFail = results.some((r) => r.status !== 'pass');
264
+ return anyFail ? 1 : 0;
265
+ }
266
+ //# sourceMappingURL=eval-v1.js.map
@@ -0,0 +1,184 @@
1
+ /**
2
+ * `pugi feedback` + `/feedback` slash — .
3
+ *
4
+ * In-CLI feedback collector. Parity with the upstream tool's `/feedback`
5
+ * built-in. The operator never has to leave the terminal to file a
6
+ * bug / feature / general comment / praise. The wizard collects:
7
+ *
8
+ * 1. category (bug / feature / general / praise)
9
+ * 2. rating (1-5)
10
+ * 3. comment (multi-line free text)
11
+ * 4. optional redacted session context (last 5 turns)
12
+ * 5. confirm (final y/n)
13
+ *
14
+ * # Module contract
15
+ *
16
+ * - This file owns the WIRING from the CLI surface (TTY mount,
17
+ * non-TTY JSON, slash dispatcher) to the queue + submitter
18
+ * modules. The corpus + redactor + queue persistence live in
19
+ * `core/feedback/{queue.ts,submitter.ts}`. The Ink prompt lives
20
+ * in `tui/feedback-prompt.tsx`. Both have zero coupling to the
21
+ * CLI dispatch surface.
22
+ *
23
+ * - `runFeedbackCommand` is the single entry point. Both the top-
24
+ * level `pugi feedback` handler в `runtime/cli.ts` AND the in-REPL
25
+ * `/feedback` slash dispatcher call it. The function returns the
26
+ * resolved `FeedbackRunResult` so the slash dispatcher can route
27
+ * the outcome message to the REPL's system pane without re-prompting.
28
+ *
29
+ * - Exit code is ALWAYS 0. Feedback is a brand surface — never a
30
+ * gate. Failures land as result variants; the wrapper never
31
+ * turns a network blip into a non-zero shell exit.
32
+ *
33
+ * - The random-source-style test seam: the run helper accepts an
34
+ * `interactive` flag that the spec sets to false, plus an injected
35
+ * `draft` so unit tests can drive the submit + queue branches
36
+ * without mounting Ink.
37
+ */
38
+ import { existsSync, readFileSync } from 'node:fs';
39
+ import { resolve } from 'node:path';
40
+ import { enqueueFeedback, feedbackQueuePath, flushFeedbackQueue, } from '../../core/feedback/queue.js';
41
+ import { feedbackSubmitUrl, redactSessionContext, submitFeedback, } from '../../core/feedback/submitter.js';
42
+ /**
43
+ * Drive one feedback round. The function is async because of the
44
+ * submit round-trip, but everything else (queue write, redaction) is
45
+ * sync — no surprise concurrency.
46
+ */
47
+ export async function runFeedbackCommand(ctx) {
48
+ if (ctx.draft == null) {
49
+ return { kind: 'cancelled' };
50
+ }
51
+ const envelope = {
52
+ category: ctx.draft.category,
53
+ rating: ctx.draft.rating,
54
+ comment: ctx.draft.comment,
55
+ ts: new Date().toISOString(),
56
+ cliVersion: ctx.cliVersion,
57
+ ...(ctx.tier ? { tier: ctx.tier } : {}),
58
+ };
59
+ if (ctx.draft.includeSessionContext && ctx.sessionContext) {
60
+ const sc = ctx.sessionContext();
61
+ if (sc)
62
+ envelope.sessionContext = sc;
63
+ }
64
+ let result;
65
+ try {
66
+ result = await ctx.submit(envelope);
67
+ }
68
+ catch (err) {
69
+ // Defensive: a thrown submitter (should not happen — the live
70
+ // submitter catches everything) is treated as transient so the
71
+ // envelope lands in the queue.
72
+ const reason = err instanceof Error ? err.message : String(err);
73
+ result = { kind: 'transient', reason };
74
+ }
75
+ if (result.kind === 'ok') {
76
+ return { kind: 'submitted', envelope, httpStatus: result.httpStatus };
77
+ }
78
+ if (result.kind === 'transient') {
79
+ const path = enqueueFeedback(envelope, ctx.cwd);
80
+ return { kind: 'queued', envelope, path, reason: result.reason };
81
+ }
82
+ // permanent — log + drop
83
+ return {
84
+ kind: 'dropped',
85
+ envelope,
86
+ reason: result.reason,
87
+ httpStatus: result.httpStatus,
88
+ };
89
+ }
90
+ /**
91
+ * Render one human-readable toast for the operator. Centralised so the
92
+ * top-level `pugi feedback` shell handler + the in-REPL `/feedback`
93
+ * slash dispatcher agree on the copy.
94
+ */
95
+ export function renderFeedbackToast(result) {
96
+ switch (result.kind) {
97
+ case 'submitted':
98
+ return 'Feedback submitted. Thank you.';
99
+ case 'queued':
100
+ return `Feedback queued locally. Will sync on next online run. (${result.path})`;
101
+ case 'cancelled':
102
+ return 'Feedback cancelled. Nothing was sent.';
103
+ case 'dropped':
104
+ return `Feedback rejected by server (${result.httpStatus}): ${result.reason}. Not queued.`;
105
+ case 'noop':
106
+ return `No feedback collected: ${result.reason}`;
107
+ }
108
+ }
109
+ /**
110
+ * Background queue flush. Invoked silently on session start so any
111
+ * envelopes that landed during an offline run get drained when the
112
+ * operator next has connectivity. The function never throws — it
113
+ * returns the flush stats so the caller can log them at debug level.
114
+ */
115
+ export async function flushFeedbackQueueSilently(cwd, config) {
116
+ // Short-circuit when the queue file does not exist. Avoids a
117
+ // pointless `fs.stat` round-trip on every cold session start.
118
+ if (!existsSync(feedbackQueuePath(cwd))) {
119
+ return { attempted: 0, succeeded: 0, failed: 0 };
120
+ }
121
+ const result = await flushFeedbackQueue(cwd, async (env) => {
122
+ const r = await submitFeedback(env, config);
123
+ if (r.kind === 'ok')
124
+ return true;
125
+ if (r.kind === 'permanent') {
126
+ // Permanent failures are "done" from the queue's POV — they
127
+ // would never resolve on retry. Drop them so the queue does
128
+ // not grow without bound.
129
+ return true;
130
+ }
131
+ return false;
132
+ });
133
+ return {
134
+ attempted: result.attempted,
135
+ succeeded: result.succeeded,
136
+ failed: result.failed,
137
+ };
138
+ }
139
+ /**
140
+ * Re-exports — the spec imports these via the command module so the
141
+ * dependency graph in the test stays single-rooted.
142
+ */
143
+ export { feedbackQueuePath, feedbackSubmitUrl, redactSessionContext, submitFeedback, };
144
+ /**
145
+ * Read the persona conversation log if present. Best-effort: returns
146
+ * an empty list when the file is missing or malformed. The CLI's REPL
147
+ * persists transcripts via the session module at a canonical relative
148
+ * path under `.pugi/sessions/`. The shell-level `pugi feedback` does
149
+ * not have access to a live session, so it tries to pick up the most
150
+ * recent persisted one for the `--with-context` path.
151
+ *
152
+ * Intentionally tolerant — feedback works even with no transcript.
153
+ */
154
+ export function readMostRecentTranscript(cwd, options = {}) {
155
+ // The CLI may persist sessions in several places depending on the
156
+ // surface. We probe the conventional default; the spec drives the
157
+ // function via a fixture file instead of a live REPL.
158
+ const candidate = resolve(cwd, '.pugi', 'sessions', 'latest.jsonl');
159
+ if (!existsSync(candidate))
160
+ return [];
161
+ try {
162
+ const text = readFileSync(candidate, 'utf8');
163
+ const lines = text.split('\n').filter((l) => l.trim().length > 0);
164
+ const turns = [];
165
+ for (const line of lines) {
166
+ try {
167
+ const obj = JSON.parse(line);
168
+ if ((obj.role === 'user' || obj.role === 'assistant' || obj.role === 'system')
169
+ && typeof obj.text === 'string') {
170
+ turns.push({ role: obj.role, text: obj.text });
171
+ }
172
+ }
173
+ catch {
174
+ // skip malformed line
175
+ }
176
+ }
177
+ const cap = options.maxTurns ?? 5;
178
+ return turns.slice(-cap);
179
+ }
180
+ catch {
181
+ return [];
182
+ }
183
+ }
184
+ //# sourceMappingURL=feedback.js.map
@@ -0,0 +1,187 @@
1
+ /**
2
+ * `pugi hooks` — operator surface for user-config hooks (MVP).
3
+ *
4
+ * Two subcommands ship in the MVP:
5
+ *
6
+ * pugi hooks list List configured hooks per event.
7
+ * pugi hooks doctor Validate the config and surface any
8
+ * parse / schema errors.
9
+ *
10
+ * Both accept `--json` for scripted callers. Argument grammar is
11
+ * intentionally narrow — no `add` / `remove` / `test` subcommands in
12
+ * the MVP. Operators hand-edit `~/.pugi/hooks-mvp.json` for now.
13
+ *
14
+ * Exit codes:
15
+ * 0 -> happy path (no hooks OR config valid).
16
+ * 1 -> config present but invalid (only `doctor` returns this).
17
+ * 2 -> unknown subcommand / argument error.
18
+ *
19
+ * Brand voice: ASCII only.
20
+ */
21
+ import { ALL_HOOK_EVENTS_V2, defaultHooksMvpPath, loadHooksConfig, } from '../../core/hooks/index.js';
22
+ function parseFlags(args) {
23
+ const rest = [];
24
+ const flags = { json: false };
25
+ for (const arg of args) {
26
+ if (arg === '--json') {
27
+ flags.json = true;
28
+ continue;
29
+ }
30
+ rest.push(arg);
31
+ }
32
+ return { rest, flags };
33
+ }
34
+ /**
35
+ * Top-level dispatcher for `pugi hooks <subcommand>`. Returns the
36
+ * intended process exit code. `cli.ts` is expected to set
37
+ * `process.exitCode = <return value>` so error states propagate to
38
+ * scripted callers without throwing.
39
+ */
40
+ export async function runHooksCommand(args, ctx) {
41
+ const { rest, flags } = parseFlags(args);
42
+ const sub = rest[0];
43
+ if (!sub || sub === 'help' || sub === '--help') {
44
+ emitUsage(ctx, flags);
45
+ return sub ? 0 : 2;
46
+ }
47
+ if (sub === 'list') {
48
+ return runList(ctx, flags);
49
+ }
50
+ if (sub === 'doctor') {
51
+ return runDoctor(ctx, flags);
52
+ }
53
+ ctx.writeOutput({ ok: false, error: `unknown subcommand: ${sub}` }, `pugi hooks: unknown subcommand '${sub}'. Try 'pugi hooks --help'.`);
54
+ return 2;
55
+ }
56
+ function emitUsage(ctx, flags) {
57
+ const text = [
58
+ 'pugi hooks — user-config lifecycle hooks (MVP).',
59
+ '',
60
+ 'Subcommands:',
61
+ ' pugi hooks list Show hooks configured per event.',
62
+ ' pugi hooks doctor Validate ~/.pugi/hooks-mvp.json.',
63
+ '',
64
+ 'Flags:',
65
+ ' --json Emit a JSON envelope instead of human text.',
66
+ '',
67
+ 'Config file:',
68
+ ' ~/.pugi/hooks-mvp.json',
69
+ '',
70
+ 'Status:',
71
+ ' MVP — 2 events out of 8. Remaining events (PostToolUse,',
72
+ " UserPromptSubmit, Stop, SubagentStop, PreCompact, Notification)",
73
+ ' deferred to fast-follow PR.',
74
+ ].join('\n');
75
+ ctx.writeOutput({
76
+ ok: true,
77
+ command: 'hooks',
78
+ usage: text,
79
+ }, text);
80
+ if (flags.json) {
81
+ // The structured payload is already emitted by writeOutput when
82
+ // --json is on; nothing extra to do.
83
+ }
84
+ }
85
+ function runList(ctx, flags) {
86
+ let config;
87
+ try {
88
+ config = loadHooksConfig(ctx.configPath);
89
+ }
90
+ catch (error) {
91
+ const msg = error.message;
92
+ ctx.writeOutput({ ok: false, error: msg }, `pugi hooks list: ${msg}\nFix the config or remove the file. Run 'pugi hooks doctor' for details.`);
93
+ return 1;
94
+ }
95
+ const perEvent = {
96
+ SessionStart: [],
97
+ PreToolUse: [],
98
+ PostToolUse: [],
99
+ UserPromptSubmit: [],
100
+ Stop: [],
101
+ SubagentStop: [],
102
+ PreCompact: [],
103
+ Notification: [],
104
+ // PUGI-487 - worktree lifecycle events.
105
+ WorktreeCreate: [],
106
+ WorktreeRemove: [],
107
+ };
108
+ for (const event of ALL_HOOK_EVENTS_V2) {
109
+ perEvent[event] = config.list(event).map((entry) => ({
110
+ matcher: entry.matcher,
111
+ command: entry.command,
112
+ timeoutMs: entry.timeoutMs,
113
+ blocking: entry.blocking,
114
+ }));
115
+ }
116
+ const total = Object.values(perEvent).reduce((acc, list) => acc + list.length, 0);
117
+ const payload = {
118
+ ok: true,
119
+ configPath: config.configPath(),
120
+ total,
121
+ perEvent,
122
+ };
123
+ if (flags.json) {
124
+ ctx.writeOutput(payload, JSON.stringify(payload, null, 2));
125
+ return 0;
126
+ }
127
+ const lines = [];
128
+ lines.push(`pugi hooks (${total} configured)`);
129
+ lines.push(` config: ${config.configPath()}`);
130
+ if (total === 0) {
131
+ lines.push(' no hooks configured — create the file above to add one.');
132
+ }
133
+ else {
134
+ for (const event of ALL_HOOK_EVENTS_V2) {
135
+ const list = perEvent[event];
136
+ if (list.length === 0)
137
+ continue;
138
+ lines.push(` ${event}:`);
139
+ for (const entry of list) {
140
+ const tags = [];
141
+ if (entry.matcher)
142
+ tags.push(`matcher=${entry.matcher}`);
143
+ if (entry.timeoutMs)
144
+ tags.push(`timeoutMs=${entry.timeoutMs}`);
145
+ if (entry.blocking)
146
+ tags.push('blocking');
147
+ const suffix = tags.length ? ` [${tags.join(', ')}]` : '';
148
+ lines.push(` - ${entry.command}${suffix}`);
149
+ }
150
+ }
151
+ }
152
+ const text = lines.join('\n');
153
+ ctx.writeOutput(payload, text);
154
+ return 0;
155
+ }
156
+ function runDoctor(ctx, flags) {
157
+ const path = ctx.configPath ?? defaultHooksMvpPath();
158
+ try {
159
+ const config = loadHooksConfig(ctx.configPath);
160
+ const total = config.flatten().length;
161
+ const payload = {
162
+ ok: true,
163
+ configPath: config.configPath(),
164
+ total,
165
+ issues: [],
166
+ };
167
+ const text = total
168
+ ? `pugi hooks doctor: ${path} OK (${total} hooks).`
169
+ : `pugi hooks doctor: ${path} not present (no hooks configured).`;
170
+ ctx.writeOutput(payload, text);
171
+ return 0;
172
+ }
173
+ catch (error) {
174
+ const msg = error.message;
175
+ const payload = {
176
+ ok: false,
177
+ configPath: path,
178
+ error: msg,
179
+ };
180
+ const text = `pugi hooks doctor: ${msg}`;
181
+ ctx.writeOutput(payload, text);
182
+ return 1;
183
+ }
184
+ // flags.json is consumed by writeOutput in the host shell.
185
+ void flags;
186
+ }
187
+ //# sourceMappingURL=hooks.js.map