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

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 (445) hide show
  1. package/CHANGELOG.md +132 -0
  2. package/LICENSE +1 -1
  3. package/README.md +53 -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 +71 -0
  45. package/dist/core/codegraph/types.js +34 -0
  46. package/dist/core/compact/auto-trigger.js +96 -0
  47. package/dist/core/compact/buffer-rewriter.js +115 -0
  48. package/dist/core/compact/summarizer.js +208 -0
  49. package/dist/core/compact/token-counter.js +108 -0
  50. package/dist/core/consensus/anvil-fanout.js +25 -25
  51. package/dist/core/consensus/diff-capture.js +121 -12
  52. package/dist/core/consensus/rubric.js +21 -21
  53. package/dist/core/context/builder.js +6 -6
  54. package/dist/core/context/compaction-events.js +8 -8
  55. package/dist/core/context/compaction.js +31 -31
  56. package/dist/core/context/index.js +15 -8
  57. package/dist/core/context/invariants.js +51 -51
  58. package/dist/core/context/markdown-loader.js +28 -10
  59. package/dist/core/context/markdown-traverse.js +255 -0
  60. package/dist/core/context/pugiignore.js +41 -41
  61. package/dist/core/context/repo-skeleton.js +37 -37
  62. package/dist/core/context/tool-eviction.js +55 -0
  63. package/dist/core/context/watcher.js +32 -32
  64. package/dist/core/context/working-set.js +23 -23
  65. package/dist/core/coordinator/agent-tools.js +77 -0
  66. package/dist/core/coordinator/agent-toolset.js +65 -0
  67. package/dist/core/coordinator/fsm.js +73 -0
  68. package/dist/core/coordinator/mode-fsm.js +70 -0
  69. package/dist/core/cost/rate-card.js +129 -0
  70. package/dist/core/cost/tracker.js +221 -0
  71. package/dist/core/credentials.js +13 -13
  72. package/dist/core/cron/scheduler.js +138 -0
  73. package/dist/core/denial-tracking/index.js +8 -0
  74. package/dist/core/denial-tracking/state.js +264 -0
  75. package/dist/core/diagnostics/probe-runner.js +93 -0
  76. package/dist/core/diagnostics/probes/api.js +46 -0
  77. package/dist/core/diagnostics/probes/auth.js +93 -0
  78. package/dist/core/diagnostics/probes/bare-mode.js +42 -0
  79. package/dist/core/diagnostics/probes/cli-version.js +127 -0
  80. package/dist/core/diagnostics/probes/config.js +72 -0
  81. package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
  82. package/dist/core/diagnostics/probes/disk.js +81 -0
  83. package/dist/core/diagnostics/probes/engine-live.js +46 -0
  84. package/dist/core/diagnostics/probes/git.js +65 -0
  85. package/dist/core/diagnostics/probes/hooks.js +118 -0
  86. package/dist/core/diagnostics/probes/mcp.js +75 -0
  87. package/dist/core/diagnostics/probes/node.js +59 -0
  88. package/dist/core/diagnostics/probes/pnpm.js +36 -0
  89. package/dist/core/diagnostics/probes/pugi-md.js +89 -0
  90. package/dist/core/diagnostics/probes/sandbox.js +72 -0
  91. package/dist/core/diagnostics/probes/session.js +74 -0
  92. package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
  93. package/dist/core/diagnostics/probes/workspace.js +63 -0
  94. package/dist/core/diagnostics/types.js +70 -0
  95. package/dist/core/dispatch/cache-cleanup.js +197 -0
  96. package/dist/core/dispatch/cache-handoff.js +295 -0
  97. package/dist/core/edits/apply-patch-layer-e.js +189 -0
  98. package/dist/core/edits/dispatch.js +333 -7
  99. package/dist/core/edits/format-detector.js +260 -0
  100. package/dist/core/edits/format-matrix.js +26 -0
  101. package/dist/core/edits/fuzzy-ladder.js +650 -0
  102. package/dist/core/edits/index.js +5 -1
  103. package/dist/core/edits/journal.js +199 -0
  104. package/dist/core/edits/layer-a-apply.js +15 -15
  105. package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
  106. package/dist/core/edits/layer-b-apply.js +9 -9
  107. package/dist/core/edits/layer-c-apply.js +6 -6
  108. package/dist/core/edits/layer-d-ast.js +557 -14
  109. package/dist/core/edits/marker-parser.js +12 -12
  110. package/dist/core/edits/security-gate.js +27 -27
  111. package/dist/core/edits/verify-hook.js +273 -0
  112. package/dist/core/edits/worktree.js +29 -29
  113. package/dist/core/engine/anvil-client.js +214 -26
  114. package/dist/core/engine/auto-compact.js +247 -0
  115. package/dist/core/engine/budgets.js +220 -0
  116. package/dist/core/engine/compact-llm-summarizer.js +124 -0
  117. package/dist/core/engine/context-prefix.js +155 -0
  118. package/dist/core/engine/index.js +1 -1
  119. package/dist/core/engine/intensity.js +163 -0
  120. package/dist/core/engine/intent.js +260 -0
  121. package/dist/core/engine/native-pugi.js +1559 -227
  122. package/dist/core/engine/prompts.js +187 -19
  123. package/dist/core/engine/strip-internal-fields.js +124 -0
  124. package/dist/core/engine/tool-bridge.js +1887 -59
  125. package/dist/core/engine/verification-patterns.js +195 -0
  126. package/dist/core/evaluation/golden-dataset.js +293 -0
  127. package/dist/core/feedback/queue.js +177 -0
  128. package/dist/core/feedback/submitter.js +145 -0
  129. package/dist/core/file-cache.js +113 -1
  130. package/dist/core/flatten/flatten-repo.js +439 -0
  131. package/dist/core/format/osc8-link.js +28 -0
  132. package/dist/core/hook-chains.js +392 -0
  133. package/dist/core/hooks/citation-verify-hook.js +138 -0
  134. package/dist/core/hooks/citation-verify.js +112 -0
  135. package/dist/core/hooks/events.js +46 -0
  136. package/dist/core/hooks/index.js +15 -0
  137. package/dist/core/hooks/registry.js +216 -0
  138. package/dist/core/hooks/runner.js +236 -0
  139. package/dist/core/hooks/v2/event-emitter.js +115 -0
  140. package/dist/core/hooks/v2/executor.js +282 -0
  141. package/dist/core/hooks/v2/index.js +25 -0
  142. package/dist/core/hooks/v2/lifecycle.js +104 -0
  143. package/dist/core/hooks/v2/loader.js +216 -0
  144. package/dist/core/hooks/v2/matcher.js +125 -0
  145. package/dist/core/hooks/v2/trust.js +143 -0
  146. package/dist/core/hooks/v2/types.js +86 -0
  147. package/dist/core/hooks/worktree-events.js +158 -0
  148. package/dist/core/image/renderer.js +71 -0
  149. package/dist/core/init/detector.js +582 -0
  150. package/dist/core/init/template-renderer.js +242 -0
  151. package/dist/core/jobs/registry.js +18 -18
  152. package/dist/core/ledger/results-tsv.js +142 -0
  153. package/dist/core/log-discipline/stdout-redirect.js +51 -0
  154. package/dist/core/lsp/cache.js +105 -0
  155. package/dist/core/lsp/client.js +551 -41
  156. package/dist/core/lsp/language-detect.js +66 -0
  157. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  158. package/dist/core/lsp/server-detect.js +173 -0
  159. package/dist/core/lsp/symbol-cache.js +162 -0
  160. package/dist/core/lsp/symbol-tools.js +664 -0
  161. package/dist/core/mcp/client.js +97 -28
  162. package/dist/core/mcp/http-server.js +553 -0
  163. package/dist/core/mcp/orchestrator-config.js +192 -0
  164. package/dist/core/mcp/orchestrator-tools.js +806 -0
  165. package/dist/core/mcp/permission.js +190 -0
  166. package/dist/core/mcp/registry.js +39 -17
  167. package/dist/core/mcp/server-tools.js +219 -0
  168. package/dist/core/mcp/server.js +397 -0
  169. package/dist/core/mcp/trust.js +10 -10
  170. package/dist/core/memory/dual-write.js +416 -0
  171. package/dist/core/memory/passive-extract.js +130 -0
  172. package/dist/core/memory/phase1-kinds.js +20 -0
  173. package/dist/core/memory/secret-scanner.js +304 -0
  174. package/dist/core/memory-sync/queue.js +170 -0
  175. package/dist/core/metrics/extract.js +113 -0
  176. package/dist/core/modes/roo-modes.js +68 -0
  177. package/dist/core/notes/notes-paths.js +113 -0
  178. package/dist/core/notes/notes-recorder.js +140 -0
  179. package/dist/core/notes/notes-writer.js +53 -0
  180. package/dist/core/notes/renderers.js +0 -0
  181. package/dist/core/notes/slug.js +105 -0
  182. package/dist/core/onboarding/ensure-initialized.js +133 -0
  183. package/dist/core/onboarding/marker.js +111 -0
  184. package/dist/core/onboarding/telemetry-state.js +108 -0
  185. package/dist/core/output-style/presets.js +176 -0
  186. package/dist/core/output-style/state.js +185 -0
  187. package/dist/core/path-security.js +287 -5
  188. package/dist/core/permission.js +82 -22
  189. package/dist/core/permissions/auto-classifier.js +124 -0
  190. package/dist/core/permissions/bash-parser.js +371 -0
  191. package/dist/core/permissions/circuit-breaker.js +83 -0
  192. package/dist/core/permissions/constrained-edit.js +91 -0
  193. package/dist/core/permissions/gate.js +278 -0
  194. package/dist/core/permissions/index.js +20 -0
  195. package/dist/core/permissions/mode.js +174 -0
  196. package/dist/core/permissions/network-egress.js +137 -0
  197. package/dist/core/permissions/state.js +241 -0
  198. package/dist/core/permissions/tool-class.js +107 -0
  199. package/dist/core/plan-mode/ui-state.js +51 -0
  200. package/dist/core/plans/plan-artifact.js +721 -0
  201. package/dist/core/policy-limits/etag-store.js +122 -0
  202. package/dist/core/prd-check/parser.js +215 -0
  203. package/dist/core/prd-check/reporter.js +127 -0
  204. package/dist/core/prd-check/session-review.js +557 -0
  205. package/dist/core/prd-check/verifiers.js +223 -0
  206. package/dist/core/prompt-cache/client-cache.js +99 -0
  207. package/dist/core/prompts/assembly.js +29 -0
  208. package/dist/core/prompts/registry.js +364 -0
  209. package/dist/core/pugi-gitignore.js +52 -0
  210. package/dist/core/pugi-md/cc-compat-rules.js +735 -0
  211. package/dist/core/pugi-md/context-injector.js +76 -0
  212. package/dist/core/pugi-md/walk-up.js +207 -0
  213. package/dist/core/python/uv-installer.js +270 -0
  214. package/dist/core/python/uv-resolver.js +83 -0
  215. package/dist/core/rate-limit/narrator.js +146 -0
  216. package/dist/core/recipes/cli-types.js +20 -0
  217. package/dist/core/recipes/loader.js +103 -0
  218. package/dist/core/recipes/runner.js +345 -0
  219. package/dist/core/recipes/schema.js +587 -0
  220. package/dist/core/release-notes/parser.js +241 -0
  221. package/dist/core/release-notes/state.js +116 -0
  222. package/dist/core/repl/ask.js +37 -37
  223. package/dist/core/repl/cancellation.js +26 -26
  224. package/dist/core/repl/cap-warning.js +4 -4
  225. package/dist/core/repl/clipboard-read.js +11 -11
  226. package/dist/core/repl/dispatch-fsm.js +12 -12
  227. package/dist/core/repl/engine-bridge.js +303 -0
  228. package/dist/core/repl/history-search.js +15 -15
  229. package/dist/core/repl/history.js +28 -18
  230. package/dist/core/repl/kill-ring.js +5 -5
  231. package/dist/core/repl/model-pricing.js +135 -0
  232. package/dist/core/repl/privacy-banner.js +22 -22
  233. package/dist/core/repl/session.js +2690 -229
  234. package/dist/core/repl/slash-commands.js +540 -41
  235. package/dist/core/repl/store/index.js +1 -1
  236. package/dist/core/repl/store/jsonl-log.js +22 -22
  237. package/dist/core/repl/store/lockfile.js +10 -10
  238. package/dist/core/repl/store/session-store.js +136 -107
  239. package/dist/core/repl/store/types.js +15 -15
  240. package/dist/core/repl/store/uuid-v7.js +12 -12
  241. package/dist/core/repl/tool-route.js +382 -0
  242. package/dist/core/repl/workspace-context.js +43 -21
  243. package/dist/core/repo-map/build.js +125 -0
  244. package/dist/core/repo-map/cache.js +185 -0
  245. package/dist/core/repo-map/extractor.js +254 -0
  246. package/dist/core/repo-map/formatter.js +145 -0
  247. package/dist/core/repo-map/page-rank.js +105 -0
  248. package/dist/core/repo-map/scanner.js +211 -0
  249. package/dist/core/retro/git-collector.js +251 -0
  250. package/dist/core/retro/health-card.js +25 -0
  251. package/dist/core/retro/metrics.js +342 -0
  252. package/dist/core/retro/narrative.js +249 -0
  253. package/dist/core/retro/plane-collector.js +274 -0
  254. package/dist/core/retro/pr-issue-link.js +65 -0
  255. package/dist/core/retro/types.js +16 -0
  256. package/dist/core/retry-budget/budget.js +284 -0
  257. package/dist/core/retry-budget/index.js +5 -0
  258. package/dist/core/retry-budget/retry-cap.js +74 -0
  259. package/dist/core/routing/lead-worker.js +43 -0
  260. package/dist/core/routing/pre-flight-estimator.js +108 -0
  261. package/dist/core/runs/run-tree.js +103 -0
  262. package/dist/core/sandboxing/adapter.js +29 -0
  263. package/dist/core/sandboxing/index.js +49 -0
  264. package/dist/core/sandboxing/none.js +19 -0
  265. package/dist/core/sandboxing/seatbelt.js +183 -0
  266. package/dist/core/security/injection-scanner.js +367 -0
  267. package/dist/core/security/output-filter.js +418 -0
  268. package/dist/core/session/env-file.js +105 -0
  269. package/dist/core/session/section-budgets.js +140 -0
  270. package/dist/core/session.js +119 -0
  271. package/dist/core/settings.js +378 -5
  272. package/dist/core/share/formatter.js +271 -0
  273. package/dist/core/share/redactor.js +221 -0
  274. package/dist/core/share/uploader.js +267 -0
  275. package/dist/core/skills/defaults.js +30 -30
  276. package/dist/core/skills/loader.js +22 -22
  277. package/dist/core/skills/sources.js +27 -27
  278. package/dist/core/smoke/headless-driver.js +174 -0
  279. package/dist/core/smoke/orchestrator.js +194 -0
  280. package/dist/core/smoke/runner.js +238 -0
  281. package/dist/core/smoke/scenario-parser.js +316 -0
  282. package/dist/core/statusline.js +99 -0
  283. package/dist/core/subagents/dispatcher-real.js +600 -0
  284. package/dist/core/subagents/dispatcher.js +146 -52
  285. package/dist/core/subagents/index.js +19 -6
  286. package/dist/core/subagents/isolation-matrix.js +213 -0
  287. package/dist/core/subagents/spawn.js +19 -4
  288. package/dist/core/telemetry/emitter.js +229 -0
  289. package/dist/core/telemetry/queue.js +251 -0
  290. package/dist/core/theme/context.js +91 -0
  291. package/dist/core/theme/presets.js +228 -0
  292. package/dist/core/theme/state.js +181 -0
  293. package/dist/core/todos/invariant.js +10 -0
  294. package/dist/core/todos/state.js +177 -0
  295. package/dist/core/tool-schema/compressor.js +89 -0
  296. package/dist/core/transport/version-interceptor.js +166 -0
  297. package/dist/core/trust.js +2 -2
  298. package/dist/core/tui/thinking-block.js +64 -0
  299. package/dist/core/vim/keymap.js +288 -0
  300. package/dist/core/vim/state.js +92 -0
  301. package/dist/core/watch-markers/marker-watcher.js +133 -0
  302. package/dist/core/worktree/include-parser.js +249 -0
  303. package/dist/core/worktree-manager/cleanup.js +123 -0
  304. package/dist/core/worktree-manager/manager.js +303 -0
  305. package/dist/index.js +36 -0
  306. package/dist/runtime/bootstrap.js +190 -0
  307. package/dist/runtime/cli.js +4345 -561
  308. package/dist/runtime/commands/agents.js +31 -31
  309. package/dist/runtime/commands/budget.js +5 -5
  310. package/dist/runtime/commands/cancel.js +231 -0
  311. package/dist/runtime/commands/chain.js +489 -0
  312. package/dist/runtime/commands/codegraph-status.js +227 -0
  313. package/dist/runtime/commands/compact.js +297 -0
  314. package/dist/runtime/commands/config.js +74 -40
  315. package/dist/runtime/commands/cost.js +199 -0
  316. package/dist/runtime/commands/delegate.js +27 -4
  317. package/dist/runtime/commands/dispatch.js +126 -0
  318. package/dist/runtime/commands/doctor.js +579 -0
  319. package/dist/runtime/commands/feedback.js +184 -0
  320. package/dist/runtime/commands/hooks.js +187 -0
  321. package/dist/runtime/commands/index-cmd.js +353 -0
  322. package/dist/runtime/commands/init.js +254 -0
  323. package/dist/runtime/commands/lsp.js +200 -38
  324. package/dist/runtime/commands/mcp.js +935 -0
  325. package/dist/runtime/commands/memory.js +582 -0
  326. package/dist/runtime/commands/model.js +237 -0
  327. package/dist/runtime/commands/onboarding.js +275 -0
  328. package/dist/runtime/commands/patch.js +12 -12
  329. package/dist/runtime/commands/permissions.js +112 -0
  330. package/dist/runtime/commands/plan.js +143 -0
  331. package/dist/runtime/commands/prd-check.js +285 -0
  332. package/dist/runtime/commands/privacy.js +17 -17
  333. package/dist/runtime/commands/recipe.js +325 -0
  334. package/dist/runtime/commands/redo-blob-store.js +92 -0
  335. package/dist/runtime/commands/redo.js +361 -0
  336. package/dist/runtime/commands/release-notes.js +229 -0
  337. package/dist/runtime/commands/repo-map.js +95 -0
  338. package/dist/runtime/commands/report.js +299 -0
  339. package/dist/runtime/commands/resume.js +118 -0
  340. package/dist/runtime/commands/review-consensus.js +68 -53
  341. package/dist/runtime/commands/rewind.js +333 -0
  342. package/dist/runtime/commands/roster.js +14 -14
  343. package/dist/runtime/commands/servers.js +236 -0
  344. package/dist/runtime/commands/sessions.js +163 -0
  345. package/dist/runtime/commands/share.js +316 -0
  346. package/dist/runtime/commands/skills.js +31 -31
  347. package/dist/runtime/commands/status.js +186 -0
  348. package/dist/runtime/commands/stickers.js +82 -0
  349. package/dist/runtime/commands/style.js +194 -0
  350. package/dist/runtime/commands/theme.js +196 -0
  351. package/dist/runtime/commands/undo.js +54 -22
  352. package/dist/runtime/commands/update.js +289 -0
  353. package/dist/runtime/commands/vim.js +140 -0
  354. package/dist/runtime/commands/worktree.js +8 -8
  355. package/dist/runtime/commands/worktrees.js +155 -0
  356. package/dist/runtime/deprecation-warning.js +69 -0
  357. package/dist/runtime/engine-exit-code.js +50 -0
  358. package/dist/runtime/headless-repl.js +195 -0
  359. package/dist/runtime/headless.js +548 -0
  360. package/dist/runtime/load-hooks-or-exit.js +71 -0
  361. package/dist/runtime/plan-decompose.js +22 -22
  362. package/dist/runtime/sigint-guard.js +272 -0
  363. package/dist/runtime/stream-renderer.js +195 -0
  364. package/dist/runtime/update-check.js +28 -28
  365. package/dist/runtime/version.js +65 -0
  366. package/dist/runtime/worktree-bootstrap.js +579 -0
  367. package/dist/skills/bundled/batch.js +617 -0
  368. package/dist/skills/bundled/index.js +45 -0
  369. package/dist/skills/bundled/loop.js +358 -0
  370. package/dist/skills/bundled/remember.js +383 -0
  371. package/dist/skills/bundled/simplify.js +289 -0
  372. package/dist/skills/bundled/skillify.js +373 -0
  373. package/dist/skills/bundled/stuck.js +558 -0
  374. package/dist/skills/bundled/verify.js +439 -0
  375. package/dist/testing/vcr.js +486 -0
  376. package/dist/tools/agent-tool.js +229 -0
  377. package/dist/tools/apply-patch.js +89 -28
  378. package/dist/tools/ask-user-question.js +337 -0
  379. package/dist/tools/ask-user.js +115 -0
  380. package/dist/tools/bash.js +624 -46
  381. package/dist/tools/brief.js +224 -0
  382. package/dist/tools/cron.js +433 -0
  383. package/dist/tools/enter-worktree.js +250 -0
  384. package/dist/tools/exit-worktree.js +147 -0
  385. package/dist/tools/file-tools.js +161 -44
  386. package/dist/tools/http-request.js +336 -0
  387. package/dist/tools/lsp-tools.js +377 -1
  388. package/dist/tools/mcp-tool.js +260 -0
  389. package/dist/tools/multi-edit.js +361 -0
  390. package/dist/tools/powershell.js +268 -0
  391. package/dist/tools/registry.js +120 -5
  392. package/dist/tools/server-tools.js +892 -0
  393. package/dist/tools/skill-tool.js +96 -0
  394. package/dist/tools/sleep.js +99 -0
  395. package/dist/tools/synthetic-output.js +133 -0
  396. package/dist/tools/tasks.js +208 -0
  397. package/dist/tools/todo-write.js +184 -0
  398. package/dist/tools/verify-plan-execution.js +295 -0
  399. package/dist/tools/web-fetch-injection-scanner.js +207 -0
  400. package/dist/tools/web-fetch.js +195 -10
  401. package/dist/tools/web-search.js +458 -0
  402. package/dist/tui/agent-progress-card.js +111 -0
  403. package/dist/tui/agent-tree.js +22 -1
  404. package/dist/tui/ask-modal.js +14 -14
  405. package/dist/tui/ask-user-question-chips.js +315 -0
  406. package/dist/tui/ask-user-question-prompt.js +203 -0
  407. package/dist/tui/compact-banner.js +81 -0
  408. package/dist/tui/conversation-pane.js +85 -11
  409. package/dist/tui/cost-table.js +111 -0
  410. package/dist/tui/device-flow.js +2 -2
  411. package/dist/tui/doctor-table.js +46 -0
  412. package/dist/tui/feedback-prompt.js +156 -0
  413. package/dist/tui/input-box.js +247 -32
  414. package/dist/tui/login-picker.js +3 -3
  415. package/dist/tui/markdown-render.js +6 -6
  416. package/dist/tui/multi-file-diff-approval.js +375 -0
  417. package/dist/tui/onboarding-wizard.js +240 -0
  418. package/dist/tui/permissions-picker.js +86 -0
  419. package/dist/tui/render.js +36 -1
  420. package/dist/tui/repl-render.js +239 -25
  421. package/dist/tui/repl-splash-art.js +16 -16
  422. package/dist/tui/repl-splash-mascot.js +48 -24
  423. package/dist/tui/repl-splash.js +22 -22
  424. package/dist/tui/repl.js +125 -45
  425. package/dist/tui/slash-palette.js +6 -6
  426. package/dist/tui/splash.js +2 -2
  427. package/dist/tui/status-bar.js +109 -31
  428. package/dist/tui/status-table.js +7 -0
  429. package/dist/tui/stickers-art.js +136 -0
  430. package/dist/tui/style-table.js +28 -0
  431. package/dist/tui/theme-table.js +29 -0
  432. package/dist/tui/thinking-spinner.js +123 -0
  433. package/dist/tui/tool-stream-pane.js +53 -4
  434. package/dist/tui/update-banner.js +27 -2
  435. package/dist/tui/vim-input.js +267 -0
  436. package/dist/tui/welcome-banner.js +107 -0
  437. package/dist/tui/welcome-data.js +293 -0
  438. package/dist/tui/workspace-context.js +2 -2
  439. package/package.json +21 -5
  440. package/test/scenarios/codegen-create-file.scenario.txt +13 -0
  441. package/test/scenarios/compact-force.scenario.txt +12 -0
  442. package/test/scenarios/identity.scenario.txt +11 -0
  443. package/test/scenarios/persona-handoff.scenario.txt +12 -0
  444. package/test/scenarios/walkback.scenario.txt +12 -0
  445. package/dist/core/engine/compaction-hook.js +0 -154
