@pugi/cli 0.1.0-beta.9 → 0.1.0-beta.90

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 (409) hide show
  1. package/CHANGELOG.md +132 -0
  2. package/LICENSE +1 -1
  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 +3 -3
  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 +13 -13
  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 +333 -7
  94. package/dist/core/edits/format-detector.js +260 -0
  95. package/dist/core/edits/format-matrix.js +26 -0
  96. package/dist/core/edits/fuzzy-ladder.js +650 -0
  97. package/dist/core/edits/index.js +5 -1
  98. package/dist/core/edits/journal.js +199 -0
  99. package/dist/core/edits/layer-a-apply.js +15 -15
  100. package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
  101. package/dist/core/edits/layer-b-apply.js +9 -9
  102. package/dist/core/edits/layer-c-apply.js +6 -6
  103. package/dist/core/edits/layer-d-ast.js +557 -14
  104. package/dist/core/edits/marker-parser.js +12 -12
  105. package/dist/core/edits/security-gate.js +27 -27
  106. package/dist/core/edits/verify-hook.js +273 -0
  107. package/dist/core/edits/worktree.js +29 -29
  108. package/dist/core/engine/anvil-client.js +214 -26
  109. package/dist/core/engine/auto-compact.js +179 -0
  110. package/dist/core/engine/budgets.js +186 -0
  111. package/dist/core/engine/context-prefix.js +155 -0
  112. package/dist/core/engine/index.js +1 -1
  113. package/dist/core/engine/intensity.js +158 -0
  114. package/dist/core/engine/intent.js +260 -0
  115. package/dist/core/engine/native-pugi.js +1295 -227
  116. package/dist/core/engine/prompts.js +129 -19
  117. package/dist/core/engine/strip-internal-fields.js +124 -0
  118. package/dist/core/engine/tool-bridge.js +1731 -59
  119. package/dist/core/evaluation/golden-dataset.js +293 -0
  120. package/dist/core/feedback/queue.js +177 -0
  121. package/dist/core/feedback/submitter.js +145 -0
  122. package/dist/core/file-cache.js +113 -1
  123. package/dist/core/flatten/flatten-repo.js +439 -0
  124. package/dist/core/format/osc8-link.js +28 -0
  125. package/dist/core/hook-chains.js +392 -0
  126. package/dist/core/hooks/citation-verify-hook.js +138 -0
  127. package/dist/core/hooks/citation-verify.js +112 -0
  128. package/dist/core/hooks/events.js +46 -0
  129. package/dist/core/hooks/index.js +15 -0
  130. package/dist/core/hooks/registry.js +216 -0
  131. package/dist/core/hooks/runner.js +236 -0
  132. package/dist/core/hooks/v2/event-emitter.js +115 -0
  133. package/dist/core/hooks/v2/executor.js +282 -0
  134. package/dist/core/hooks/v2/index.js +25 -0
  135. package/dist/core/hooks/v2/lifecycle.js +104 -0
  136. package/dist/core/hooks/v2/loader.js +216 -0
  137. package/dist/core/hooks/v2/matcher.js +125 -0
  138. package/dist/core/hooks/v2/trust.js +143 -0
  139. package/dist/core/hooks/v2/types.js +86 -0
  140. package/dist/core/hooks/worktree-events.js +158 -0
  141. package/dist/core/image/renderer.js +71 -0
  142. package/dist/core/init/detector.js +582 -0
  143. package/dist/core/init/template-renderer.js +242 -0
  144. package/dist/core/jobs/registry.js +18 -18
  145. package/dist/core/ledger/results-tsv.js +142 -0
  146. package/dist/core/log-discipline/stdout-redirect.js +51 -0
  147. package/dist/core/lsp/cache.js +105 -0
  148. package/dist/core/lsp/client.js +551 -41
  149. package/dist/core/lsp/language-detect.js +66 -0
  150. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  151. package/dist/core/lsp/server-detect.js +173 -0
  152. package/dist/core/lsp/symbol-cache.js +162 -0
  153. package/dist/core/lsp/symbol-tools.js +664 -0
  154. package/dist/core/mcp/client.js +97 -28
  155. package/dist/core/mcp/http-server.js +553 -0
  156. package/dist/core/mcp/orchestrator-tools.js +662 -0
  157. package/dist/core/mcp/permission.js +190 -0
  158. package/dist/core/mcp/registry.js +39 -17
  159. package/dist/core/mcp/server-tools.js +219 -0
  160. package/dist/core/mcp/server.js +397 -0
  161. package/dist/core/mcp/trust.js +10 -10
  162. package/dist/core/memory/dual-write.js +416 -0
  163. package/dist/core/memory/passive-extract.js +130 -0
  164. package/dist/core/memory/phase1-kinds.js +20 -0
  165. package/dist/core/memory/secret-scanner.js +304 -0
  166. package/dist/core/memory-sync/queue.js +170 -0
  167. package/dist/core/metrics/extract.js +113 -0
  168. package/dist/core/modes/roo-modes.js +68 -0
  169. package/dist/core/onboarding/ensure-initialized.js +133 -0
  170. package/dist/core/onboarding/marker.js +111 -0
  171. package/dist/core/onboarding/telemetry-state.js +108 -0
  172. package/dist/core/output-style/presets.js +176 -0
  173. package/dist/core/output-style/state.js +185 -0
  174. package/dist/core/path-security.js +287 -5
  175. package/dist/core/permission.js +82 -22
  176. package/dist/core/permissions/auto-classifier.js +124 -0
  177. package/dist/core/permissions/bash-parser.js +371 -0
  178. package/dist/core/permissions/circuit-breaker.js +83 -0
  179. package/dist/core/permissions/constrained-edit.js +91 -0
  180. package/dist/core/permissions/gate.js +278 -0
  181. package/dist/core/permissions/index.js +20 -0
  182. package/dist/core/permissions/mode.js +174 -0
  183. package/dist/core/permissions/network-egress.js +137 -0
  184. package/dist/core/permissions/state.js +241 -0
  185. package/dist/core/permissions/tool-class.js +93 -0
  186. package/dist/core/plan-mode/ui-state.js +51 -0
  187. package/dist/core/plans/plan-artifact.js +721 -0
  188. package/dist/core/policy-limits/etag-store.js +122 -0
  189. package/dist/core/prd-check/parser.js +215 -0
  190. package/dist/core/prd-check/reporter.js +127 -0
  191. package/dist/core/prd-check/session-review.js +557 -0
  192. package/dist/core/prd-check/verifiers.js +223 -0
  193. package/dist/core/prompt-cache/client-cache.js +99 -0
  194. package/dist/core/prompts/assembly.js +29 -0
  195. package/dist/core/prompts/registry.js +364 -0
  196. package/dist/core/pugi-md/cc-compat-rules.js +735 -0
  197. package/dist/core/pugi-md/context-injector.js +76 -0
  198. package/dist/core/pugi-md/walk-up.js +207 -0
  199. package/dist/core/python/uv-installer.js +270 -0
  200. package/dist/core/python/uv-resolver.js +83 -0
  201. package/dist/core/rate-limit/narrator.js +146 -0
  202. package/dist/core/recipes/cli-types.js +20 -0
  203. package/dist/core/recipes/loader.js +103 -0
  204. package/dist/core/recipes/runner.js +345 -0
  205. package/dist/core/recipes/schema.js +587 -0
  206. package/dist/core/release-notes/parser.js +241 -0
  207. package/dist/core/release-notes/state.js +116 -0
  208. package/dist/core/repl/ask.js +37 -37
  209. package/dist/core/repl/cancellation.js +26 -26
  210. package/dist/core/repl/cap-warning.js +4 -4
  211. package/dist/core/repl/clipboard-read.js +11 -11
  212. package/dist/core/repl/dispatch-fsm.js +12 -12
  213. package/dist/core/repl/history-search.js +15 -15
  214. package/dist/core/repl/history.js +28 -18
  215. package/dist/core/repl/kill-ring.js +5 -5
  216. package/dist/core/repl/model-pricing.js +135 -0
  217. package/dist/core/repl/privacy-banner.js +22 -22
  218. package/dist/core/repl/session.js +2148 -217
  219. package/dist/core/repl/slash-commands.js +501 -41
  220. package/dist/core/repl/store/index.js +1 -1
  221. package/dist/core/repl/store/jsonl-log.js +22 -22
  222. package/dist/core/repl/store/lockfile.js +10 -10
  223. package/dist/core/repl/store/session-store.js +136 -107
  224. package/dist/core/repl/store/types.js +15 -15
  225. package/dist/core/repl/store/uuid-v7.js +12 -12
  226. package/dist/core/repl/workspace-context.js +43 -21
  227. package/dist/core/repo-map/build.js +125 -0
  228. package/dist/core/repo-map/cache.js +185 -0
  229. package/dist/core/repo-map/extractor.js +254 -0
  230. package/dist/core/repo-map/formatter.js +145 -0
  231. package/dist/core/repo-map/page-rank.js +105 -0
  232. package/dist/core/repo-map/scanner.js +211 -0
  233. package/dist/core/retry-budget/budget.js +284 -0
  234. package/dist/core/retry-budget/index.js +5 -0
  235. package/dist/core/retry-budget/retry-cap.js +74 -0
  236. package/dist/core/routing/lead-worker.js +43 -0
  237. package/dist/core/routing/pre-flight-estimator.js +108 -0
  238. package/dist/core/runs/run-tree.js +103 -0
  239. package/dist/core/security/injection-scanner.js +367 -0
  240. package/dist/core/security/output-filter.js +418 -0
  241. package/dist/core/session/env-file.js +105 -0
  242. package/dist/core/session/section-budgets.js +140 -0
  243. package/dist/core/session.js +92 -0
  244. package/dist/core/settings.js +324 -5
  245. package/dist/core/share/formatter.js +271 -0
  246. package/dist/core/share/redactor.js +221 -0
  247. package/dist/core/share/uploader.js +267 -0
  248. package/dist/core/skills/defaults.js +30 -30
  249. package/dist/core/skills/loader.js +22 -22
  250. package/dist/core/skills/sources.js +27 -27
  251. package/dist/core/smoke/headless-driver.js +174 -0
  252. package/dist/core/smoke/orchestrator.js +194 -0
  253. package/dist/core/smoke/runner.js +238 -0
  254. package/dist/core/smoke/scenario-parser.js +316 -0
  255. package/dist/core/statusline.js +99 -0
  256. package/dist/core/subagents/dispatcher-real.js +600 -0
  257. package/dist/core/subagents/dispatcher.js +132 -43
  258. package/dist/core/subagents/index.js +19 -6
  259. package/dist/core/subagents/isolation-matrix.js +213 -0
  260. package/dist/core/subagents/spawn.js +19 -4
  261. package/dist/core/telemetry/emitter.js +229 -0
  262. package/dist/core/telemetry/queue.js +251 -0
  263. package/dist/core/theme/context.js +91 -0
  264. package/dist/core/theme/presets.js +228 -0
  265. package/dist/core/theme/state.js +181 -0
  266. package/dist/core/todos/invariant.js +10 -0
  267. package/dist/core/todos/state.js +177 -0
  268. package/dist/core/tool-schema/compressor.js +89 -0
  269. package/dist/core/transport/version-interceptor.js +166 -0
  270. package/dist/core/trust.js +2 -2
  271. package/dist/core/tui/thinking-block.js +64 -0
  272. package/dist/core/vim/keymap.js +288 -0
  273. package/dist/core/vim/state.js +92 -0
  274. package/dist/core/watch-markers/marker-watcher.js +133 -0
  275. package/dist/core/worktree/include-parser.js +249 -0
  276. package/dist/core/worktree-manager/cleanup.js +123 -0
  277. package/dist/core/worktree-manager/manager.js +303 -0
  278. package/dist/index.js +36 -0
  279. package/dist/runtime/bootstrap.js +190 -0
  280. package/dist/runtime/cli.js +4185 -549
  281. package/dist/runtime/commands/agents.js +31 -31
  282. package/dist/runtime/commands/budget.js +5 -5
  283. package/dist/runtime/commands/cancel.js +231 -0
  284. package/dist/runtime/commands/chain.js +489 -0
  285. package/dist/runtime/commands/codegraph-status.js +227 -0
  286. package/dist/runtime/commands/compact.js +297 -0
  287. package/dist/runtime/commands/config.js +73 -39
  288. package/dist/runtime/commands/cost.js +199 -0
  289. package/dist/runtime/commands/delegate.js +27 -4
  290. package/dist/runtime/commands/dispatch.js +126 -0
  291. package/dist/runtime/commands/doctor.js +579 -0
  292. package/dist/runtime/commands/feedback.js +184 -0
  293. package/dist/runtime/commands/hooks.js +187 -0
  294. package/dist/runtime/commands/init.js +254 -0
  295. package/dist/runtime/commands/lsp.js +200 -38
  296. package/dist/runtime/commands/mcp.js +879 -0
  297. package/dist/runtime/commands/memory.js +582 -0
  298. package/dist/runtime/commands/model.js +237 -0
  299. package/dist/runtime/commands/onboarding.js +275 -0
  300. package/dist/runtime/commands/patch.js +12 -12
  301. package/dist/runtime/commands/permissions.js +112 -0
  302. package/dist/runtime/commands/plan.js +143 -0
  303. package/dist/runtime/commands/prd-check.js +285 -0
  304. package/dist/runtime/commands/privacy.js +17 -17
  305. package/dist/runtime/commands/recipe.js +325 -0
  306. package/dist/runtime/commands/redo-blob-store.js +92 -0
  307. package/dist/runtime/commands/redo.js +361 -0
  308. package/dist/runtime/commands/release-notes.js +229 -0
  309. package/dist/runtime/commands/repo-map.js +95 -0
  310. package/dist/runtime/commands/report.js +299 -0
  311. package/dist/runtime/commands/resume.js +118 -0
  312. package/dist/runtime/commands/review-consensus.js +68 -53
  313. package/dist/runtime/commands/rewind.js +333 -0
  314. package/dist/runtime/commands/roster.js +14 -14
  315. package/dist/runtime/commands/sessions.js +163 -0
  316. package/dist/runtime/commands/share.js +316 -0
  317. package/dist/runtime/commands/skills.js +31 -31
  318. package/dist/runtime/commands/status.js +186 -0
  319. package/dist/runtime/commands/stickers.js +82 -0
  320. package/dist/runtime/commands/style.js +194 -0
  321. package/dist/runtime/commands/theme.js +196 -0
  322. package/dist/runtime/commands/undo.js +54 -22
  323. package/dist/runtime/commands/update.js +289 -0
  324. package/dist/runtime/commands/vim.js +140 -0
  325. package/dist/runtime/commands/worktree.js +8 -8
  326. package/dist/runtime/commands/worktrees.js +155 -0
  327. package/dist/runtime/headless-repl.js +195 -0
  328. package/dist/runtime/headless.js +543 -0
  329. package/dist/runtime/load-hooks-or-exit.js +71 -0
  330. package/dist/runtime/plan-decompose.js +22 -22
  331. package/dist/runtime/sigint-guard.js +272 -0
  332. package/dist/runtime/update-check.js +28 -28
  333. package/dist/runtime/version.js +65 -0
  334. package/dist/runtime/worktree-bootstrap.js +579 -0
  335. package/dist/skills/bundled/batch.js +617 -0
  336. package/dist/skills/bundled/index.js +45 -0
  337. package/dist/skills/bundled/loop.js +358 -0
  338. package/dist/skills/bundled/remember.js +383 -0
  339. package/dist/skills/bundled/simplify.js +289 -0
  340. package/dist/skills/bundled/skillify.js +373 -0
  341. package/dist/skills/bundled/stuck.js +558 -0
  342. package/dist/skills/bundled/verify.js +439 -0
  343. package/dist/testing/vcr.js +486 -0
  344. package/dist/tools/agent-tool.js +229 -0
  345. package/dist/tools/apply-patch.js +89 -28
  346. package/dist/tools/ask-user-question.js +337 -0
  347. package/dist/tools/ask-user.js +115 -0
  348. package/dist/tools/bash.js +624 -46
  349. package/dist/tools/brief.js +224 -0
  350. package/dist/tools/enter-worktree.js +250 -0
  351. package/dist/tools/exit-worktree.js +147 -0
  352. package/dist/tools/file-tools.js +161 -44
  353. package/dist/tools/lsp-tools.js +377 -1
  354. package/dist/tools/mcp-tool.js +260 -0
  355. package/dist/tools/multi-edit.js +361 -0
  356. package/dist/tools/powershell.js +268 -0
  357. package/dist/tools/registry.js +86 -4
  358. package/dist/tools/skill-tool.js +96 -0
  359. package/dist/tools/sleep.js +99 -0
  360. package/dist/tools/synthetic-output.js +133 -0
  361. package/dist/tools/tasks.js +208 -0
  362. package/dist/tools/todo-write.js +184 -0
  363. package/dist/tools/verify-plan-execution.js +295 -0
  364. package/dist/tools/web-fetch-injection-scanner.js +207 -0
  365. package/dist/tools/web-fetch.js +195 -10
  366. package/dist/tools/web-search.js +458 -0
  367. package/dist/tui/agent-progress-card.js +111 -0
  368. package/dist/tui/agent-tree.js +11 -1
  369. package/dist/tui/ask-modal.js +14 -14
  370. package/dist/tui/ask-user-question-chips.js +315 -0
  371. package/dist/tui/ask-user-question-prompt.js +203 -0
  372. package/dist/tui/compact-banner.js +81 -0
  373. package/dist/tui/conversation-pane.js +85 -11
  374. package/dist/tui/cost-table.js +111 -0
  375. package/dist/tui/device-flow.js +2 -2
  376. package/dist/tui/doctor-table.js +46 -0
  377. package/dist/tui/feedback-prompt.js +156 -0
  378. package/dist/tui/input-box.js +247 -32
  379. package/dist/tui/login-picker.js +3 -3
  380. package/dist/tui/markdown-render.js +6 -6
  381. package/dist/tui/onboarding-wizard.js +240 -0
  382. package/dist/tui/permissions-picker.js +86 -0
  383. package/dist/tui/render.js +36 -1
  384. package/dist/tui/repl-render.js +176 -25
  385. package/dist/tui/repl-splash-art.js +16 -16
  386. package/dist/tui/repl-splash-mascot.js +48 -24
  387. package/dist/tui/repl-splash.js +22 -22
  388. package/dist/tui/repl.js +125 -45
  389. package/dist/tui/slash-palette.js +6 -6
  390. package/dist/tui/splash.js +2 -2
  391. package/dist/tui/status-bar.js +109 -31
  392. package/dist/tui/status-table.js +7 -0
  393. package/dist/tui/stickers-art.js +136 -0
  394. package/dist/tui/style-table.js +28 -0
  395. package/dist/tui/theme-table.js +29 -0
  396. package/dist/tui/thinking-spinner.js +123 -0
  397. package/dist/tui/tool-stream-pane.js +53 -4
  398. package/dist/tui/update-banner.js +27 -2
  399. package/dist/tui/vim-input.js +267 -0
  400. package/dist/tui/welcome-banner.js +107 -0
  401. package/dist/tui/welcome-data.js +293 -0
  402. package/dist/tui/workspace-context.js +2 -2
  403. package/package.json +31 -16
  404. package/test/scenarios/codegen-create-file.scenario.txt +13 -0
  405. package/test/scenarios/compact-force.scenario.txt +12 -0
  406. package/test/scenarios/identity.scenario.txt +12 -0
  407. package/test/scenarios/persona-handoff.scenario.txt +12 -0
  408. package/test/scenarios/walkback.scenario.txt +12 -0
  409. package/dist/core/engine/compaction-hook.js +0 -154
