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

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 (411) 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 +1792 -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/cron.js +433 -0
  351. package/dist/tools/enter-worktree.js +250 -0
  352. package/dist/tools/exit-worktree.js +147 -0
  353. package/dist/tools/file-tools.js +161 -44
  354. package/dist/tools/lsp-tools.js +377 -1
  355. package/dist/tools/mcp-tool.js +260 -0
  356. package/dist/tools/multi-edit.js +361 -0
  357. package/dist/tools/powershell.js +268 -0
  358. package/dist/tools/registry.js +99 -4
  359. package/dist/tools/skill-tool.js +96 -0
  360. package/dist/tools/sleep.js +99 -0
  361. package/dist/tools/synthetic-output.js +133 -0
  362. package/dist/tools/tasks.js +208 -0
  363. package/dist/tools/todo-write.js +184 -0
  364. package/dist/tools/verify-plan-execution.js +295 -0
  365. package/dist/tools/web-fetch-injection-scanner.js +207 -0
  366. package/dist/tools/web-fetch.js +195 -10
  367. package/dist/tools/web-search.js +458 -0
  368. package/dist/tui/agent-progress-card.js +111 -0
  369. package/dist/tui/agent-tree.js +11 -1
  370. package/dist/tui/ask-modal.js +14 -14
  371. package/dist/tui/ask-user-question-chips.js +315 -0
  372. package/dist/tui/ask-user-question-prompt.js +203 -0
  373. package/dist/tui/compact-banner.js +81 -0
  374. package/dist/tui/conversation-pane.js +85 -11
  375. package/dist/tui/cost-table.js +111 -0
  376. package/dist/tui/device-flow.js +2 -2
  377. package/dist/tui/doctor-table.js +46 -0
  378. package/dist/tui/feedback-prompt.js +156 -0
  379. package/dist/tui/input-box.js +247 -32
  380. package/dist/tui/login-picker.js +3 -3
  381. package/dist/tui/markdown-render.js +6 -6
  382. package/dist/tui/multi-file-diff-approval.js +375 -0
  383. package/dist/tui/onboarding-wizard.js +240 -0
  384. package/dist/tui/permissions-picker.js +86 -0
  385. package/dist/tui/render.js +36 -1
  386. package/dist/tui/repl-render.js +176 -25
  387. package/dist/tui/repl-splash-art.js +16 -16
  388. package/dist/tui/repl-splash-mascot.js +48 -24
  389. package/dist/tui/repl-splash.js +22 -22
  390. package/dist/tui/repl.js +125 -45
  391. package/dist/tui/slash-palette.js +6 -6
  392. package/dist/tui/splash.js +2 -2
  393. package/dist/tui/status-bar.js +109 -31
  394. package/dist/tui/status-table.js +7 -0
  395. package/dist/tui/stickers-art.js +136 -0
  396. package/dist/tui/style-table.js +28 -0
  397. package/dist/tui/theme-table.js +29 -0
  398. package/dist/tui/thinking-spinner.js +123 -0
  399. package/dist/tui/tool-stream-pane.js +53 -4
  400. package/dist/tui/update-banner.js +27 -2
  401. package/dist/tui/vim-input.js +267 -0
  402. package/dist/tui/welcome-banner.js +107 -0
  403. package/dist/tui/welcome-data.js +293 -0
  404. package/dist/tui/workspace-context.js +2 -2
  405. package/package.json +31 -16
  406. package/test/scenarios/codegen-create-file.scenario.txt +13 -0
  407. package/test/scenarios/compact-force.scenario.txt +12 -0
  408. package/test/scenarios/identity.scenario.txt +12 -0
  409. package/test/scenarios/persona-handoff.scenario.txt +12 -0
  410. package/test/scenarios/walkback.scenario.txt +12 -0
  411. package/dist/core/engine/compaction-hook.js +0 -154