@@ -2,7 +2,7 @@ import { mkdirSync, renameSync, rmSync, writeFileSync, } from 'node:fs';
2
2
  import { relative, resolve, sep } from 'node:path';
3
3
  import { z } from 'zod';
4
4
  /**
5
- * α6.8 `pugi plan --decompose <idea>` — pattern ported (not copied) from
5
+ * `pugi plan --decompose <idea>` — pattern ported (not copied) from
6
6
  * the piercelamb/deep-project plugin (MIT-licensed, 134 stars). The Deep
7
7
  * Trilogy splits a high-level idea into focused components, each backed
8
8
  * by its own `spec.md`, plus a `project-manifest.md` with the dependency
@@ -31,17 +31,17 @@ import { z } from 'zod';
31
31
  *
32
32
  * Design choices anchored by the Deep-Project pattern + our own
33
33
  * planning ergonomics:
34
- * - 3-7 component sweet spot: more than 7 produces unreviewable
35
- * spaghetti, fewer than 3 wastes the decomposition step. The
36
- * schema enforces this hard so the prompt promise matches the
37
- * contract.
38
- * - `dependsOn` references component names verbatim — the writer
39
- * enforces uniqueness and resolves the DAG, so a typo there
40
- * fails loud at parse time rather than producing a silently
41
- * wrong manifest.
42
- * - JSON fence is mandatory so the parser can `slice` it out
43
- * deterministically; loose JSON-in-prose extractions are
44
- * brittle and trigger false positives.
34
+ * - 3-7 component sweet spot: more than 7 produces unreviewable
35
+ * spaghetti, fewer than 3 wastes the decomposition step. The
36
+ * schema enforces this hard so the prompt promise matches the
37
+ * contract.
38
+ * - `dependsOn` references component names verbatim — the writer
39
+ * enforces uniqueness and resolves the DAG, so a typo there
40
+ * fails loud at parse time rather than producing a silently
41
+ * wrong manifest.
42
+ * - JSON fence is mandatory so the parser can `slice` it out
43
+ * deterministically; loose JSON-in-prose extractions are
44
+ * brittle and trigger false positives.
45
45
  */