@@ -18,6 +18,98 @@ export function openSession(root) {
18
18
  enabled,
19
19
  };
20
20
  }
21
+ /**
22
+ * MVP — fire the `SessionStart` lifecycle event for all hooks
23
+ * declared in `~/.pugi/hooks-mvp.json`. Single-call surface; the REPL
24
+ * boot path invokes this once after `openSession`. Best-effort: any
25
+ * failure (missing config, hook spawn error) is swallowed so a
26
+ * misconfigured hook can never crash the REPL.
27
+ *
28
+ * Returns the number of hooks that fired (0 when no config / no
29
+ * matching hooks). Tests assert on the return value as the
30
+ * single-call invariant.
31
+ */
32
+ export async function fireSessionStartMvp(session) {
33
+ try {
34
+ const { loadHooksConfig, fireHooks } = await import('./hooks/index.js');
35
+ // Defense-in-depth: `loadHooksConfig` is contractually non-null
36
+ // (returns `HooksConfig.empty(path)` when the file is absent), but
37
+ // the dynamic import boundary above can in principle return an
38
+ // unexpected shape if the module is mis-resolved at runtime. Guard
39
+ // the optional-chained `isEmpty()` call so a malformed loader can
40
+ // never raise `TypeError: Cannot read properties of undefined` and
41
+ // crash the REPL boot path. Belt-and-suspenders with the
42
+ // surrounding try/catch — the catch still swallows everything else.
43
+ const config = loadHooksConfig();
44
+ if (!config || config.isEmpty())
45
+ return 0;
46
+ const outcome = await fireHooks({
47
+ config,
48
+ event: 'SessionStart',
49
+ payload: {
50
+ event: 'SessionStart',
51
+ sessionId: session.id,
52
+ workspaceRoot: session.root,
53
+ startedAt: new Date().toISOString(),
54
+ },
55
+ workspaceRoot: session.root,
56
+ });
57
+ return outcome.results.length;
58
+ }
59
+ catch {
60
+ // SessionStart is never blocking — log nothing, return 0. A
61
+ // broken `hooks-mvp.json` is surfaced via `pugi hooks doctor`.
62
+ return 0;
63
+ }
64
+ }
65
+ /**
66
+ * P1 — fire the v2 `SessionStart` event from `~/.pugi/hooks.json`
67
+ * (global) + `<workspaceRoot>/.pugi/hooks.json` (project). Companion to
68
+ * `fireSessionStartMvp`; both surfaces run because they read different
69
+ * config files.
70
+ *
71
+ * Headless by default (no trust prompt) — the v2 trust ledger gates
72
+ * first-run executions. Operators with no prior trust decision will see
73
+ * the SessionStart hook skipped with a `denied by trust ledger` stderr
74
+ * note; running `pugi hooks trust allow <command>` enrolls it.
75
+ *
76
+ * Returns the number of hooks that ran (excluding trust-denied skips).
77
+ * Never throws.
78
+ */
79
+ export async function fireSessionStartV2(session) {
80
+ try {
81
+ const { fireSessionStart } = await import('./hooks/v2/index.js');
82
+ const outcome = await fireSessionStart({
83
+ sessionId: session.id,
84
+ workspaceRoot: session.root,
85
+ transcriptPath: session.eventsPath,
86
+ permissionMode: 'ask',
87
+ });
88
+ return outcome.results.filter((r) => r.exitCode !== -1).length;
89
+ }
90
+ catch {
91
+ return 0;
92
+ }
93
+ }
94
+ /**
95
+ * P1 — fire the v2 `SessionEnd` event. Called by the REPL
96
+ * teardown path. Companion to `fireSessionStartV2`.
97
+ */
98
+ export async function fireSessionEndV2(session) {
99
+ try {
100
+ const { fireSessionEnd } = await import('./hooks/v2/index.js');
101
+ const outcome = await fireSessionEnd({
102
+ sessionId: session.id,
103
+ workspaceRoot: session.root,
104
+ transcriptPath: session.eventsPath,
105
+ permissionMode: 'ask',
106
+ });
107
+ return outcome.results.filter((r) => r.exitCode !== -1).length;
108
+ }
109
+ catch {
110
+ return 0;
111
+ }
112
+ }
21
113
  export function recordCommandStarted(session, command) {
22
114
  if (!session.enabled)
23
115
  return;
@@ -20,6 +20,22 @@ const pugiSettingsSchema = z.object({
20
20
  allow: z.array(z.string()).default([]),
21
21
  deny: z.array(z.string()).default([]),
22
22
  notAutomatic: z.array(z.string()).default([]),
23
+ // task — operator-declared read-only paths. Every
24
+ // edit / write tool call whose target matches one of these
25
+ // patterns is denied with source `readonly_paths`, regardless
26
+ // of the active permission mode. Match grammar mirrors
27
+ // allow/deny: exact path OR tail-* glob. Targets are
28
+ // workspace-relative, no leading slash.
29
+ //
30
+ // Why a dedicated field вместо leaning on a generic deny rule
31
+ // like `deny: ["edit:vendor/x"]`?
32
+ // - Explicit intent surfaces в `pugi doctor` and the REPL
33
+ // ("3 paths read-only") без parsing rule strings.
34
+ // - Covers BOTH the `edit` tool AND the `write` tool with
35
+ // one entry; a generic deny needs two rules.
36
+ // - Survives a future migration of the `permissions` schema
37
+ // because the field carries its own contract.
38
+ readonlyPaths: z.array(z.string()).default([]),
23
39
  })
24
40
  .default({}),
25
41
  privacy: z
@@ -28,6 +44,17 @@ const pugiSettingsSchema = z.object({
28
44
  telemetry: z.enum(['off', 'anonymous', 'community']).default('off'),
29
45
  })
30
46
  .default({}),
47
+ // beta.13 P1 fix: ui.cyberZoo gates the cyber-zoo splash +
48
+ // ambient art in the REPL. Schema must declare the key explicitly
49
+ // because Zod's strip pass swallows unknown keys, which is how the
50
+ // initial `pugi init` write (which serialises `ui.cyberZoo`) was
51
+ // bypassed by the runtime reader — the value never made it past the
52
+ // schema gate so admin-api always saw the historical 'on' default.
53
+ ui: z
54
+ .object({
55
+ cyberZoo: z.enum(['on', 'off']).default('on'),
56
+ })
57
+ .default({}),
31
58
  artifacts: z
32
59
  .object({
33
60
  defaultPath: z.string().default('.pugi/artifacts'),
@@ -38,6 +65,12 @@ const pugiSettingsSchema = z.object({
38
65
  // fetcher. Default-off matches the spec posture; the schema must
39
66
  // declare it explicitly because Zod's strict-pass strips unknown
40
67
  // keys and would silently swallow the operator's intent.
68
+ //
69
+ // β1b T4 : added `web.search.enabled` to gate the
70
+ // Brave-Search-backed `web_search` tool. Distinct from `web.fetch`
71
+ // because search queries themselves are an egress event that can
72
+ // leak operator intent — an operator may want fetch without
73
+ // implicitly enabling search-as-egress.
41
74
  web: z
42
75
  .object({
43
76
  fetch: z
@@ -45,15 +78,301 @@ const pugiSettingsSchema = z.object({
45
78
  enabled: z.boolean().optional(),
46
79
  })
47
80
  .optional(),
81
+ search: z
82
+ .object({
83
+ enabled: z.boolean().optional(),
84
+ })
85
+ .optional(),
86
+ })
87
+ .optional(),
88
+ // β7 L9 — per-language LSP toggle. When omitted, every supported
89
+ // server is available subject to binary detection on PATH. When
90
+ // present, only languages set to `true` are launched (false silently
91
+ // skips that language even if the binary is installed). Use this in
92
+ // workspaces where a heavyweight server (rust-analyzer indexing a
93
+ // monorepo, pyright on a fresh venv) wastes resources for the
94
+ // current task. The `pugi lsp servers` subcommand surfaces the
95
+ // current toggle state per server.
96
+ //
97
+ // Schema is intentionally permissive (`optional()` on the section AND
98
+ // on every per-language flag) so a partial config keeps the
99
+ // backwards-compatible "every language enabled" default.
100
+ lsp: z
101
+ .object({
102
+ typescript: z.boolean().optional(),
103
+ javascript: z.boolean().optional(),
104
+ python: z.boolean().optional(),
105
+ go: z.boolean().optional(),
106
+ rust: z.boolean().optional(),
107
+ // post-edit auto-diagnostics. When `true`,
108
+ // a successful `edit`/`write`/`multi_edit` triggers a diagnostic
109
+ // pull on the touched file(s) and the result is appended to the
110
+ // tool envelope so the model can self-correct in the same turn.
111
+ // Off by default — the cold-start of `typescript-language-server`
112
+ // is heavy enough that we opt in explicitly until dogfood proves
113
+ // the throughput trade is worth it. Also enabled via env var
114
+ // `PUGI_LSP_POST_EDIT=1` for CI / one-off operator probes.
115
+ postEditDiagnostics: z.boolean().optional(),
116
+ })
117
+ .optional(),
118
+ // β1 Pl9 — per-command budget overrides. Optional. Partial
119
+ // overrides merge against the β1 defaults in
120
+ // `core/engine/budgets.ts::beta1DefaultBudgets`. The schema is
121
+ // intentionally loose at the leaf (positive integers) so a typo lands
122
+ // a deterministic `BudgetConfigError` at `resolveBudget()` instead of
123
+ // a Zod parse error two layers up.
124
+ // task — operator-customizable status line.
125
+ // When `command` is set, Pugi spawns it on each turn boundary with
126
+ // a single JSON document on stdin (see `statusline.ts` for the
127
+ // schema). The command's stdout (first non-empty line, trimmed) is
128
+ // displayed in the Ink Footer. Failures are non-fatal — they emit
129
+ // a structured log line and the footer falls back to the default.
130
+ // Mirrors CC's `statusLine` config so cross-tool muscle memory
131
+ // carries over для operators who lived in CC first.
132
+ statusLine: z
133
+ .object({
134
+ command: z.string().min(1),
135
+ timeoutMs: z.number().int().positive().max(5000).default(500),
136
+ })
137
+ .optional(),
138
+ budgets: z
139
+ .object({
140
+ code: z
141
+ .object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
142
+ .optional(),
143
+ fix: z
144
+ .object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
145
+ .optional(),
146
+ build: z
147
+ .object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
148
+ .optional(),
149
+ plan: z
150
+ .object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
151
+ .optional(),
152
+ explain: z
153
+ .object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
154
+ .optional(),
155
+ review_triple: z
156
+ .object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
157
+ .optional(),
158
+ })
159
+ .optional(),
160
+ // #24 (CEO P1) — hook chains. First-class config
161
+ // for `PostToolUseFailure` + `TaskCompleted` event chains. The shape
162
+ // is intentionally loose (`z.any()` at the leaf) because the
163
+ // canonical reader lives in `core/hook-chains.ts` where the
164
+ // matcher/run/timeoutMs grammar is validated. Declaring the key here
165
+ // keeps Zod's strip-pass from swallowing it before the chain reader
166
+ // sees it. See `hook-chains.ts` for the full schema.
167
+ hooks: z.any().optional(),
168
+ // PUGI-260 — persistent default for the 1M context tier opt-in.
169
+ // `pugi config set context.tier 1m` writes this; per-invocation
170
+ // `--context-tier=...` flags override it. When omitted, the CLI
171
+ // sends no `contextTier` field на the wire (server treats as
172
+ // `standard` routing). The closed enum mirrors the CLI flag и the
173
+ // admin-api DTO; an unrecognised value triggers a Zod parse error
174
+ // at load time rather than a silent fallback.
175
+ context: z
176
+ .object({
177
+ tier: z.enum(['1m', 'standard']).optional(),
178
+ })
179
+ .optional(),
180
+ // PUGI-487 - `pugi --worktree` flag governance.
181
+ //
182
+ // Two knobs control the user-facing --worktree flag introduced for
183
+ // parity with parallel-agent isolation patterns in other coding
184
+ // CLIs:
185
+ //
186
+ // - `baseRef`: which ref the new worktree branches FROM.
187
+ // `'fresh'` (default) resolves origin/<default-branch> so each
188
+ // parallel session starts from a clean trunk.
189
+ // `'head'` carries the operator's current local HEAD (including
190
+ // unpushed work) into the new tree.
191
+ //
192
+ // - `cleanupPeriodDays`: integer days. The daily sweep removes
193
+ // user-facing worktrees older than N days that have no
194
+ // uncommitted, untracked, or unpushed state. Default 7 mirrors
195
+ // a one-work-week window. Set to 0 to disable auto-cleanup.
196
+ //
197
+ // Both knobs are optional - the consumers (`bootstrapWorktree`,
198
+ // `runUserWorktreeCleanup`) carry their own defaults so a missing
199
+ // section produces standard behaviour.
200
+ worktree: z
201
+ .object({
202
+ baseRef: z.enum(['fresh', 'head']).optional(),
203
+ cleanupPeriodDays: z.number().int().min(0).max(365).optional(),
48
204
  })
49
205
  .optional(),
50
206
  });
51
- export function loadSettings(root) {
207
+ /**
208
+ * #20 — the upstream tool drop-in compat ingest.
209
+ *
210
+ * Operators migrating from the upstream tool typically keep a `.claude/`
211
+ * directory at workspace root with settings.json, slash commands,
212
+ * and ambient guidance files. We honour the existence of that
213
+ * directory and mirror the subset of keys that map cleanly onto
214
+ * Pugi's own settings surface — Pugi values ALWAYS win on conflict
215
+ * (the operator opted into Pugi as their primary), CC fills gaps.
216
+ *
217
+ * Opt-out: `PUGI_CC_COMPAT_DISABLE=1` short-circuits the merger and
218
+ * loads only `.pugi/settings.json` (or the empty default).
219
+ *
220
+ * Key mirror table:
221
+ * - `permissions.defaultMode` → `permissions.mode`
222
+ * (CC values map: `acceptEdits|plan|bypassPermissions|default` →
223
+ * `acceptEdits|plan|bypassPermissions|ask`).
224
+ * - `permissions.allow` → `permissions.allow` (concatenated, deduped).
225
+ * - `permissions.deny` → `permissions.deny` (concatenated, deduped).
226
+ * - `enabledPlugins` → ignored (CC-only concept; Pugi has its own
227
+ * plugin surface and we do not want to silently activate them).
228
+ * - `hooks` → currently ignored. Pugi's hook system is
229
+ * managed via `apps/pugi-cli/src/core/hooks/`; future work can
230
+ * wire CC hook entries through that bridge.
231
+ *
232
+ * Unknown CC keys are dropped on the floor by Zod's strip semantics
233
+ * just like the existing PUGI settings path — we never warn on
234
+ * unrecognised CC keys, because the CC surface is wider and we want
235
+ * fallthrough to be silent (operator does not need a stream of "we
236
+ * skipped this CC concept" warnings on every CLI invocation).
237
+ */
238
+ const ccPermissionsSchema = z
239
+ .object({
240
+ defaultMode: z.string().optional(),
241
+ allow: z.array(z.string()).optional(),
242
+ deny: z.array(z.string()).optional(),
243
+ })
244
+ .passthrough()
245
+ .optional();
246
+ const ccSettingsSchema = z
247
+ .object({
248
+ permissions: ccPermissionsSchema,
249
+ enabledPlugins: z.unknown().optional(),
250
+ hooks: z.unknown().optional(),
251
+ })
252
+ .passthrough();
253
+ /**
254
+ * Env var that disables ingest entirely. Useful for CI
255
+ * sandboxes where a stray `.claude/` from a parent checkout could
256
+ * otherwise leak permissions into Pugi.
257
+ */
258
+ export const CC_COMPAT_DISABLE_ENV = 'PUGI_CC_COMPAT_DISABLE';
259
+ /**
260
+ * Map a CC `permissions.defaultMode` to the closest Pugi permission
261
+ * mode. Unknown / missing values map to `undefined` so the caller
262
+ * keeps Pugi's own default.
263
+ *
264
+ * CC's `default` mode = "ask the user for each tool" which is Pugi's
265
+ * `ask` mode. `acceptEdits` / `plan` / `bypassPermissions` map 1:1.
266
+ */
267
+ export function mapCcPermissionMode(mode) {
268
+ if (typeof mode !== 'string')
269
+ return undefined;
270
+ switch (mode) {
271
+ case 'acceptEdits':
272
+ return 'acceptEdits';
273
+ case 'plan':
274
+ return 'plan';
275
+ case 'bypassPermissions':
276
+ return 'bypassPermissions';
277
+ case 'default':
278
+ return 'ask';
279
+ default:
280
+ return undefined;
281
+ }
282
+ }
283
+ /**
284
+ * Merge a parsed CC settings object into a Pugi settings object.
285
+ * Pugi ALWAYS wins on conflict; CC values fill gaps only.
286
+ */
287
+ export function mergeCcIntoPugi(pugi, cc, opts) {
288
+ const merged = {
289
+ ...pugi,
290
+ permissions: { ...pugi.permissions },
291
+ };
292
+ const pugiWroteMode = pugiPermissionKeyPresent(opts.pugiRawJson, 'mode');
293
+ if (!pugiWroteMode) {
294
+ const ccMode = mapCcPermissionMode(cc.permissions?.defaultMode);
295
+ if (ccMode)
296
+ merged.permissions.mode = ccMode;
297
+ }
298
+ if (Array.isArray(cc.permissions?.allow)) {
299
+ merged.permissions.allow = dedupeKeepFirst([
300
+ ...pugi.permissions.allow,
301
+ ...cc.permissions.allow,
302
+ ]);
303
+ }
304
+ if (Array.isArray(cc.permissions?.deny)) {
305
+ merged.permissions.deny = dedupeKeepFirst([
306
+ ...pugi.permissions.deny,
307
+ ...cc.permissions.deny,
308
+ ]);
309
+ }
310
+ // `enabledPlugins` and `hooks` are intentionally NOT mirrored. See
311
+ // the doc-block above for rationale.
312
+ void opts.pugiSettingsExisted;
313
+ return merged;
314
+ }
315
+ function pugiPermissionKeyPresent(raw, key) {
316
+ if (!raw || typeof raw !== 'object')
317
+ return false;
318
+ const permissions = raw.permissions;
319
+ if (!permissions || typeof permissions !== 'object')
320
+ return false;
321
+ return Object.prototype.hasOwnProperty.call(permissions, key);
322
+ }
323
+ function dedupeKeepFirst(items) {
324
+ const seen = new Set();
325
+ const out = [];
326
+ for (const item of items) {
327
+ if (seen.has(item))
328
+ continue;
329
+ seen.add(item);
330
+ out.push(item);
331
+ }
332
+ return out;
333
+ }
334
+ /**
335
+ * Read + parse `.claude/settings.json` at `root`. Returns `undefined`
336
+ * when the file is absent, malformed, or the operator has opted out
337
+ * via `PUGI_CC_COMPAT_DISABLE=1`. Never throws — a broken CC settings
338
+ * file degrades to "no ingest", not a Pugi boot crash.
339
+ */
340
+ export function loadCcSettings(root, env = process.env) {
341
+ if (env[CC_COMPAT_DISABLE_ENV] === '1')
342
+ return undefined;
343
+ const ccPath = resolve(root, '.claude/settings.json');
344
+ if (!existsSync(ccPath))
345
+ return undefined;
346
+ let parsed;
347
+ try {
348
+ parsed = JSON.parse(readFileSync(ccPath, 'utf8'));
349
+ }
350
+ catch {
351
+ return undefined;
352
+ }
353
+ const result = ccSettingsSchema.safeParse(parsed);
354
+ if (!result.success)
355
+ return undefined;
356
+ return result.data;
357
+ }
358
+ export function loadSettings(root, env = process.env) {
52
359
  const settingsPath = resolve(root, '.pugi/settings.json');
53
- if (!existsSync(settingsPath)) {
54
- return pugiSettingsSchema.parse({});
360
+ const pugiExists = existsSync(settingsPath);
361
+ let pugiRawJson = undefined;
362
+ let pugi;
363
+ if (pugiExists) {
364
+ pugiRawJson = JSON.parse(readFileSync(settingsPath, 'utf8'));
365
+ pugi = pugiSettingsSchema.parse(pugiRawJson);
366
+ }
367
+ else {
368
+ pugi = pugiSettingsSchema.parse({});
55
369
  }
56
- const parsed = JSON.parse(readFileSync(settingsPath, 'utf8'));
57
- return pugiSettingsSchema.parse(parsed);
370
+ const cc = loadCcSettings(root, env);
371
+ if (!cc)
372
+ return pugi;
373
+ return mergeCcIntoPugi(pugi, cc, {
374
+ pugiSettingsExisted: pugiExists,
375
+ pugiRawJson,
376
+ });
58
377
  }
59
378
  //# sourceMappingURL=settings.js.map