@@ -0,0 +1,184 @@
1
+ /**
2
+ * `pugi feedback` + `/feedback` slash — .
3
+ *
4
+ * In-CLI feedback collector. Parity with the upstream tool's `/feedback`
5
+ * built-in. The operator never has to leave the terminal to file a
6
+ * bug / feature / general comment / praise. The wizard collects:
7
+ *
8
+ * 1. category (bug / feature / general / praise)
9
+ * 2. rating (1-5)
10
+ * 3. comment (multi-line free text)
11
+ * 4. optional redacted session context (last 5 turns)
12
+ * 5. confirm (final y/n)
13
+ *
14
+ * # Module contract
15
+ *
16
+ * - This file owns the WIRING from the CLI surface (TTY mount,
17
+ * non-TTY JSON, slash dispatcher) to the queue + submitter
18
+ * modules. The corpus + redactor + queue persistence live in
19
+ * `core/feedback/{queue.ts,submitter.ts}`. The Ink prompt lives
20
+ * in `tui/feedback-prompt.tsx`. Both have zero coupling to the
21
+ * CLI dispatch surface.
22
+ *
23
+ * - `runFeedbackCommand` is the single entry point. Both the top-
24
+ * level `pugi feedback` handler в `runtime/cli.ts` AND the in-REPL
25
+ * `/feedback` slash dispatcher call it. The function returns the
26
+ * resolved `FeedbackRunResult` so the slash dispatcher can route
27
+ * the outcome message to the REPL's system pane without re-prompting.
28
+ *
29
+ * - Exit code is ALWAYS 0. Feedback is a brand surface — never a
30
+ * gate. Failures land as result variants; the wrapper never
31
+ * turns a network blip into a non-zero shell exit.
32
+ *
33
+ * - The random-source-style test seam: the run helper accepts an
34
+ * `interactive` flag that the spec sets to false, plus an injected
35
+ * `draft` so unit tests can drive the submit + queue branches
36
+ * without mounting Ink.
37
+ */
38
+ import { existsSync, readFileSync } from 'node:fs';
39
+ import { resolve } from 'node:path';
40
+ import { enqueueFeedback, feedbackQueuePath, flushFeedbackQueue, } from '../../core/feedback/queue.js';
41
+ import { feedbackSubmitUrl, redactSessionContext, submitFeedback, } from '../../core/feedback/submitter.js';
42
+ /**
43
+ * Drive one feedback round. The function is async because of the
44
+ * submit round-trip, but everything else (queue write, redaction) is
45
+ * sync — no surprise concurrency.
46
+ */
47
+ export async function runFeedbackCommand(ctx) {
48
+ if (ctx.draft == null) {
49
+ return { kind: 'cancelled' };
50
+ }
51
+ const envelope = {
52
+ category: ctx.draft.category,
53
+ rating: ctx.draft.rating,
54
+ comment: ctx.draft.comment,
55
+ ts: new Date().toISOString(),
56
+ cliVersion: ctx.cliVersion,
57
+ ...(ctx.tier ? { tier: ctx.tier } : {}),
58
+ };
59
+ if (ctx.draft.includeSessionContext && ctx.sessionContext) {
60
+ const sc = ctx.sessionContext();
61
+ if (sc)
62
+ envelope.sessionContext = sc;
63
+ }
64
+ let result;
65
+ try {
66
+ result = await ctx.submit(envelope);
67
+ }
68
+ catch (err) {
69
+ // Defensive: a thrown submitter (should not happen — the live
70
+ // submitter catches everything) is treated as transient so the
71
+ // envelope lands in the queue.
72
+ const reason = err instanceof Error ? err.message : String(err);
73
+ result = { kind: 'transient', reason };
74
+ }
75
+ if (result.kind === 'ok') {
76
+ return { kind: 'submitted', envelope, httpStatus: result.httpStatus };
77
+ }
78
+ if (result.kind === 'transient') {
79
+ const path = enqueueFeedback(envelope, ctx.cwd);
80
+ return { kind: 'queued', envelope, path, reason: result.reason };
81
+ }
82
+ // permanent — log + drop
83
+ return {
84
+ kind: 'dropped',
85
+ envelope,
86
+ reason: result.reason,
87
+ httpStatus: result.httpStatus,
88
+ };
89
+ }
90
+ /**
91
+ * Render one human-readable toast for the operator. Centralised so the
92
+ * top-level `pugi feedback` shell handler + the in-REPL `/feedback`
93
+ * slash dispatcher agree on the copy.
94
+ */
95
+ export function renderFeedbackToast(result) {
96
+ switch (result.kind) {
97
+ case 'submitted':
98
+ return 'Feedback submitted. Thank you.';
99
+ case 'queued':
100
+ return `Feedback queued locally. Will sync on next online run. (${result.path})`;
101
+ case 'cancelled':
102
+ return 'Feedback cancelled. Nothing was sent.';
103
+ case 'dropped':
104
+ return `Feedback rejected by server (${result.httpStatus}): ${result.reason}. Not queued.`;
105
+ case 'noop':
106
+ return `No feedback collected: ${result.reason}`;
107
+ }
108
+ }
109
+ /**
110
+ * Background queue flush. Invoked silently on session start so any
111
+ * envelopes that landed during an offline run get drained when the
112
+ * operator next has connectivity. The function never throws — it
113
+ * returns the flush stats so the caller can log them at debug level.
114
+ */
115
+ export async function flushFeedbackQueueSilently(cwd, config) {
116
+ // Short-circuit when the queue file does not exist. Avoids a
117
+ // pointless `fs.stat` round-trip on every cold session start.
118
+ if (!existsSync(feedbackQueuePath(cwd))) {
119
+ return { attempted: 0, succeeded: 0, failed: 0 };
120
+ }
121
+ const result = await flushFeedbackQueue(cwd, async (env) => {
122
+ const r = await submitFeedback(env, config);
123
+ if (r.kind === 'ok')
124
+ return true;
125
+ if (r.kind === 'permanent') {
126
+ // Permanent failures are "done" from the queue's POV — they
127
+ // would never resolve on retry. Drop them so the queue does
128
+ // not grow without bound.
129
+ return true;
130
+ }
131
+ return false;
132
+ });
133
+ return {
134
+ attempted: result.attempted,
135
+ succeeded: result.succeeded,
136
+ failed: result.failed,
137
+ };
138
+ }
139
+ /**
140
+ * Re-exports — the spec imports these via the command module so the
141
+ * dependency graph in the test stays single-rooted.
142
+ */
143
+ export { feedbackQueuePath, feedbackSubmitUrl, redactSessionContext, submitFeedback, };
144
+ /**
145
+ * Read the persona conversation log if present. Best-effort: returns
146
+ * an empty list when the file is missing or malformed. The CLI's REPL
147
+ * persists transcripts via the session module at a canonical relative
148
+ * path under `.pugi/sessions/`. The shell-level `pugi feedback` does
149
+ * not have access to a live session, so it tries to pick up the most
150
+ * recent persisted one for the `--with-context` path.
151
+ *
152
+ * Intentionally tolerant — feedback works even with no transcript.
153
+ */
154
+ export function readMostRecentTranscript(cwd, options = {}) {
155
+ // The CLI may persist sessions in several places depending on the
156
+ // surface. We probe the conventional default; the spec drives the
157
+ // function via a fixture file instead of a live REPL.
158
+ const candidate = resolve(cwd, '.pugi', 'sessions', 'latest.jsonl');
159
+ if (!existsSync(candidate))
160
+ return [];
161
+ try {
162
+ const text = readFileSync(candidate, 'utf8');
163
+ const lines = text.split('\n').filter((l) => l.trim().length > 0);
164
+ const turns = [];
165
+ for (const line of lines) {
166
+ try {
167
+ const obj = JSON.parse(line);
168
+ if ((obj.role === 'user' || obj.role === 'assistant' || obj.role === 'system')
169
+ && typeof obj.text === 'string') {
170
+ turns.push({ role: obj.role, text: obj.text });
171
+ }
172
+ }
173
+ catch {
174
+ // skip malformed line
175
+ }
176
+ }
177
+ const cap = options.maxTurns ?? 5;
178
+ return turns.slice(-cap);
179
+ }
180
+ catch {
181
+ return [];
182
+ }
183
+ }
184
+ //# sourceMappingURL=feedback.js.map
@@ -0,0 +1,187 @@
1
+ /**
2
+ * `pugi hooks` — operator surface for user-config hooks (MVP).
3
+ *
4
+ * Two subcommands ship in the MVP:
5
+ *
6
+ * pugi hooks list List configured hooks per event.
7
+ * pugi hooks doctor Validate the config and surface any
8
+ * parse / schema errors.
9
+ *
10
+ * Both accept `--json` for scripted callers. Argument grammar is
11
+ * intentionally narrow — no `add` / `remove` / `test` subcommands in
12
+ * the MVP. Operators hand-edit `~/.pugi/hooks-mvp.json` for now.
13
+ *
14
+ * Exit codes:
15
+ * 0 -> happy path (no hooks OR config valid).
16
+ * 1 -> config present but invalid (only `doctor` returns this).
17
+ * 2 -> unknown subcommand / argument error.
18
+ *
19
+ * Brand voice: ASCII only.
20
+ */
21
+ import { ALL_HOOK_EVENTS_V2, defaultHooksMvpPath, loadHooksConfig, } from '../../core/hooks/index.js';
22
+ function parseFlags(args) {
23
+ const rest = [];
24
+ const flags = { json: false };
25
+ for (const arg of args) {
26
+ if (arg === '--json') {
27
+ flags.json = true;
28
+ continue;
29
+ }
30
+ rest.push(arg);
31
+ }
32
+ return { rest, flags };
33
+ }
34
+ /**
35
+ * Top-level dispatcher for `pugi hooks <subcommand>`. Returns the
36
+ * intended process exit code. `cli.ts` is expected to set
37
+ * `process.exitCode = <return value>` so error states propagate to
38
+ * scripted callers without throwing.
39
+ */
40
+ export async function runHooksCommand(args, ctx) {
41
+ const { rest, flags } = parseFlags(args);
42
+ const sub = rest[0];
43
+ if (!sub || sub === 'help' || sub === '--help') {
44
+ emitUsage(ctx, flags);
45
+ return sub ? 0 : 2;
46
+ }
47
+ if (sub === 'list') {
48
+ return runList(ctx, flags);
49
+ }
50
+ if (sub === 'doctor') {
51
+ return runDoctor(ctx, flags);
52
+ }
53
+ ctx.writeOutput({ ok: false, error: `unknown subcommand: ${sub}` }, `pugi hooks: unknown subcommand '${sub}'. Try 'pugi hooks --help'.`);
54
+ return 2;
55
+ }
56
+ function emitUsage(ctx, flags) {
57
+ const text = [
58
+ 'pugi hooks — user-config lifecycle hooks (MVP).',
59
+ '',
60
+ 'Subcommands:',
61
+ ' pugi hooks list Show hooks configured per event.',
62
+ ' pugi hooks doctor Validate ~/.pugi/hooks-mvp.json.',
63
+ '',
64
+ 'Flags:',
65
+ ' --json Emit a JSON envelope instead of human text.',
66
+ '',
67
+ 'Config file:',
68
+ ' ~/.pugi/hooks-mvp.json',
69
+ '',
70
+ 'Status:',
71
+ ' MVP — 2 events out of 8. Remaining events (PostToolUse,',
72
+ " UserPromptSubmit, Stop, SubagentStop, PreCompact, Notification)",
73
+ ' deferred to fast-follow PR.',
74
+ ].join('\n');
75
+ ctx.writeOutput({
76
+ ok: true,
77
+ command: 'hooks',
78
+ usage: text,
79
+ }, text);
80
+ if (flags.json) {
81
+ // The structured payload is already emitted by writeOutput when
82
+ // --json is on; nothing extra to do.
83
+ }
84
+ }
85
+ function runList(ctx, flags) {
86
+ let config;
87
+ try {
88
+ config = loadHooksConfig(ctx.configPath);
89
+ }
90
+ catch (error) {
91
+ const msg = error.message;
92
+ ctx.writeOutput({ ok: false, error: msg }, `pugi hooks list: ${msg}\nFix the config or remove the file. Run 'pugi hooks doctor' for details.`);
93
+ return 1;
94
+ }
95
+ const perEvent = {
96
+ SessionStart: [],
97
+ PreToolUse: [],
98
+ PostToolUse: [],
99
+ UserPromptSubmit: [],
100
+ Stop: [],
101
+ SubagentStop: [],
102
+ PreCompact: [],
103
+ Notification: [],
104
+ // PUGI-487 - worktree lifecycle events.
105
+ WorktreeCreate: [],
106
+ WorktreeRemove: [],
107
+ };
108
+ for (const event of ALL_HOOK_EVENTS_V2) {
109
+ perEvent[event] = config.list(event).map((entry) => ({
110
+ matcher: entry.matcher,
111
+ command: entry.command,
112
+ timeoutMs: entry.timeoutMs,
113
+ blocking: entry.blocking,
114
+ }));
115
+ }
116
+ const total = Object.values(perEvent).reduce((acc, list) => acc + list.length, 0);
117
+ const payload = {
118
+ ok: true,
119
+ configPath: config.configPath(),
120
+ total,
121
+ perEvent,
122
+ };
123
+ if (flags.json) {
124
+ ctx.writeOutput(payload, JSON.stringify(payload, null, 2));
125
+ return 0;
126
+ }
127
+ const lines = [];
128
+ lines.push(`pugi hooks (${total} configured)`);
129
+ lines.push(` config: ${config.configPath()}`);
130
+ if (total === 0) {
131
+ lines.push(' no hooks configured — create the file above to add one.');
132
+ }
133
+ else {
134
+ for (const event of ALL_HOOK_EVENTS_V2) {
135
+ const list = perEvent[event];
136
+ if (list.length === 0)
137
+ continue;
138
+ lines.push(` ${event}:`);
139
+ for (const entry of list) {
140
+ const tags = [];
141
+ if (entry.matcher)
142
+ tags.push(`matcher=${entry.matcher}`);
143
+ if (entry.timeoutMs)
144
+ tags.push(`timeoutMs=${entry.timeoutMs}`);
145
+ if (entry.blocking)
146
+ tags.push('blocking');
147
+ const suffix = tags.length ? ` [${tags.join(', ')}]` : '';
148
+ lines.push(` - ${entry.command}${suffix}`);
149
+ }
150
+ }
151
+ }
152
+ const text = lines.join('\n');
153
+ ctx.writeOutput(payload, text);
154
+ return 0;
155
+ }
156
+ function runDoctor(ctx, flags) {
157
+ const path = ctx.configPath ?? defaultHooksMvpPath();
158
+ try {
159
+ const config = loadHooksConfig(ctx.configPath);
160
+ const total = config.flatten().length;
161
+ const payload = {
162
+ ok: true,
163
+ configPath: config.configPath(),
164
+ total,
165
+ issues: [],
166
+ };
167
+ const text = total
168
+ ? `pugi hooks doctor: ${path} OK (${total} hooks).`
169
+ : `pugi hooks doctor: ${path} not present (no hooks configured).`;
170
+ ctx.writeOutput(payload, text);
171
+ return 0;
172
+ }
173
+ catch (error) {
174
+ const msg = error.message;
175
+ const payload = {
176
+ ok: false,
177
+ configPath: path,
178
+ error: msg,
179
+ };
180
+ const text = `pugi hooks doctor: ${msg}`;
181
+ ctx.writeOutput(payload, text);
182
+ return 1;
183
+ }
184
+ // flags.json is consumed by writeOutput in the host shell.
185
+ void flags;
186
+ }
187
+ //# sourceMappingURL=hooks.js.map
@@ -0,0 +1,254 @@
1
+ /**
2
+ * `pugi init` codebase scan → PUGI.md auto-gen (backlog #82).
3
+ *
4
+ * Extends the existing `init()` scaffold with a project-aware PUGI.md
5
+ * generator. Runs AFTER `scaffoldPugiWorkspace` so the `.pugi/` workspace
6
+ * exists for the `init-report.json` write.
7
+ *
8
+ * Flow:
9
+ *
10
+ * 1. Detect existing PUGI.md. If present without `--force`, refuse
11
+ * with `refused_existing`.
12
+ * 2. Run `detectWorkspace` against cwd.
13
+ * 3. Render PUGI.md content for the requested style.
14
+ * 4. Confirm with the operator unless `--yes` / `--json` / `--force`.
15
+ * 5. Write `<cwd>/PUGI.md` atomically (tmp + rename, fallback к
16
+ * direct write on EXDEV — happens когда tmp lives на a different
17
+ * volume).
18
+ * 6. Write `.pugi/init-report.json` (mode 0o600) with the raw
19
+ * detection payload for re-runs / auditing.
20
+ * 7. Emit a JSON envelope describing the outcome.
21
+ */
22
+ import { chmodSync, existsSync, mkdirSync, readFileSync, renameSync, unlinkSync, writeFileSync, } from 'node:fs';
23
+ import { tmpdir } from 'node:os';
24
+ import { dirname, join, resolve } from 'node:path';
25
+ import { createInterface } from 'node:readline';
26
+ import { detectWorkspace } from '../../core/init/detector.js';
27
+ import { renderPugiMd, STYLE_WORD_CAPS, } from '../../core/init/template-renderer.js';
28
+ export const DEFAULT_PUGI_MD_STYLE = 'standard';
29
+ /** Entry point invoked by `cli.ts::init`. */
30
+ export async function runInitScanCommand(args, ctx) {
31
+ const cwd = resolve(ctx.cwd);
32
+ const pugiMdPath = join(cwd, 'PUGI.md');
33
+ const reportPath = join(cwd, '.pugi', 'init-report.json');
34
+ // `args` is currently consumed only via flags. Keep the parameter to
35
+ // mirror every other command surface so we can grow positional args
36
+ // (e.g. an explicit target path) without breaking the call site.
37
+ void args;
38
+ try {
39
+ if (existsSync(pugiMdPath) && !ctx.force) {
40
+ const envelope = {
41
+ command: 'init-scan',
42
+ status: 'refused_existing',
43
+ pugiMdPath,
44
+ reportPath,
45
+ style: ctx.style,
46
+ message: 'PUGI.md already exists. Re-run with --force to overwrite, or edit by hand.',
47
+ };
48
+ ctx.writeOutput(envelope, formatHuman(envelope));
49
+ return envelope;
50
+ }
51
+ const detection = detectWorkspace(cwd);
52
+ const rendered = renderPugiMd(detection, { style: ctx.style, now: ctx.now });
53
+ const shouldConfirm = !ctx.yes && !ctx.json && !ctx.force;
54
+ if (shouldConfirm) {
55
+ const confirm = ctx.confirm ?? defaultConfirm;
56
+ const ok = await confirm(`Write PUGI.md at ${pugiMdPath} (${rendered.wordCount} words, style: ${ctx.style})? [y/N] `);
57
+ if (!ok) {
58
+ const envelope = {
59
+ command: 'init-scan',
60
+ status: 'aborted',
61
+ pugiMdPath,
62
+ reportPath,
63
+ style: ctx.style,
64
+ message: 'Aborted by operator before write.',
65
+ };
66
+ ctx.writeOutput(envelope, formatHuman(envelope));
67
+ return envelope;
68
+ }
69
+ }
70
+ atomicWrite(pugiMdPath, rendered.content);
71
+ writeReport(reportPath, detection, rendered.wordCount, ctx.style);
72
+ const envelope = {
73
+ command: 'init-scan',
74
+ status: 'written',
75
+ pugiMdPath,
76
+ reportPath,
77
+ detection,
78
+ wordCount: rendered.wordCount,
79
+ style: ctx.style,
80
+ };
81
+ ctx.writeOutput(envelope, formatHuman(envelope));
82
+ return envelope;
83
+ }
84
+ catch (error) {
85
+ const message = error instanceof Error ? error.message : String(error);
86
+ const envelope = {
87
+ command: 'init-scan',
88
+ status: 'error',
89
+ pugiMdPath,
90
+ reportPath,
91
+ style: ctx.style,
92
+ message,
93
+ };
94
+ ctx.writeOutput(envelope, formatHuman(envelope));
95
+ return envelope;
96
+ }
97
+ }
98
+ /* ------------------------------------------------------------------ */
99
+ /* Atomic write */
100
+ /* ------------------------------------------------------------------ */
101
+ function atomicWrite(targetPath, content) {
102
+ const dir = dirname(targetPath);
103
+ mkdirSync(dir, { recursive: true });
104
+ const tmpName = `.pugi-md-${process.pid}-${Date.now()}.tmp`;
105
+ const tmpPath = join(tmpdir(), tmpName);
106
+ try {
107
+ writeFileSync(tmpPath, content, { encoding: 'utf8', mode: 0o644 });
108
+ renameSync(tmpPath, targetPath);
109
+ return;
110
+ }
111
+ catch (error) {
112
+ // EXDEV — cross-device link: tmpdir lives on a different volume
113
+ // than the workspace. Retry with the tmpfile inside the workspace.
114
+ cleanupTmp(tmpPath);
115
+ const code = error.code;
116
+ if (code !== 'EXDEV') {
117
+ // Fall through to the in-dir fallback for any rename failure so
118
+ // we keep the contract: PUGI.md is either fully written or not
119
+ // written. The direct writeFileSync at the end is the final
120
+ // safety net.
121
+ }
122
+ const localTmpPath = join(dir, `.pugi-md-${process.pid}-${Date.now()}.tmp`);
123
+ try {
124
+ writeFileSync(localTmpPath, content, { encoding: 'utf8', mode: 0o644 });
125
+ renameSync(localTmpPath, targetPath);
126
+ return;
127
+ }
128
+ catch {
129
+ cleanupTmp(localTmpPath);
130
+ }
131
+ // Final fallback — direct write. We sacrifice the atomicity guarantee
132
+ // here but keep behavioural correctness when both the tmp + in-dir
133
+ // strategies fail (rare; usually FS quirks).
134
+ writeFileSync(targetPath, content, { encoding: 'utf8', mode: 0o644 });
135
+ }
136
+ }
137
+ function cleanupTmp(path) {
138
+ try {
139
+ unlinkSync(path);
140
+ }
141
+ catch {
142
+ // best effort
143
+ }
144
+ }
145
+ function writeReport(reportPath, detection, wordCount, style) {
146
+ mkdirSync(dirname(reportPath), { recursive: true });
147
+ const payload = {
148
+ generatedAt: new Date().toISOString(),
149
+ style,
150
+ wordCap: STYLE_WORD_CAPS[style],
151
+ wordCount,
152
+ detection,
153
+ };
154
+ writeFileSync(reportPath, JSON.stringify(payload, null, 2), {
155
+ encoding: 'utf8',
156
+ mode: 0o600,
157
+ });
158
+ try {
159
+ chmodSync(reportPath, 0o600);
160
+ }
161
+ catch {
162
+ // Some filesystems (FAT, NTFS over CIFS) reject chmod; the mode
163
+ // option on writeFileSync is the primary guard. Best-effort here.
164
+ }
165
+ }
166
+ /* ------------------------------------------------------------------ */
167
+ /* Confirm prompt */
168
+ /* ------------------------------------------------------------------ */
169
+ async function defaultConfirm(question) {
170
+ if (process.stdin.isTTY !== true) {
171
+ return true;
172
+ }
173
+ const rl = createInterface({
174
+ input: process.stdin,
175
+ output: process.stderr,
176
+ });
177
+ return new Promise((resolveAnswer) => {
178
+ let settled = false;
179
+ const settle = (value) => {
180
+ if (settled)
181
+ return;
182
+ settled = true;
183
+ rl.close();
184
+ resolveAnswer(value);
185
+ };
186
+ rl.once('close', () => settle(false));
187
+ rl.question(question, (answer) => {
188
+ settle(/^y(es)?$/i.test(answer.trim()));
189
+ });
190
+ });
191
+ }
192
+ /* ------------------------------------------------------------------ */
193
+ /* Human-readable formatting */
194
+ /* ------------------------------------------------------------------ */
195
+ function formatHuman(env) {
196
+ switch (env.status) {
197
+ case 'written':
198
+ return [
199
+ `PUGI.md written to ${env.pugiMdPath}`,
200
+ ` Style: ${env.style}`,
201
+ ` Word count: ${env.wordCount ?? 0}`,
202
+ ` Report: ${env.reportPath}`,
203
+ ].join('\n');
204
+ case 'refused_existing':
205
+ return env.message ?? 'PUGI.md already exists. Use --force to overwrite.';
206
+ case 'aborted':
207
+ return env.message ?? 'Aborted.';
208
+ case 'error':
209
+ return `pugi init scan failed: ${env.message ?? 'unknown error'}`;
210
+ default:
211
+ return 'pugi init scan complete.';
212
+ }
213
+ }
214
+ /* ------------------------------------------------------------------ */
215
+ /* Helpers consumed by cli.ts */
216
+ /* ------------------------------------------------------------------ */
217
+ /** Parse `--style=<minimal|standard|detailed>` (or `--style minimal`). */
218
+ export function parseInitStyle(args) {
219
+ for (let i = 0; i < args.length; i += 1) {
220
+ const arg = args[i];
221
+ if (typeof arg !== 'string')
222
+ continue;
223
+ if (arg.startsWith('--style=')) {
224
+ const value = arg.slice('--style='.length);
225
+ const parsed = coerceStyle(value);
226
+ if (parsed)
227
+ return parsed;
228
+ }
229
+ else if (arg === '--style') {
230
+ const next = args[i + 1];
231
+ if (typeof next === 'string') {
232
+ const parsed = coerceStyle(next);
233
+ if (parsed)
234
+ return parsed;
235
+ }
236
+ }
237
+ }
238
+ return DEFAULT_PUGI_MD_STYLE;
239
+ }
240
+ function coerceStyle(value) {
241
+ if (value === 'minimal' || value === 'standard' || value === 'detailed')
242
+ return value;
243
+ return null;
244
+ }
245
+ /** Test surface: read back the on-disk report. */
246
+ export function readInitReport(reportPath) {
247
+ try {
248
+ return JSON.parse(readFileSync(reportPath, 'utf8'));
249
+ }
250
+ catch {
251
+ return null;
252
+ }
253
+ }
254
+ //# sourceMappingURL=init.js.map