46
46
  export const DECOMPOSE_PROMPT_SUFFIX = [
47
47
  '',
@@ -57,14 +57,14 @@ export const DECOMPOSE_PROMPT_SUFFIX = [
57
57
  '',
58
58
  '```json',
59
59
  '{',
60
- ' "components": [',
61
- ' {',
62
- ' "name": "kebab-case-name",',
63
- ' "summary": "one-sentence headline",',
64
- ' "spec": "multi-line spec body — what to build, acceptance criteria, files touched",',
65
- ' "dependsOn": ["other-component-name", "..."]',
66
- ' }',
67
- ' ]',
60
+ ' "components": [',
61
+ ' {',
62
+ ' "name": "kebab-case-name",',
63
+ ' "summary": "one-sentence headline",',
64
+ ' "spec": "multi-line spec body — what to build, acceptance criteria, files touched",',
65
+ ' "dependsOn": ["other-component-name", "..."]',
66
+ ' }',
67
+ ' ]',
68
68
  '}',
69
69
  '```',
70
70
  '',
@@ -419,13 +419,13 @@ export function formatManifest(input) {
419
419
  lines.push('graph TD');
420
420
  for (const component of decomposition.components) {
421
421
  const safeNode = sanitizeMermaidId(component.name);
422
- lines.push(` ${safeNode}["${component.name}"]`);
422
+ lines.push(` ${safeNode}["${component.name}"]`);
423
423
  }
424
424
  for (const component of decomposition.components) {
425
425
  const toNode = sanitizeMermaidId(component.name);
426
426
  for (const dep of component.dependsOn) {
427
427
  const fromNode = sanitizeMermaidId(dep);
428
- lines.push(` ${fromNode} --> ${toNode}`);
428
+ lines.push(` ${fromNode} --> ${toNode}`);
429
429
  }
430
430
  }
431
431
  lines.push('```');
@@ -0,0 +1,272 @@
1
+ /**
2
+ * Double-press Ctrl+C exit guard for the Pugi CLI top-level lifecycle.
3
+ *
4
+ * # Problem
5
+ *
6
+ * Operator dogfood reported a single Ctrl+C exits the REPL / headless
7
+ * loop. Operators expect a forgiving "press again to confirm" gesture
8
+ * — the same shell convention `^C ^C` already used by the per-engine
9
+ * task abort path in `runtime/cli.ts` (`runEngineTask`). Without the
10
+ * gesture at the top level, a stray Ctrl+C while typing a slash command
11
+ * or scrolling a transcript kills the session and any in-memory state
12
+ * the operator hasn't synced yet.
13
+ *
14
+ * # Behavior
15
+ *
16
+ * 1. **First Ctrl+C** — emit a one-line stderr prompt
17
+ * "Press Ctrl+C again to exit (within 2s), or any key to continue."
18
+ * Arm a 2-second window timer. Install a one-shot `stdin.once('data', …)`
19
+ * so the FIRST keystroke after the prompt cancels the exit gesture.
20
+ * 2. **Second Ctrl+C inside window** — flush log streams, persist a
21
+ * minimal session-state snapshot to `~/.pugi/session-state.json`,
22
+ * and exit with code 0. The state file is best-effort: any write
23
+ * error is swallowed so the operator's exit is never blocked on
24
+ * a disk hiccup.
25
+ * 3. **Other key inside window** — clear the timer, drop the prompt,
26
+ * emit "Exit cancelled." to stderr, and resume normally.
27
+ * 4. **Window expires** — clear `lastSigintTs`. A subsequent isolated
28
+ * Ctrl+C is treated as a NEW first press, not a confirmation.
29
+ * 5. **Headless mode** — when `process.stdin.isTTY === false`, the
30
+ * guard switches strategy: it closes stdin (so any `for await rl`
31
+ * loop in `headless-repl.ts` unwinds naturally), emits a
32
+ * `session-end` envelope to stdout (single JSON line, matches
33
+ * the envelope schema), and exits 0. Stdin can't deliver "any
34
+ * other key" when it's not a TTY, so the double-press dance is
35
+ * skipped in this mode.
36
+ *
37
+ * # Coexistence with the per-engine-run handler
38
+ *
39
+ * `runEngineTask` in `runtime/cli.ts` installs its OWN
40
+ * `process.on('SIGINT', …)` for the duration of an engine dispatch
41
+ * (lines around 6233). That handler aborts the in-flight turn on the
42
+ * first press and exits 130 on a second press inside its OWN 2s
43
+ * window. Both handlers receive every SIGINT — Node delivers signals
44
+ * to every listener.
45
+ *
46
+ * To avoid a double prompt while an engine turn is running, this
47
+ * guard checks `process.listenerCount('SIGINT')` at the start of its
48
+ * handler: if any other listener is attached (i.e. an engine run owns
49
+ * the foreground), we step aside and let that handler drive the UX.
50
+ * The engine handler's "press again to exit" prompt already covers
51
+ * the abort-then-quit story for that window. When the engine run
52
+ * unwinds, it detaches its listener and the REPL-level guard regains
53
+ * control.
54
+ *
55
+ * # Why module-scope, not per-call closure
56
+ *
57
+ * The press-count state must survive between two distinct SIGINT
58
+ * deliveries. A closure-scoped flag would reset on the second SIGINT
59
+ * because Node invokes the handler in a fresh microtask each time.
60
+ * Module-scope `let` is the simplest store that gives us cross-press
61
+ * persistence without leaking to other files.
62
+ *
63
+ * # Testability
64
+ *
65
+ * `installSigintGuard()` takes an optional `SigintGuardOptions` bag
66
+ * so the spec can inject:
67
+ * - a fake `stdin` (for "any key cancels"),
68
+ * - a fake `stdout` / `stderr` sink,
69
+ * - a `now()` clock seam,
70
+ * - a `setTimeout` / `clearTimeout` pair,
71
+ * - a `exit(code)` seam (the test never lets the real process exit),
72
+ * - and a `persistSessionState(payload)` injection so the spec
73
+ * observes the persisted snapshot without touching `~/`.
74
+ *
75
+ * In production all seams default to the real Node primitives.
76
+ */
77
+ import { writeFile, mkdir } from 'node:fs/promises';
78
+ import { homedir } from 'node:os';
79
+ import { resolve as resolvePath, dirname } from 'node:path';
80
+ /**
81
+ * Default double-press window. Matches the per-engine-run handler so
82
+ * operators see one consistent timing rule across the CLI.
83
+ */
84
+ export const SIGINT_DOUBLE_PRESS_WINDOW_MS = 2000;
85
+ /**
86
+ * Default location for the session-state snapshot the guard writes on
87
+ * a confirmed exit. Resolved at call time so `homedir()` is read late
88
+ * enough to honor a test override of the `HOME` env var.
89
+ */
90
+ export function defaultSessionStatePath(home = homedir()) {
91
+ return resolvePath(home, '.pugi', 'session-state.json');
92
+ }
93
+ /**
94
+ * Default JSON-file persister. Best-effort: a failed write is logged
95
+ * to stderr (so the operator notices in debug runs) and then
96
+ * swallowed — the exit must not block on filesystem health.
97
+ */
98
+ async function defaultPersist(snapshot) {
99
+ const filePath = defaultSessionStatePath();
100
+ try {
101
+ await mkdir(dirname(filePath), { recursive: true });
102
+ await writeFile(filePath, JSON.stringify(snapshot, null, 2), {
103
+ mode: 0o600,
104
+ encoding: 'utf8',
105
+ });
106
+ }
107
+ catch (error) {
108
+ const message = error instanceof Error ? error.message : String(error);
109
+ process.stderr.write(`pugi: session-state write failed: ${message}\n`);
110
+ }
111
+ }
112
+ /**
113
+ * Default SIGINT subscription wires the handler onto `process` and
114
+ * returns an unsubscribe closure that detaches it. We use `.on` (not
115
+ * `.once`) so the handler stays attached across multiple presses
116
+ * within the same process lifetime.
117
+ */
118
+ function defaultOnSigint(handler) {
119
+ process.on('SIGINT', handler);
120
+ return () => {
121
+ process.off('SIGINT', handler);
122
+ };
123
+ }
124
+ /**
125
+ * Install the double-press Ctrl+C exit guard. Returns a handle whose
126
+ * `uninstall()` detaches the SIGINT listener and clears any pending
127
+ * window timer; production never calls it.
128
+ *
129
+ * This function is idempotent: calling it a second time installs a
130
+ * NEW guard alongside the old one, which would cause duplicate
131
+ * prompts. The caller (cli.ts main entry) MUST call it exactly once,
132
+ * at the very top of the run. Tests that need multiple installs are
133
+ * expected to `uninstall()` between scenarios.
134
+ */
135
+ export function installSigintGuard(options = {}) {
136
+ const stdin = options.stdin ?? process.stdin;
137
+ const stderr = options.stderr ?? process.stderr;
138
+ const stdout = options.stdout ?? process.stdout;
139
+ const now = options.now ?? Date.now;
140
+ const setTimer = options.setTimer ?? ((fn, ms) => setTimeout(fn, ms));
141
+ const clearTimer = options.clearTimer ?? ((handle) => clearTimeout(handle));
142
+ const exit = options.exit ?? ((code) => process.exit(code));
143
+ const persist = options.persistSessionState ?? defaultPersist;
144
+ const subscribe = options.onSigint ?? defaultOnSigint;
145
+ const isHeadless = options.isHeadless ?? (() => stdin.isTTY !== true);
146
+ const windowMs = options.windowMs ?? SIGINT_DOUBLE_PRESS_WINDOW_MS;
147
+ // State scoped to THIS install. Each call to `installSigintGuard`
148
+ // gets a fresh closure so concurrent tests do not bleed into each
149
+ // other. Production calls the function once at the top of the run.
150
+ let lastSigintTs = null;
151
+ let pendingTimer = null;
152
+ let pendingDataListener = null;
153
+ const resetWindow = () => {
154
+ lastSigintTs = null;
155
+ if (pendingTimer !== null) {
156
+ clearTimer(pendingTimer);
157
+ pendingTimer = null;
158
+ }
159
+ if (pendingDataListener !== null) {
160
+ // Detach via the same instance we attached; never the typed
161
+ // overload that re-binds 'data' to all listeners.
162
+ stdin.removeListener('data', pendingDataListener);
163
+ pendingDataListener = null;
164
+ }
165
+ };
166
+ const performExit = (reason) => {
167
+ const snapshot = {
168
+ exitedAt: new Date(now()).toISOString(),
169
+ reason,
170
+ cwd: process.cwd(),
171
+ pid: process.pid,
172
+ };
173
+ // Fire-and-forget persistence: we attempt to flush, but do not
174
+ // block the exit on the result. The promise is observed only to
175
+ // suppress unhandled-rejection noise in the test runner.
176
+ void persist(snapshot).catch(() => {
177
+ /* defaultPersist already logged */
178
+ });
179
+ exit(0);
180
+ };
181
+ const handleHeadless = () => {
182
+ // In headless mode we never prompt — the operator (or harness)
183
+ // has no keyboard to confirm with. We emit one final
184
+ // session-end envelope so any line-buffered consumer sees a
185
+ // clean terminator, close stdin so the `for await rl` loop in
186
+ // headless-repl.ts unwinds, and exit 0.
187
+ const envelope = {
188
+ kind: 'session-end',
189
+ body: JSON.stringify({ reason: 'sigint' }),
190
+ ts: now(),
191
+ };
192
+ stdout.write(`${JSON.stringify(envelope)}\n`);
193
+ // Best-effort stdin close so any in-flight readline loop terminates.
194
+ // Some stream implementations (e.g. test doubles) lack `.destroy`;
195
+ // guard the call so we never throw out of a signal handler.
196
+ const stdinAsAny = stdin;
197
+ try {
198
+ if (typeof stdinAsAny.destroy === 'function') {
199
+ stdinAsAny.destroy();
200
+ }
201
+ else if (typeof stdinAsAny.pause === 'function') {
202
+ stdinAsAny.pause();
203
+ }
204
+ }
205
+ catch {
206
+ /* ignore — destroy/pause is opportunistic */
207
+ }
208
+ performExit('sigint-headless');
209
+ };
210
+ const handleInteractive = () => {
211
+ const ts = now();
212
+ if (lastSigintTs !== null && ts - lastSigintTs <= windowMs) {
213
+ // Confirmed double-press. Drop the prompt artifacts, persist
214
+ // state, exit clean. resetWindow() handles the listener
215
+ // cleanup so a stray `data` event after exit does not fire.
216
+ resetWindow();
217
+ stderr.write('\npugi: exiting (^C^C confirmed).\n');
218
+ performExit('sigint-double-press');
219
+ return;
220
+ }
221
+ // First press — arm the window.
222
+ lastSigintTs = ts;
223
+ stderr.write('\nPress Ctrl+C again to exit (within 2s), or any key to continue.\n');
224
+ // Schedule a window-expiry reset. When the timer fires the
225
+ // operator's prior press no longer "counts" — the next ^C is
226
+ // treated as a fresh first press.
227
+ pendingTimer = setTimer(() => {
228
+ lastSigintTs = null;
229
+ pendingTimer = null;
230
+ if (pendingDataListener !== null) {
231
+ stdin.removeListener('data', pendingDataListener);
232
+ pendingDataListener = null;
233
+ }
234
+ }, windowMs);
235
+ // Install a one-shot 'data' listener so any keystroke other than
236
+ // a follow-up SIGINT cancels the exit gesture. We use
237
+ // `removeListener` after firing rather than `.once` because we
238
+ // also detach the listener from `resetWindow()` (the timer or
239
+ // a second SIGINT can both kill it).
240
+ const onData = (_chunk) => {
241
+ resetWindow();
242
+ stderr.write('Exit cancelled.\n');
243
+ };
244
+ pendingDataListener = onData;
245
+ stdin.on('data', onData);
246
+ };
247
+ const handler = () => {
248
+ // Coexistence guard: if another SIGINT listener is registered
249
+ // (e.g. the per-engine-run handler in runEngineTask), step aside
250
+ // and let it drive the UX. We count `> 1` because OUR handler
251
+ // is also in the list.
252
+ if (process.listenerCount('SIGINT') > 1 && !options.onSigint) {
253
+ // Drop any state we'd accumulated so an interactive prompt
254
+ // after the engine run starts from a clean slate.
255
+ resetWindow();
256
+ return;
257
+ }
258
+ if (isHeadless()) {
259
+ handleHeadless();
260
+ return;
261
+ }
262
+ handleInteractive();
263
+ };
264
+ const unsubscribe = subscribe(handler);
265
+ return {
266
+ uninstall: () => {
267
+ resetWindow();
268
+ unsubscribe();
269
+ },
270
+ };
271
+ }
272
+ //# sourceMappingURL=sigint-guard.js.map
@@ -0,0 +1,195 @@
1
+ /**
2
+ * Streaming event renderer (Trust Sprint item 5).
3
+ *
4
+ * Subscribes to the engine adapter's `streamEmitter` and writes
5
+ * human-readable progress lines to stderr while the dispatch is in
6
+ * flight. Solves Codex dogfood 2026-06-04 finding: `pugi
7
+ * code/fix/build/plan` were silent until the full dispatch
8
+ * completed, which produced an empty terminal for 20-30s and made
9
+ * customers think the CLI had hung.
10
+ *
11
+ * Design constraints:
12
+ *
13
+ * 1. STDERR ONLY. The CLI's machine-readable contract is "JSON on
14
+ * stdout, prose on stderr". Streaming on stdout would break the
15
+ * JSON envelope for `pugi code --json`.
16
+ *
17
+ * 2. No engine modifications. We attach a listener to the existing
18
+ * `streamEmitter` that the engine already emits to (used by the
19
+ * headless mode). The engine adapter file is owned by the other
20
+ * agent on PUGI-VERIFY-GATE and intentionally NOT touched here.
21
+ *
22
+ * 3. TTY-only by default. CI / pipes / JSON consumers get a clean
23
+ * stream. Explicit `--stream` overrides to force progress lines
24
+ * on; `--no-stream` overrides off.
25
+ *
26
+ * 4. Compact. One line per tool call (start) and one line per
27
+ * finish. Text deltas are aggregated to a single "thinking..."
28
+ * pulse so we don't paint the terminal with token-level noise.
29
+ */
30
+ /**
31
+ * Decide whether to enable the streaming renderer.
32
+ *
33
+ * - explicit=true → ON regardless of TTY.
34
+ * - explicit=false → OFF regardless of TTY.
35
+ * - explicit=null → ON when stderr is a TTY, OFF otherwise.
36
+ *
37
+ * Centralised so the CLI dispatcher and tests share one rule.
38
+ */
39
+ export function shouldStream(opts) {
40
+ if (opts.explicit === true)
41
+ return true;
42
+ if (opts.explicit === false)
43
+ return false;
44
+ return opts.isTty;
45
+ }
46
+ /**
47
+ * Attach a stderr-based event renderer to the engine emitter.
48
+ *
49
+ * Returns a handle whose `enabled` field is true when the renderer
50
+ * actually attached. When disabled (non-TTY or explicit-off) the
51
+ * returned handle is a no-op so callers don't need to branch.
52
+ */
53
+ export function attachStreamRenderer(emitter, opts) {
54
+ const enabled = shouldStream({ isTty: opts.isTty, explicit: opts.explicit });
55
+ if (!enabled) {
56
+ return { enabled: false, detach: () => { } };
57
+ }
58
+ const write = opts.write ?? ((line) => process.stderr.write(line));
59
+ // Aggregate text deltas so we don't paint a line per token. We
60
+ // print one "thinking..." pulse every ~500ms while text is
61
+ // streaming so the operator sees the loop is alive without a wall
62
+ // of partial sentences.
63
+ let pendingTextChars = 0;
64
+ let lastTextPulseAt = 0;
65
+ // Aggregate thinking deltas the same way. Separate budget so a
66
+ // chatty reasoning trace doesn't blow out the visible-text counter.
67
+ let pendingThinkingChars = 0;
68
+ let lastThinkingPulseAt = 0;
69
+ // Cache the running tool calls so the "end" line can report a
70
+ // duration without forcing the producer to round-trip the start
71
+ // timestamp through the discriminated union.
72
+ const toolStartedAt = new Map();
73
+ const handler = (event) => {
74
+ switch (event.type) {
75
+ case 'status': {
76
+ // Only surface the high-signal status frames. Turn-boundary
77
+ // frames are noise for a live operator; the tool-level lines
78
+ // already tell them what's happening. Dispatch start /
79
+ // budget / abort frames are worth one line each.
80
+ const msg = event.message;
81
+ if (msg.startsWith('dispatch_start') ||
82
+ msg.startsWith('budget') ||
83
+ msg.startsWith('aborted') ||
84
+ msg.startsWith('cancelled')) {
85
+ write(`pugi: ${msg}\n`);
86
+ }
87
+ return;
88
+ }
89
+ case 'tool.start': {
90
+ toolStartedAt.set(event.callId, Date.now());
91
+ const argsBrief = briefArgs(event.arguments);
92
+ write(`pugi: tool ${event.name}(${argsBrief}) running\n`);
93
+ return;
94
+ }
95
+ case 'tool.end': {
96
+ const startedAt = toolStartedAt.get(event.callId);
97
+ toolStartedAt.delete(event.callId);
98
+ const ms = startedAt ? Date.now() - startedAt : null;
99
+ const glyph = event.ok ? 'ok' : 'fail';
100
+ const summary = (event.summary ?? '').slice(0, 80).replace(/\s+/g, ' ').trim();
101
+ const tail = ms !== null ? ` (${ms}ms)` : '';
102
+ write(`pugi: tool ${glyph}${tail}${summary ? `: ${summary}` : ''}\n`);
103
+ return;
104
+ }
105
+ case 'tool.delta': {
106
+ // Most bash tools emit a single delta with the entire stdout
107
+ // tail. We surface the first non-empty chunk only, capped to
108
+ // a single line, so the operator gets a peek without
109
+ // flooding.
110
+ const chunk = (event.chunk ?? '').replace(/\s+/g, ' ').trim();
111
+ if (chunk.length === 0)
112
+ return;
113
+ const oneLine = chunk.slice(0, 80);
114
+ write(`pugi: stdout: ${oneLine}\n`);
115
+ return;
116
+ }
117
+ case 'text.delta': {
118
+ pendingTextChars += (event.chunk ?? '').length;
119
+ const now = Date.now();
120
+ if (now - lastTextPulseAt >= 500 && pendingTextChars >= 40) {
121
+ lastTextPulseAt = now;
122
+ pendingTextChars = 0;
123
+ write(`pugi: writing reply...\n`);
124
+ }
125
+ return;
126
+ }
127
+ case 'thinking.start':
128
+ write(`pugi: thinking...\n`);
129
+ lastThinkingPulseAt = Date.now();
130
+ return;
131
+ case 'thinking.delta': {
132
+ pendingThinkingChars += (event.chunk ?? '').length;
133
+ const now = Date.now();
134
+ if (now - lastThinkingPulseAt >= 1500 && pendingThinkingChars >= 80) {
135
+ lastThinkingPulseAt = now;
136
+ pendingThinkingChars = 0;
137
+ write(`pugi: still thinking...\n`);
138
+ }
139
+ return;
140
+ }
141
+ case 'thinking.end':
142
+ // Quiet — text.delta will pick up the visible answer next.
143
+ return;
144
+ default:
145
+ return;
146
+ }
147
+ };
148
+ emitter.on('event', handler);
149
+ return {
150
+ enabled: true,
151
+ detach: () => {
152
+ emitter.off('event', handler);
153
+ },
154
+ };
155
+ }
156
+ /**
157
+ * Render a tool's JSON-serialised arguments to a single-line brief.
158
+ * Keep this very cheap — runs on every tool.start event.
159
+ */
160
+ function briefArgs(serialised) {
161
+ if (!serialised || serialised === '{}')
162
+ return '';
163
+ try {
164
+ const parsed = JSON.parse(serialised);
165
+ // Prioritise the most signal-bearing field name for known tools.
166
+ const path = pickStringField(parsed, ['path', 'file', 'file_path', 'target', 'cwd']);
167
+ if (path)
168
+ return path.slice(0, 60);
169
+ const command = pickStringField(parsed, ['command', 'cmd', 'query', 'pattern']);
170
+ if (command)
171
+ return command.slice(0, 60);
172
+ // Fallback — surface the first scalar field so we surface
173
+ // something rather than an empty string.
174
+ for (const [, value] of Object.entries(parsed)) {
175
+ if (typeof value === 'string')
176
+ return value.slice(0, 60);
177
+ if (typeof value === 'number' || typeof value === 'boolean') {
178
+ return String(value);
179
+ }
180
+ }
181
+ return '';
182
+ }
183
+ catch {
184
+ return serialised.slice(0, 60);
185
+ }
186
+ }
187
+ function pickStringField(obj, keys) {
188
+ for (const key of keys) {
189
+ const value = obj[key];
190
+ if (typeof value === 'string' && value.length > 0)
191
+ return value;
192
+ }
193
+ return null;
194
+ }
195
+ //# sourceMappingURL=stream-renderer.js.map
@@ -1,5 +1,5 @@
1
1
  /**
2
- * Update check + install-method detection — Sprint α6.2.
2
+ * Update check + install-method detection — Sprint .
3
3
  *
4
4
  * Shows the REPL operator a one-shot banner at startup when the npm
5
5
  * registry advertises a `@pugi/cli` version newer than what is running.
@@ -8,19 +8,19 @@
8
8
  *
9
9
  * Constraints baked into the spec:
10
10
  *
11
- * - **Default-quiet.** Network errors, cache misses, and the
12
- * `PUGI_SKIP_UPDATE_BANNER=1` env var all return `null` so the
13
- * REPL renders unchanged.
14
- * - **24h cache.** The check runs at most once per day, persisted to
15
- * `~/.pugi/update-check.json`. Repeated REPL launches in the same
16
- * day skip the registry call entirely.
17
- * - **3s timeout.** A slow registry never blocks REPL startup; the
18
- * undici request is aborted after 3 seconds and treated as a
19
- * silent miss.
20
- * - **Pure helpers, IO at the edge.** Detection, comparison, cache
21
- * decode/encode are exported as pure functions with explicit env /
22
- * home / now / fetch seams so the spec's 8 tests can drive every
23
- * branch without touching the network or the real filesystem.
11
+ * - **Default-quiet.** Network errors, cache misses, and the
12
+ * `PUGI_SKIP_UPDATE_BANNER=1` env var all return `null` so the
13
+ * REPL renders unchanged.
14
+ * - **24h cache.** The check runs at most once per day, persisted to
15
+ * `~/.pugi/update-check.json`. Repeated REPL launches in the same
16
+ * day skip the registry call entirely.
17
+ * - **3s timeout.** A slow registry never blocks REPL startup; the
18
+ * undici request is aborted after 3 seconds and treated as a
19
+ * silent miss.
20
+ * - **Pure helpers, IO at the edge.** Detection, comparison, cache
21
+ * decode/encode are exported as pure functions with explicit env /
22
+ * home / now / fetch seams so the spec's 8 tests can drive every
23
+ * branch without touching the network or the real filesystem.
24
24
  */
25
25
  import { existsSync, mkdirSync, readFileSync, writeFileSync } from 'node:fs';
26
26
  import { homedir } from 'node:os';
@@ -120,16 +120,16 @@ export function isCacheFresh(record, nowMs) {
120
120
  * `0.1.0-alpha.6` vs `0.1.0-alpha.7` / `0.1.0-beta.1` / `1.0.0`).
121
121
  *
122
122
  * Returns:
123
- * -1 if `a < b`
124
- * 0 if `a === b`
125
- * 1 if `a > b`
123
+ * -1 if `a < b`
124
+ * 0 if `a === b`
125
+ * 1 if `a > b`
126
126
  *
127
127
  * Rules follow semver §11:
128
- * - Major.Minor.Patch compared numerically left-to-right.
129
- * - A version with a prerelease tag is LOWER than the same version
130
- * without one (`1.0.0-alpha` < `1.0.0`).
131
- * - Prerelease identifiers compared dot-by-dot; numeric chunks
132
- * numerically, mixed chunks ASCII-lexicographic.
128
+ * - Major.Minor.Patch compared numerically left-to-right.
129
+ * - A version with a prerelease tag is LOWER than the same version
130
+ * without one (`1.0.0-alpha` < `1.0.0`).
131
+ * - Prerelease identifiers compared dot-by-dot; numeric chunks
132
+ * numerically, mixed chunks ASCII-lexicographic.
133
133
  */
134
134
  export function compareVersions(a, b) {
135
135
  const [coreA, preA] = splitVersion(a);
@@ -233,11 +233,11 @@ export async function fetchLatestVersion(fetcher = request) {
233
233
  * declines to fire, otherwise `null`.
234
234
  *
235
235
  * Silence rules (any one of these short-circuits to `null`):
236
- * - `cliSkip === true` (operator passed `--no-update-check`).
237
- * - `env.PUGI_SKIP_UPDATE_BANNER === '1'`.
238
- * - `isTty === false` (CI / piped / scripted invocation).
239
- * - Cache + network both miss — silent skip per spec.
240
- * - `installed >= latest` — no upgrade to advertise.
236
+ * - `cliSkip === true` (operator passed `--no-update-check`).
237
+ * - `env.PUGI_SKIP_UPDATE_BANNER === '1'`.
238
+ * - `isTty === false` (CI / piped / scripted invocation).
239
+ * - Cache + network both miss — silent skip per spec.
240
+ * - `installed >= latest` — no upgrade to advertise.
241
241
  */
242
242
  export async function checkForUpdate(options) {
243
243
  const env = options.env ?? process.env;
@@ -288,7 +288,7 @@ export function upgradeCommand(method) {
288
288
  return 'curl -fsSL https://install.pugi.io | sh';
289
289
  case 'unknown':
290
290
  default:
291
- return 'npm install -g @pugi/cli@latest # or your install method';
291
+ return 'npm install -g @pugi/cli@latest # or your install method';
292
292
  }
293
293
  }
294
294
  //# sourceMappingURL=update-check.js.map