@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,337 @@
1
+ /**
2
+ * AskUserQuestion structured tool — .
3
+ *
4
+ * Mirrors the standard structured-question convention
5
+ * pattern: clarifying questions go through a structured multi-choice
6
+ * tool, NOT free-text prose. The model dispatches the tool with a
7
+ * `question` + a `header` chip + 2-4 `options` (each with `label` +
8
+ * `description`). The UI renders the modal, auto-appends an "Other"
9
+ * fallback for custom text, and surfaces the operator's pick back to
10
+ * the model as a tool_result frame.
11
+ *
12
+ * Why P0 leverage: the structured form forecloses Pugi's recurring
13
+ * "agent rambles instead of dispatching" failure mode at the schema
14
+ * level. When the model is uncertain, the cheapest legal output is
15
+ * `ask_user_question` — not a prose menu, not a fake "Шипану через
16
+ * 8 минут" dispatch promise.
17
+ *
18
+ * Relationship to ask-user.ts (β1 T2):
19
+ * - ask-user.ts is the LEGACY string-array form (`options: string[]`).
20
+ * Kept for back-compat; the existing prompt envelope `<pugi-ask>`
21
+ * and the persona prompts still emit that grammar.
22
+ * - ask-user-question.ts is the STRUCTURED form layered on top. It
23
+ * normalises a {label, description} option into the legacy string
24
+ * before delegating to `askUser`, so the Ink modal does not need
25
+ * two render paths and the abort/timeout race logic is shared.
26
+ *
27
+ * Hard rules (enforced by Zod):
28
+ * - question: 5-500 chars, must end with "?". Plain English.
29
+ * - header: 2-12 chars (short chip label, e.g. "Auth method").
30
+ * - options: 2-4 strict (no more, no less). Mutually exclusive.
31
+ * UI auto-adds "Other" — the model NEVER emits it.
32
+ * - multiSelect: default false.
33
+ */
34
+ import { z } from 'zod';
35
+ import { askUser } from './ask-user.js';
36
+ /** Cap matches the Ink modal layout: 12 chars fits the header chip. */
37
+ export const ASK_USER_QUESTION_HEADER_MIN = 2;
38
+ export const ASK_USER_QUESTION_HEADER_MAX = 12;
39
+ /** Question must be a real question (ends with ?). 5-500 chars. */
40
+ export const ASK_USER_QUESTION_MIN = 5;
41
+ export const ASK_USER_QUESTION_MAX = 500;
42
+ /** Each option label: 2-40 chars (1-5 words). */
43
+ export const ASK_USER_QUESTION_OPTION_LABEL_MIN = 2;
44
+ export const ASK_USER_QUESTION_OPTION_LABEL_MAX = 40;
45
+ /** Each option description: 10-200 chars (one short sentence). */
46
+ export const ASK_USER_QUESTION_OPTION_DESC_MIN = 10;
47
+ export const ASK_USER_QUESTION_OPTION_DESC_MAX = 200;
48
+ /** Option count: 2-4 strict. UI adds "Other" automatically. */
49
+ export const ASK_USER_QUESTION_OPTIONS_MIN = 2;
50
+ export const ASK_USER_QUESTION_OPTIONS_MAX = 4;
51
+ /**
52
+ * Preview pane cap (PUGI-130). Lets the model attach a multi-line
53
+ * ASCII / code / diagram preview to an option. When ANY option в a
54
+ * single-select question carries `preview`, the UI switches к
55
+ * side-by-side layout (options column left, preview pane right).
56
+ *
57
+ * Capped at 5000 chars так что a single payload can stash a full
58
+ * component sketch / SQL migration / diagram, but не enough rope для
59
+ * a model to dump the entire codebase в a preview field.
60
+ *
61
+ * **multiSelect rule:** previews are REJECTED on multi-select questions.
62
+ * Rendering N preview panes simultaneously breaks the layout invariant
63
+ * (single 80-col terminal can host ONE side panel, not N). The schema
64
+ * validator enforces this at the tool boundary с a clear error message.
65
+ */
66
+ export const ASK_USER_QUESTION_OPTION_PREVIEW_MIN = 1;
67
+ export const ASK_USER_QUESTION_OPTION_PREVIEW_MAX = 5000;
68
+ /** PUGI-480 short-format chip rules: ≤ 5 words / option label, ≤ 3 questions / call. */
69
+ export const ASK_USER_QUESTION_CHIP_LABEL_WORD_MAX = 5;
70
+ export const ASK_USER_QUESTION_CHIPS_MAX = 3;
71
+ /**
72
+ * Reusable validator for option labels rendered в chip mode.
73
+ * Counts whitespace-delimited words. Throws с a clear message that
74
+ * names the offending question + option so the model can self-correct.
75
+ */
76
+ export function assertChipLabelWordCap(questionHeader, optionLabel) {
77
+ const words = optionLabel.trim().split(/\s+/u).filter((w) => w.length > 0);
78
+ if (words.length > ASK_USER_QUESTION_CHIP_LABEL_WORD_MAX) {
79
+ throw new Error(`ask_user_question chip "${questionHeader}" option "${optionLabel}" ` +
80
+ `exceeds the ${ASK_USER_QUESTION_CHIP_LABEL_WORD_MAX}-word label cap ` +
81
+ `(saw ${words.length} words). Shorten the label before dispatch.`);
82
+ }
83
+ }
84
+ /**
85
+ * Structured option. `label` is the display text; `description` is the
86
+ * implication line shown dim below it. Both are required — the model
87
+ * cannot ship a label-only option (forces it to think about why each
88
+ * choice exists).
89
+ */
90
+ export const askUserQuestionOptionSchema = z.strictObject({
91
+ label: z
92
+ .string()
93
+ .min(ASK_USER_QUESTION_OPTION_LABEL_MIN)
94
+ .max(ASK_USER_QUESTION_OPTION_LABEL_MAX)
95
+ .describe('Display text. Concise (1-5 words).'),
96
+ description: z
97
+ .string()
98
+ .min(ASK_USER_QUESTION_OPTION_DESC_MIN)
99
+ .max(ASK_USER_QUESTION_OPTION_DESC_MAX)
100
+ .describe('What this option means / implications.'),
101
+ preview: z
102
+ .string()
103
+ .min(ASK_USER_QUESTION_OPTION_PREVIEW_MIN)
104
+ .max(ASK_USER_QUESTION_OPTION_PREVIEW_MAX)
105
+ .optional()
106
+ .describe('Optional ASCII / code preview (≤ 5000 chars, multi-line OK). When ' +
107
+ 'ANY option in the set carries this field, the UI switches к ' +
108
+ 'side-by-side layout (options column left, preview pane right). ' +
109
+ 'Use для visual comparison of mockups, config snippets, diagram ' +
110
+ 'variations. REJECTED on multiSelect questions (one pane fits one ' +
111
+ 'preview).'),
112
+ });
113
+ export const askUserQuestionSchema = z
114
+ .strictObject({
115
+ question: z
116
+ .string()
117
+ .min(ASK_USER_QUESTION_MIN)
118
+ .max(ASK_USER_QUESTION_MAX)
119
+ .refine((q) => q.trim().endsWith('?'), {
120
+ message: 'question must end with "?"',
121
+ })
122
+ .describe('The complete question. Must end with "?". Plain English, no jargon.'),
123
+ header: z
124
+ .string()
125
+ .min(ASK_USER_QUESTION_HEADER_MIN)
126
+ .max(ASK_USER_QUESTION_HEADER_MAX)
127
+ .describe('Short chip label (max 12 chars). E.g. "Auth method".'),
128
+ options: z
129
+ .array(askUserQuestionOptionSchema)
130
+ .min(ASK_USER_QUESTION_OPTIONS_MIN)
131
+ .max(ASK_USER_QUESTION_OPTIONS_MAX)
132
+ .describe('2-4 mutually-exclusive options. NEVER add "Other" — UI auto-adds.'),
133
+ multiSelect: z
134
+ .boolean()
135
+ .optional()
136
+ .default(false)
137
+ .describe('Allow multiple selections. Default false.'),
138
+ })
139
+ .superRefine((payload, ctx) => {
140
+ // PUGI-130 invariant: previews are single-pane only. Side-by-side
141
+ // layout can host ONE preview at a time; multiSelect implies the
142
+ // operator may toggle several options and each would want its own
143
+ // pane. Reject at the schema boundary so the model gets immediate
144
+ // feedback instead of a silently dropped preview field at render.
145
+ if (payload.multiSelect === true) {
146
+ payload.options.forEach((opt, idx) => {
147
+ if (typeof opt.preview === 'string' && opt.preview.length > 0) {
148
+ ctx.addIssue({
149
+ code: z.ZodIssueCode.custom,
150
+ path: ['options', idx, 'preview'],
151
+ message: 'preview is not allowed on multiSelect questions — ' +
152
+ 'side-by-side layout supports ONE preview pane at a time. ' +
153
+ 'Drop the preview field OR switch к single-select.',
154
+ });
155
+ }
156
+ });
157
+ }
158
+ });
159
+ /**
160
+ * PUGI-480 multi-question chip payload — a bundle of up to 3 short-format
161
+ * chip questions rendered side-by-side. Schema is intentionally narrow:
162
+ * the chip renderer relies on the ≤ 5-word label invariant и will
163
+ * truncate с "…" if the model ever bypasses Zod. The 3-question cap
164
+ * forecloses paragraph-wall prompts at the schema level — the model
165
+ * cannot legally ask "10 quick questions" in one shot.
166
+ */
167
+ export const askUserQuestionChipsQuestionSchema = z.strictObject({
168
+ header: z
169
+ .string()
170
+ .min(ASK_USER_QUESTION_HEADER_MIN)
171
+ .max(ASK_USER_QUESTION_HEADER_MAX)
172
+ .describe('Short chip label (max 12 chars). E.g. "Stack".'),
173
+ question: z
174
+ .string()
175
+ .min(ASK_USER_QUESTION_MIN)
176
+ .max(ASK_USER_QUESTION_MAX)
177
+ .optional()
178
+ .describe('Optional full-prose question (shown in non-TTY fallback only).'),
179
+ options: z
180
+ .array(askUserQuestionOptionSchema)
181
+ .min(ASK_USER_QUESTION_OPTIONS_MIN)
182
+ .max(ASK_USER_QUESTION_OPTIONS_MAX + 1)
183
+ .superRefine((opts, ctx) => {
184
+ // Enforce ≤ 5-word label cap on every option. Schema-level so
185
+ // the model gets immediate feedback on overflow rather than a
186
+ // silent truncation at render time.
187
+ for (const opt of opts) {
188
+ const words = opt.label
189
+ .trim()
190
+ .split(/\s+/u)
191
+ .filter((w) => w.length > 0);
192
+ if (words.length > ASK_USER_QUESTION_CHIP_LABEL_WORD_MAX) {
193
+ ctx.addIssue({
194
+ code: z.ZodIssueCode.custom,
195
+ message: `option "${opt.label}" has ${words.length} words; chip labels must be ≤ ${ASK_USER_QUESTION_CHIP_LABEL_WORD_MAX} words`,
196
+ });
197
+ }
198
+ }
199
+ })
200
+ .describe('2-5 mutually-exclusive options. Each label ≤ 5 words. UI inserts "Skip — use defaults" as a final option when defaults are present.'),
201
+ });
202
+ export const askUserQuestionChipsSchema = z.strictObject({
203
+ questions: z
204
+ .array(askUserQuestionChipsQuestionSchema)
205
+ .min(1)
206
+ .max(ASK_USER_QUESTION_CHIPS_MAX)
207
+ .describe('Bundle of 1-3 short clarifier questions rendered side-by-side as chips.'),
208
+ });
209
+ /**
210
+ * Dispatch the structured tool: validate args via Zod, then route
211
+ * through the shared `askUser` primitive so abort/timeout/non-TTY
212
+ * envelope behaviour is identical to the legacy form.
213
+ *
214
+ * The bridge surface is the same `AskUserBridge` signature — the
215
+ * structured form just gives the Ink modal richer metadata to render
216
+ * (header chip + per-option description). The bridge sees the legacy
217
+ * `{question, options: string[]}` shape because all production bridges
218
+ * (Ink modal + non-TTY envelope emitter) already consume that shape.
219
+ * Per-option descriptions and the header chip are surfaced separately
220
+ * via `enrich` — the modal layer reads them off the dispatched payload
221
+ * stash, NOT off the bridge input, so structured callers can layer on
222
+ * top of the legacy interface without touching the modal contract.
223
+ *
224
+ * Return contract:
225
+ * - Interactive + bridge present + operator picks N options →
226
+ * `[ask_user_question:answered] <labels joined by ", ">`.
227
+ * - Interactive + bridge present + operator cancels →
228
+ * `[ask_user_question:cancelled]`.
229
+ * - Interactive + bridge present + timeout →
230
+ * `[ask_user_question:timeout]`.
231
+ * - Non-TTY or no bridge → `[user_input_required]<json>[/...]`
232
+ * envelope identical to the legacy form. Includes `header` +
233
+ * structured options so a scripted caller can parse the full shape.
234
+ */
235
+ export async function dispatchAskUserQuestion(ctx, rawArgs) {
236
+ const parsed = askUserQuestionSchema.parse(rawArgs);
237
+ // Schema-level guard against the "Other" leak: the prompt rules tell
238
+ // the model NEVER to include "Other" in `options`, but we still reject
239
+ // it defensively in case a future model misreads the spec. The Ink
240
+ // modal auto-appends "Other" itself; a model-supplied duplicate would
241
+ // render two "Other" rows.
242
+ for (const opt of parsed.options) {
243
+ const trimmed = opt.label.trim().toLowerCase();
244
+ if (trimmed === 'other' || trimmed === 'другое') {
245
+ throw new Error('ask_user_question: do NOT include "Other" in options — UI auto-adds it');
246
+ }
247
+ }
248
+ const legacyOptions = parsed.options.map((opt) => opt.label);
249
+ const result = await askUser(ctx, {
250
+ question: parsed.question,
251
+ options: legacyOptions,
252
+ multiSelect: parsed.multiSelect ?? false,
253
+ });
254
+ if (result.answers && result.answers.length > 0) {
255
+ return {
256
+ answers: result.answers,
257
+ envelope: `[ask_user_question:answered] ${result.answers.join(', ')}`,
258
+ };
259
+ }
260
+ // Non-TTY / cancelled / timeout. Re-wrap the envelope so callers can
261
+ // grep for the structured tool name even when the underlying primitive
262
+ // surfaced its legacy `[user_input_required]` envelope.
263
+ if (result.envelope.includes('"reason":"timeout"')) {
264
+ return { envelope: '[ask_user_question:timeout]' };
265
+ }
266
+ if (result.envelope.includes('"reason":"cancelled"')) {
267
+ return { envelope: '[ask_user_question:cancelled]' };
268
+ }
269
+ // Default to the legacy envelope verbatim — it is still
270
+ // grep-friendly and includes the structured payload above.
271
+ return { envelope: result.envelope };
272
+ }
273
+ /**
274
+ * JSON-Schema fragment surfaced to the model via the tool-bridge
275
+ * `parameters` field. Mirrors the Zod schema 1:1 — kept hand-written
276
+ * because the runtime engine wires OpenAI-compatible JSON Schema and
277
+ * the Zod-to-JSON-Schema converter pulls in a transitive dep we have
278
+ * not greenlit. If the Zod schema above changes, mirror the change here.
279
+ */
280
+ export const askUserQuestionJsonSchema = {
281
+ type: 'object',
282
+ additionalProperties: false,
283
+ required: ['question', 'header', 'options'],
284
+ properties: {
285
+ question: {
286
+ type: 'string',
287
+ minLength: ASK_USER_QUESTION_MIN,
288
+ maxLength: ASK_USER_QUESTION_MAX,
289
+ description: 'The complete question. Must end with "?". Plain English, no jargon.',
290
+ },
291
+ header: {
292
+ type: 'string',
293
+ minLength: ASK_USER_QUESTION_HEADER_MIN,
294
+ maxLength: ASK_USER_QUESTION_HEADER_MAX,
295
+ description: 'Short chip label (max 12 chars). E.g. "Auth method".',
296
+ },
297
+ options: {
298
+ type: 'array',
299
+ minItems: ASK_USER_QUESTION_OPTIONS_MIN,
300
+ maxItems: ASK_USER_QUESTION_OPTIONS_MAX,
301
+ items: {
302
+ type: 'object',
303
+ additionalProperties: false,
304
+ required: ['label', 'description'],
305
+ properties: {
306
+ label: {
307
+ type: 'string',
308
+ minLength: ASK_USER_QUESTION_OPTION_LABEL_MIN,
309
+ maxLength: ASK_USER_QUESTION_OPTION_LABEL_MAX,
310
+ description: 'Display text. Concise (1-5 words).',
311
+ },
312
+ description: {
313
+ type: 'string',
314
+ minLength: ASK_USER_QUESTION_OPTION_DESC_MIN,
315
+ maxLength: ASK_USER_QUESTION_OPTION_DESC_MAX,
316
+ description: 'What this option means / implications.',
317
+ },
318
+ preview: {
319
+ type: 'string',
320
+ minLength: ASK_USER_QUESTION_OPTION_PREVIEW_MIN,
321
+ maxLength: ASK_USER_QUESTION_OPTION_PREVIEW_MAX,
322
+ description: 'Optional multi-line ASCII / code / diagram preview (≤ 5000 chars). ' +
323
+ 'When ANY option carries this, UI switches к side-by-side layout. ' +
324
+ 'Single-select only — REJECTED if multiSelect is true.',
325
+ },
326
+ },
327
+ },
328
+ description: '2-4 mutually-exclusive options. NEVER add "Other" — UI auto-adds.',
329
+ },
330
+ multiSelect: {
331
+ type: 'boolean',
332
+ description: 'Allow multiple selections. Default false. When true, option.preview ' +
333
+ 'is forbidden (one terminal pane fits one preview at a time).',
334
+ },
335
+ },
336
+ };
337
+ //# sourceMappingURL=ask-user-question.js.map
@@ -0,0 +1,115 @@
1
+ export const ASK_USER_DEFAULT_TIMEOUT_MS = 5 * 60 * 1_000;
2
+ /**
3
+ * Schema cap: keep the option count tight so the modal stays scannable.
4
+ * Mirrors `ASK_MAX_OPTIONS` in `core/repl/ask.ts` (4).
5
+ */
6
+ export const ASK_USER_MAX_OPTIONS = 4;
7
+ export const ASK_USER_MAX_QUESTION_LEN = 1_000;
8
+ export const ASK_USER_MAX_OPTION_LEN = 200;
9
+ export async function askUser(ctx, input) {
10
+ validate(input);
11
+ if (ctx.interactive && ctx.bridge) {
12
+ // β1a r1 : wrap the bridge in an abort-aware race so a
13
+ // pending modal cannot block the engine loop forever. Two signals
14
+ // can interrupt:
15
+ // 1. `ctx.signal` — the operator cancelled the parent task via
16
+ // Ctrl-C; the engine forwards the loop's AbortSignal here.
17
+ // 2. `ctx.timeoutMs` (default 5 minutes) — operator walked away;
18
+ // the modal stays renderable but the tool surface returns the
19
+ // `cancelled` envelope so the model can make progress.
20
+ // The bridge receives the same `signal` so an Ink-based modal can
21
+ // tear down its render loop and free its keyboard handlers on
22
+ // abort. Bridges that ignore the signal still get pre-empted by
23
+ // the race — they just leak a render until the next operator
24
+ // keystroke.
25
+ const timeoutMs = ctx.timeoutMs ?? ASK_USER_DEFAULT_TIMEOUT_MS;
26
+ // Pre-flight: short-circuit when the caller's signal is already
27
+ // aborted. Avoids constructing a bridge promise that races against
28
+ // an already-resolved abort sentinel — the race ordering is
29
+ // unspecified for promises that have all settled by the time
30
+ // Promise.race is called, which would non-deterministically let
31
+ // the bridge's answer leak through after an explicit cancel.
32
+ if (ctx.signal?.aborted) {
33
+ return { envelope: formatEnvelope(input, 'cancelled') };
34
+ }
35
+ const controller = new AbortController();
36
+ if (ctx.signal) {
37
+ ctx.signal.addEventListener('abort', () => controller.abort(), { once: true });
38
+ }
39
+ let timeoutHandle;
40
+ const timeoutPromise = new Promise((resolve) => {
41
+ timeoutHandle = setTimeout(() => resolve('timeout'), timeoutMs);
42
+ });
43
+ const abortPromise = new Promise((resolve) => {
44
+ controller.signal.addEventListener('abort', () => resolve('aborted'), { once: true });
45
+ });
46
+ let picked;
47
+ try {
48
+ picked = await Promise.race([
49
+ ctx.bridge(input, { signal: controller.signal }),
50
+ timeoutPromise,
51
+ abortPromise,
52
+ ]);
53
+ }
54
+ finally {
55
+ if (timeoutHandle)
56
+ clearTimeout(timeoutHandle);
57
+ }
58
+ if (picked === 'timeout') {
59
+ controller.abort();
60
+ return { envelope: formatEnvelope(input, 'timeout') };
61
+ }
62
+ if (picked === 'aborted') {
63
+ return { envelope: formatEnvelope(input, 'cancelled') };
64
+ }
65
+ if (!Array.isArray(picked) || picked.length === 0) {
66
+ // Operator declined / closed the modal — surface a structured
67
+ // "no answer" envelope so the model can decide whether to retry.
68
+ const envelope = formatEnvelope(input, 'cancelled');
69
+ return { envelope };
70
+ }
71
+ return {
72
+ answers: picked,
73
+ envelope: formatAnswer(picked),
74
+ };
75
+ }
76
+ // Non-TTY or no bridge — surface the envelope. Caller parses it and
77
+ // either pipes an answer back on a follow-up turn or aborts.
78
+ const envelope = formatEnvelope(input, 'no_tty');
79
+ return { envelope };
80
+ }
81
+ function validate(input) {
82
+ const question = input.question?.trim();
83
+ if (!question)
84
+ throw new Error('ask_user: question is required');
85
+ if (question.length > ASK_USER_MAX_QUESTION_LEN) {
86
+ throw new Error(`ask_user: question exceeds ${ASK_USER_MAX_QUESTION_LEN} char cap`);
87
+ }
88
+ if (!Array.isArray(input.options) || input.options.length < 2) {
89
+ throw new Error('ask_user: at least 2 options required');
90
+ }
91
+ if (input.options.length > ASK_USER_MAX_OPTIONS) {
92
+ throw new Error(`ask_user: at most ${ASK_USER_MAX_OPTIONS} options allowed`);
93
+ }
94
+ for (const opt of input.options) {
95
+ if (typeof opt !== 'string' || !opt.trim()) {
96
+ throw new Error('ask_user: every option must be a non-empty string');
97
+ }
98
+ if (opt.length > ASK_USER_MAX_OPTION_LEN) {
99
+ throw new Error(`ask_user: option exceeds ${ASK_USER_MAX_OPTION_LEN} char cap`);
100
+ }
101
+ }
102
+ }
103
+ function formatEnvelope(input, reason) {
104
+ const payload = {
105
+ question: input.question,
106
+ options: input.options,
107
+ multiSelect: input.multiSelect === true,
108
+ reason,
109
+ };
110
+ return `[user_input_required]${JSON.stringify(payload)}[/user_input_required]`;
111
+ }
112
+ function formatAnswer(answers) {
113
+ return answers.join(', ');
114
+ }
115
+ //# sourceMappingURL=ask-user.js.map