@pugi/cli 0.1.0-beta.8 → 0.1.0-beta.88

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 (405) hide show
  1. package/CHANGELOG.md +132 -0
  2. package/LICENSE +1 -1
  3. package/THIRD_PARTY_NOTICES.md +40 -0
  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/smoke.js +133 -0
  11. package/dist/core/agent-progress/cleanup.js +134 -0
  12. package/dist/core/agent-progress/schema.js +144 -0
  13. package/dist/core/agent-progress/writer.js +101 -0
  14. package/dist/core/agents/adaptive-router.js +330 -0
  15. package/dist/core/agents/query-decomposer.js +297 -0
  16. package/dist/core/agents/registry.js +3 -3
  17. package/dist/core/approvals/shortcut-resolver.js +98 -0
  18. package/dist/core/artifact-chain/dispatcher.js +148 -0
  19. package/dist/core/artifact-chain/exporter.js +164 -0
  20. package/dist/core/artifact-chain/state.js +243 -0
  21. package/dist/core/artifact-chain/steps.js +169 -0
  22. package/dist/core/ask-user/question.js +92 -0
  23. package/dist/core/audit/audit-trail.js +275 -0
  24. package/dist/core/auth/ensure-authenticated.js +129 -0
  25. package/dist/core/auth/env-provider.js +238 -0
  26. package/dist/core/auto-open-browser.js +4 -4
  27. package/dist/core/auto-update/channels.js +122 -0
  28. package/dist/core/auto-update/checker.js +241 -0
  29. package/dist/core/auto-update/state.js +235 -0
  30. package/dist/core/bare-mode/index.js +107 -0
  31. package/dist/core/bash/redirect.js +281 -0
  32. package/dist/core/bash-classifier.js +436 -40
  33. package/dist/core/checkpoint/resumer.js +149 -0
  34. package/dist/core/checkpoint/rewinder.js +291 -0
  35. package/dist/core/checkpoints/shadow-git.js +670 -0
  36. package/dist/core/citations/parser.js +109 -0
  37. package/dist/core/classifier/yolo-classifier.js +88 -0
  38. package/dist/core/codegraph/decision-store.js +248 -0
  39. package/dist/core/codegraph/detect-repo.js +459 -0
  40. package/dist/core/codegraph/install.js +134 -0
  41. package/dist/core/codegraph/offer-hook.js +220 -0
  42. package/dist/core/compact/auto-trigger.js +96 -0
  43. package/dist/core/compact/buffer-rewriter.js +115 -0
  44. package/dist/core/compact/summarizer.js +208 -0
  45. package/dist/core/compact/token-counter.js +108 -0
  46. package/dist/core/consensus/anvil-fanout.js +25 -25
  47. package/dist/core/consensus/diff-capture.js +121 -12
  48. package/dist/core/consensus/rubric.js +21 -21
  49. package/dist/core/context/builder.js +6 -6
  50. package/dist/core/context/compaction-events.js +8 -8
  51. package/dist/core/context/compaction.js +31 -31
  52. package/dist/core/context/index.js +15 -8
  53. package/dist/core/context/invariants.js +51 -51
  54. package/dist/core/context/markdown-loader.js +28 -10
  55. package/dist/core/context/markdown-traverse.js +255 -0
  56. package/dist/core/context/pugiignore.js +41 -41
  57. package/dist/core/context/repo-skeleton.js +37 -37
  58. package/dist/core/context/tool-eviction.js +55 -0
  59. package/dist/core/context/watcher.js +32 -32
  60. package/dist/core/context/working-set.js +23 -23
  61. package/dist/core/coordinator/agent-tools.js +77 -0
  62. package/dist/core/coordinator/agent-toolset.js +65 -0
  63. package/dist/core/coordinator/fsm.js +73 -0
  64. package/dist/core/coordinator/mode-fsm.js +70 -0
  65. package/dist/core/cost/rate-card.js +129 -0
  66. package/dist/core/cost/tracker.js +221 -0
  67. package/dist/core/credentials.js +12 -12
  68. package/dist/core/cron/scheduler.js +138 -0
  69. package/dist/core/denial-tracking/index.js +8 -0
  70. package/dist/core/denial-tracking/state.js +264 -0
  71. package/dist/core/diagnostics/probe-runner.js +93 -0
  72. package/dist/core/diagnostics/probes/api.js +46 -0
  73. package/dist/core/diagnostics/probes/auth.js +93 -0
  74. package/dist/core/diagnostics/probes/bare-mode.js +42 -0
  75. package/dist/core/diagnostics/probes/cli-version.js +127 -0
  76. package/dist/core/diagnostics/probes/config.js +72 -0
  77. package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
  78. package/dist/core/diagnostics/probes/disk.js +81 -0
  79. package/dist/core/diagnostics/probes/engine-live.js +46 -0
  80. package/dist/core/diagnostics/probes/git.js +65 -0
  81. package/dist/core/diagnostics/probes/hooks.js +118 -0
  82. package/dist/core/diagnostics/probes/mcp.js +75 -0
  83. package/dist/core/diagnostics/probes/node.js +59 -0
  84. package/dist/core/diagnostics/probes/pnpm.js +36 -0
  85. package/dist/core/diagnostics/probes/pugi-md.js +89 -0
  86. package/dist/core/diagnostics/probes/sandbox.js +40 -0
  87. package/dist/core/diagnostics/probes/session.js +74 -0
  88. package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
  89. package/dist/core/diagnostics/probes/workspace.js +63 -0
  90. package/dist/core/diagnostics/types.js +70 -0
  91. package/dist/core/dispatch/cache-cleanup.js +197 -0
  92. package/dist/core/dispatch/cache-handoff.js +295 -0
  93. package/dist/core/edits/apply-patch-layer-e.js +189 -0
  94. package/dist/core/edits/dispatch.js +293 -7
  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 +3 -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 +322 -0
  108. package/dist/core/engine/anvil-client.js +151 -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 +134 -16
  117. package/dist/core/engine/strip-internal-fields.js +124 -0
  118. package/dist/core/engine/tool-bridge.js +1295 -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 +44 -0
  129. package/dist/core/hooks/index.js +15 -0
  130. package/dist/core/hooks/registry.js +213 -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/image/renderer.js +71 -0
  141. package/dist/core/init/detector.js +582 -0
  142. package/dist/core/init/template-renderer.js +242 -0
  143. package/dist/core/jobs/registry.js +18 -18
  144. package/dist/core/ledger/results-tsv.js +142 -0
  145. package/dist/core/log-discipline/stdout-redirect.js +51 -0
  146. package/dist/core/lsp/cache.js +105 -0
  147. package/dist/core/lsp/client.js +776 -0
  148. package/dist/core/lsp/language-detect.js +66 -0
  149. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  150. package/dist/core/lsp/symbol-tools.js +372 -0
  151. package/dist/core/mcp/client.js +97 -28
  152. package/dist/core/mcp/http-server.js +553 -0
  153. package/dist/core/mcp/orchestrator-tools.js +662 -0
  154. package/dist/core/mcp/permission.js +190 -0
  155. package/dist/core/mcp/registry.js +39 -17
  156. package/dist/core/mcp/server-tools.js +219 -0
  157. package/dist/core/mcp/server.js +397 -0
  158. package/dist/core/mcp/trust.js +10 -10
  159. package/dist/core/memory/dual-write.js +416 -0
  160. package/dist/core/memory/passive-extract.js +130 -0
  161. package/dist/core/memory/phase1-kinds.js +20 -0
  162. package/dist/core/memory/secret-scanner.js +304 -0
  163. package/dist/core/memory-sync/queue.js +170 -0
  164. package/dist/core/metrics/extract.js +113 -0
  165. package/dist/core/modes/roo-modes.js +68 -0
  166. package/dist/core/onboarding/ensure-initialized.js +133 -0
  167. package/dist/core/onboarding/marker.js +111 -0
  168. package/dist/core/onboarding/telemetry-state.js +108 -0
  169. package/dist/core/output-style/presets.js +176 -0
  170. package/dist/core/output-style/state.js +185 -0
  171. package/dist/core/path-security.js +287 -5
  172. package/dist/core/permission.js +82 -22
  173. package/dist/core/permissions/auto-classifier.js +124 -0
  174. package/dist/core/permissions/bash-parser.js +371 -0
  175. package/dist/core/permissions/circuit-breaker.js +83 -0
  176. package/dist/core/permissions/constrained-edit.js +91 -0
  177. package/dist/core/permissions/gate.js +278 -0
  178. package/dist/core/permissions/index.js +20 -0
  179. package/dist/core/permissions/mode.js +174 -0
  180. package/dist/core/permissions/network-egress.js +137 -0
  181. package/dist/core/permissions/state.js +241 -0
  182. package/dist/core/permissions/tool-class.js +93 -0
  183. package/dist/core/plan-mode/ui-state.js +51 -0
  184. package/dist/core/plans/plan-artifact.js +721 -0
  185. package/dist/core/policy-limits/etag-store.js +122 -0
  186. package/dist/core/prd-check/parser.js +215 -0
  187. package/dist/core/prd-check/reporter.js +127 -0
  188. package/dist/core/prd-check/session-review.js +557 -0
  189. package/dist/core/prd-check/verifiers.js +223 -0
  190. package/dist/core/prompt-cache/client-cache.js +99 -0
  191. package/dist/core/prompts/assembly.js +29 -0
  192. package/dist/core/prompts/registry.js +364 -0
  193. package/dist/core/pugi-md/cc-compat-rules.js +735 -0
  194. package/dist/core/pugi-md/context-injector.js +76 -0
  195. package/dist/core/pugi-md/walk-up.js +207 -0
  196. package/dist/core/python/uv-installer.js +270 -0
  197. package/dist/core/python/uv-resolver.js +83 -0
  198. package/dist/core/rate-limit/narrator.js +146 -0
  199. package/dist/core/recipes/cli-types.js +20 -0
  200. package/dist/core/recipes/loader.js +103 -0
  201. package/dist/core/recipes/runner.js +345 -0
  202. package/dist/core/recipes/schema.js +587 -0
  203. package/dist/core/release-notes/parser.js +241 -0
  204. package/dist/core/release-notes/state.js +116 -0
  205. package/dist/core/repl/ask.js +37 -37
  206. package/dist/core/repl/cancellation.js +26 -26
  207. package/dist/core/repl/cap-warning.js +4 -4
  208. package/dist/core/repl/clipboard-read.js +11 -11
  209. package/dist/core/repl/dispatch-fsm.js +12 -12
  210. package/dist/core/repl/history-search.js +15 -15
  211. package/dist/core/repl/history.js +28 -18
  212. package/dist/core/repl/kill-ring.js +5 -5
  213. package/dist/core/repl/model-pricing.js +135 -0
  214. package/dist/core/repl/privacy-banner.js +22 -22
  215. package/dist/core/repl/session.js +2157 -214
  216. package/dist/core/repl/slash-commands.js +533 -40
  217. package/dist/core/repl/store/index.js +1 -1
  218. package/dist/core/repl/store/jsonl-log.js +22 -22
  219. package/dist/core/repl/store/lockfile.js +10 -10
  220. package/dist/core/repl/store/session-store.js +136 -107
  221. package/dist/core/repl/store/types.js +15 -15
  222. package/dist/core/repl/store/uuid-v7.js +12 -12
  223. package/dist/core/repl/workspace-context.js +43 -21
  224. package/dist/core/repo-map/build.js +125 -0
  225. package/dist/core/repo-map/cache.js +185 -0
  226. package/dist/core/repo-map/extractor.js +254 -0
  227. package/dist/core/repo-map/formatter.js +145 -0
  228. package/dist/core/repo-map/page-rank.js +105 -0
  229. package/dist/core/repo-map/scanner.js +211 -0
  230. package/dist/core/retry-budget/budget.js +284 -0
  231. package/dist/core/retry-budget/index.js +5 -0
  232. package/dist/core/retry-budget/retry-cap.js +74 -0
  233. package/dist/core/routing/lead-worker.js +43 -0
  234. package/dist/core/routing/pre-flight-estimator.js +108 -0
  235. package/dist/core/runs/run-tree.js +103 -0
  236. package/dist/core/security/injection-scanner.js +367 -0
  237. package/dist/core/security/output-filter.js +418 -0
  238. package/dist/core/session/env-file.js +105 -0
  239. package/dist/core/session/section-budgets.js +140 -0
  240. package/dist/core/session.js +92 -0
  241. package/dist/core/settings.js +298 -5
  242. package/dist/core/share/formatter.js +271 -0
  243. package/dist/core/share/redactor.js +221 -0
  244. package/dist/core/share/uploader.js +267 -0
  245. package/dist/core/skills/defaults.js +457 -0
  246. package/dist/core/skills/loader.js +22 -22
  247. package/dist/core/skills/sources.js +27 -27
  248. package/dist/core/smoke/headless-driver.js +174 -0
  249. package/dist/core/smoke/orchestrator.js +194 -0
  250. package/dist/core/smoke/runner.js +238 -0
  251. package/dist/core/smoke/scenario-parser.js +316 -0
  252. package/dist/core/statusline.js +99 -0
  253. package/dist/core/subagents/dispatcher-real.js +600 -0
  254. package/dist/core/subagents/dispatcher.js +132 -43
  255. package/dist/core/subagents/index.js +19 -6
  256. package/dist/core/subagents/isolation-matrix.js +213 -0
  257. package/dist/core/subagents/spawn.js +19 -4
  258. package/dist/core/telemetry/emitter.js +229 -0
  259. package/dist/core/telemetry/queue.js +251 -0
  260. package/dist/core/theme/context.js +91 -0
  261. package/dist/core/theme/presets.js +228 -0
  262. package/dist/core/theme/state.js +181 -0
  263. package/dist/core/todos/invariant.js +10 -0
  264. package/dist/core/todos/state.js +177 -0
  265. package/dist/core/tool-schema/compressor.js +89 -0
  266. package/dist/core/transport/version-interceptor.js +166 -0
  267. package/dist/core/trust.js +2 -2
  268. package/dist/core/tui/thinking-block.js +64 -0
  269. package/dist/core/vim/keymap.js +288 -0
  270. package/dist/core/vim/state.js +92 -0
  271. package/dist/core/watch-markers/marker-watcher.js +133 -0
  272. package/dist/core/worktree-manager/cleanup.js +123 -0
  273. package/dist/core/worktree-manager/manager.js +303 -0
  274. package/dist/index.js +36 -0
  275. package/dist/runtime/bootstrap.js +190 -0
  276. package/dist/runtime/cli.js +4203 -493
  277. package/dist/runtime/commands/agents.js +30 -30
  278. package/dist/runtime/commands/budget.js +5 -5
  279. package/dist/runtime/commands/cancel.js +231 -0
  280. package/dist/runtime/commands/chain.js +489 -0
  281. package/dist/runtime/commands/codegraph-status.js +227 -0
  282. package/dist/runtime/commands/compact.js +297 -0
  283. package/dist/runtime/commands/config.js +73 -39
  284. package/dist/runtime/commands/cost.js +199 -0
  285. package/dist/runtime/commands/delegate.js +244 -13
  286. package/dist/runtime/commands/dispatch.js +126 -0
  287. package/dist/runtime/commands/doctor.js +579 -0
  288. package/dist/runtime/commands/feedback.js +184 -0
  289. package/dist/runtime/commands/hooks.js +184 -0
  290. package/dist/runtime/commands/init.js +254 -0
  291. package/dist/runtime/commands/lsp.js +368 -0
  292. package/dist/runtime/commands/mcp.js +879 -0
  293. package/dist/runtime/commands/memory.js +582 -0
  294. package/dist/runtime/commands/model.js +237 -0
  295. package/dist/runtime/commands/onboarding.js +275 -0
  296. package/dist/runtime/commands/patch.js +128 -0
  297. package/dist/runtime/commands/permissions.js +112 -0
  298. package/dist/runtime/commands/plan.js +143 -0
  299. package/dist/runtime/commands/prd-check.js +285 -0
  300. package/dist/runtime/commands/privacy.js +17 -17
  301. package/dist/runtime/commands/recipe.js +325 -0
  302. package/dist/runtime/commands/redo-blob-store.js +92 -0
  303. package/dist/runtime/commands/redo.js +361 -0
  304. package/dist/runtime/commands/release-notes.js +229 -0
  305. package/dist/runtime/commands/repo-map.js +95 -0
  306. package/dist/runtime/commands/report.js +299 -0
  307. package/dist/runtime/commands/resume.js +118 -0
  308. package/dist/runtime/commands/review-consensus.js +68 -53
  309. package/dist/runtime/commands/rewind.js +333 -0
  310. package/dist/runtime/commands/roster.js +14 -14
  311. package/dist/runtime/commands/sessions.js +163 -0
  312. package/dist/runtime/commands/share.js +316 -0
  313. package/dist/runtime/commands/skills.js +31 -31
  314. package/dist/runtime/commands/status.js +186 -0
  315. package/dist/runtime/commands/stickers.js +82 -0
  316. package/dist/runtime/commands/style.js +194 -0
  317. package/dist/runtime/commands/theme.js +196 -0
  318. package/dist/runtime/commands/undo.js +54 -22
  319. package/dist/runtime/commands/update.js +289 -0
  320. package/dist/runtime/commands/vim.js +140 -0
  321. package/dist/runtime/commands/worktree.js +177 -0
  322. package/dist/runtime/commands/worktrees.js +155 -0
  323. package/dist/runtime/headless-repl.js +195 -0
  324. package/dist/runtime/headless.js +543 -0
  325. package/dist/runtime/load-hooks-or-exit.js +71 -0
  326. package/dist/runtime/plan-decompose.js +531 -0
  327. package/dist/runtime/sigint-guard.js +272 -0
  328. package/dist/runtime/update-check.js +28 -28
  329. package/dist/runtime/version.js +65 -0
  330. package/dist/skills/bundled/batch.js +617 -0
  331. package/dist/skills/bundled/index.js +45 -0
  332. package/dist/skills/bundled/loop.js +358 -0
  333. package/dist/skills/bundled/remember.js +383 -0
  334. package/dist/skills/bundled/simplify.js +289 -0
  335. package/dist/skills/bundled/skillify.js +373 -0
  336. package/dist/skills/bundled/stuck.js +558 -0
  337. package/dist/skills/bundled/verify.js +439 -0
  338. package/dist/testing/vcr.js +486 -0
  339. package/dist/tools/agent-tool.js +229 -0
  340. package/dist/tools/apply-patch.js +556 -0
  341. package/dist/tools/ask-user-question.js +288 -0
  342. package/dist/tools/ask-user.js +115 -0
  343. package/dist/tools/bash.js +624 -46
  344. package/dist/tools/brief.js +224 -0
  345. package/dist/tools/enter-worktree.js +250 -0
  346. package/dist/tools/exit-worktree.js +147 -0
  347. package/dist/tools/file-tools.js +161 -44
  348. package/dist/tools/lsp-tools.js +189 -0
  349. package/dist/tools/mcp-tool.js +260 -0
  350. package/dist/tools/multi-edit.js +361 -0
  351. package/dist/tools/powershell.js +268 -0
  352. package/dist/tools/registry.js +85 -0
  353. package/dist/tools/skill-tool.js +96 -0
  354. package/dist/tools/sleep.js +99 -0
  355. package/dist/tools/synthetic-output.js +133 -0
  356. package/dist/tools/tasks.js +208 -0
  357. package/dist/tools/todo-write.js +184 -0
  358. package/dist/tools/verify-plan-execution.js +295 -0
  359. package/dist/tools/web-fetch-injection-scanner.js +207 -0
  360. package/dist/tools/web-fetch.js +195 -10
  361. package/dist/tools/web-search.js +458 -0
  362. package/dist/tui/agent-progress-card.js +111 -0
  363. package/dist/tui/agent-tree.js +11 -1
  364. package/dist/tui/ask-modal.js +14 -14
  365. package/dist/tui/ask-user-question-chips.js +257 -0
  366. package/dist/tui/ask-user-question-prompt.js +203 -0
  367. package/dist/tui/compact-banner.js +81 -0
  368. package/dist/tui/conversation-pane.js +85 -11
  369. package/dist/tui/cost-table.js +111 -0
  370. package/dist/tui/device-flow.js +2 -2
  371. package/dist/tui/doctor-table.js +46 -0
  372. package/dist/tui/feedback-prompt.js +156 -0
  373. package/dist/tui/input-box.js +247 -32
  374. package/dist/tui/login-picker.js +3 -3
  375. package/dist/tui/markdown-render.js +6 -6
  376. package/dist/tui/onboarding-wizard.js +240 -0
  377. package/dist/tui/permissions-picker.js +86 -0
  378. package/dist/tui/render.js +35 -0
  379. package/dist/tui/repl-render.js +332 -54
  380. package/dist/tui/repl-splash-art.js +16 -16
  381. package/dist/tui/repl-splash-mascot.js +48 -24
  382. package/dist/tui/repl-splash.js +22 -22
  383. package/dist/tui/repl.js +124 -44
  384. package/dist/tui/slash-palette.js +6 -6
  385. package/dist/tui/splash.js +2 -2
  386. package/dist/tui/status-bar.js +109 -31
  387. package/dist/tui/status-table.js +7 -0
  388. package/dist/tui/stickers-art.js +136 -0
  389. package/dist/tui/style-table.js +28 -0
  390. package/dist/tui/theme-table.js +29 -0
  391. package/dist/tui/thinking-spinner.js +123 -0
  392. package/dist/tui/tool-stream-pane.js +53 -4
  393. package/dist/tui/update-banner.js +27 -2
  394. package/dist/tui/vim-input.js +267 -0
  395. package/dist/tui/welcome-banner.js +107 -0
  396. package/dist/tui/welcome-data.js +293 -0
  397. package/dist/tui/workspace-context.js +2 -2
  398. package/docs/examples/codegraph.mcp.json +10 -0
  399. package/package.json +25 -7
  400. package/test/scenarios/codegen-create-file.scenario.txt +13 -0
  401. package/test/scenarios/compact-force.scenario.txt +11 -0
  402. package/test/scenarios/identity.scenario.txt +11 -0
  403. package/test/scenarios/persona-handoff.scenario.txt +11 -0
  404. package/test/scenarios/walkback.scenario.txt +12 -0
  405. package/dist/core/engine/compaction-hook.js +0 -154
@@ -0,0 +1,288 @@
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
+ /** PUGI-480 short-format chip rules: ≤ 5 words / option label, ≤ 3 questions / call. */
52
+ export const ASK_USER_QUESTION_CHIP_LABEL_WORD_MAX = 5;
53
+ export const ASK_USER_QUESTION_CHIPS_MAX = 3;
54
+ /**
55
+ * Reusable validator for option labels rendered в chip mode.
56
+ * Counts whitespace-delimited words. Throws с a clear message that
57
+ * names the offending question + option so the model can self-correct.
58
+ */
59
+ export function assertChipLabelWordCap(questionHeader, optionLabel) {
60
+ const words = optionLabel.trim().split(/\s+/u).filter((w) => w.length > 0);
61
+ if (words.length > ASK_USER_QUESTION_CHIP_LABEL_WORD_MAX) {
62
+ throw new Error(`ask_user_question chip "${questionHeader}" option "${optionLabel}" ` +
63
+ `exceeds the ${ASK_USER_QUESTION_CHIP_LABEL_WORD_MAX}-word label cap ` +
64
+ `(saw ${words.length} words). Shorten the label before dispatch.`);
65
+ }
66
+ }
67
+ /**
68
+ * Structured option. `label` is the display text; `description` is the
69
+ * implication line shown dim below it. Both are required — the model
70
+ * cannot ship a label-only option (forces it to think about why each
71
+ * choice exists).
72
+ */
73
+ export const askUserQuestionOptionSchema = z.strictObject({
74
+ label: z
75
+ .string()
76
+ .min(ASK_USER_QUESTION_OPTION_LABEL_MIN)
77
+ .max(ASK_USER_QUESTION_OPTION_LABEL_MAX)
78
+ .describe('Display text. Concise (1-5 words).'),
79
+ description: z
80
+ .string()
81
+ .min(ASK_USER_QUESTION_OPTION_DESC_MIN)
82
+ .max(ASK_USER_QUESTION_OPTION_DESC_MAX)
83
+ .describe('What this option means / implications.'),
84
+ preview: z
85
+ .string()
86
+ .min(1)
87
+ .max(2000)
88
+ .optional()
89
+ .describe('Optional ASCII / code preview. When ANY option in the set carries ' +
90
+ 'this field, the UI switches к side-by-side layout (options column ' +
91
+ 'left, preview pane right). Use для visual comparison of mockups, ' +
92
+ 'config snippets, diagram variations.'),
93
+ });
94
+ export const askUserQuestionSchema = z.strictObject({
95
+ question: z
96
+ .string()
97
+ .min(ASK_USER_QUESTION_MIN)
98
+ .max(ASK_USER_QUESTION_MAX)
99
+ .refine((q) => q.trim().endsWith('?'), {
100
+ message: 'question must end with "?"',
101
+ })
102
+ .describe('The complete question. Must end with "?". Plain English, no jargon.'),
103
+ header: z
104
+ .string()
105
+ .min(ASK_USER_QUESTION_HEADER_MIN)
106
+ .max(ASK_USER_QUESTION_HEADER_MAX)
107
+ .describe('Short chip label (max 12 chars). E.g. "Auth method".'),
108
+ options: z
109
+ .array(askUserQuestionOptionSchema)
110
+ .min(ASK_USER_QUESTION_OPTIONS_MIN)
111
+ .max(ASK_USER_QUESTION_OPTIONS_MAX)
112
+ .describe('2-4 mutually-exclusive options. NEVER add "Other" — UI auto-adds.'),
113
+ multiSelect: z
114
+ .boolean()
115
+ .optional()
116
+ .default(false)
117
+ .describe('Allow multiple selections. Default false.'),
118
+ });
119
+ /**
120
+ * PUGI-480 multi-question chip payload — a bundle of up to 3 short-format
121
+ * chip questions rendered side-by-side. Schema is intentionally narrow:
122
+ * the chip renderer relies on the ≤ 5-word label invariant и will
123
+ * truncate с "…" if the model ever bypasses Zod. The 3-question cap
124
+ * forecloses paragraph-wall prompts at the schema level — the model
125
+ * cannot legally ask "10 quick questions" in one shot.
126
+ */
127
+ export const askUserQuestionChipsQuestionSchema = z.strictObject({
128
+ header: z
129
+ .string()
130
+ .min(ASK_USER_QUESTION_HEADER_MIN)
131
+ .max(ASK_USER_QUESTION_HEADER_MAX)
132
+ .describe('Short chip label (max 12 chars). E.g. "Stack".'),
133
+ question: z
134
+ .string()
135
+ .min(ASK_USER_QUESTION_MIN)
136
+ .max(ASK_USER_QUESTION_MAX)
137
+ .optional()
138
+ .describe('Optional full-prose question (shown in non-TTY fallback only).'),
139
+ options: z
140
+ .array(askUserQuestionOptionSchema)
141
+ .min(ASK_USER_QUESTION_OPTIONS_MIN)
142
+ .max(ASK_USER_QUESTION_OPTIONS_MAX + 1)
143
+ .superRefine((opts, ctx) => {
144
+ // Enforce ≤ 5-word label cap on every option. Schema-level so
145
+ // the model gets immediate feedback on overflow rather than a
146
+ // silent truncation at render time.
147
+ for (const opt of opts) {
148
+ const words = opt.label
149
+ .trim()
150
+ .split(/\s+/u)
151
+ .filter((w) => w.length > 0);
152
+ if (words.length > ASK_USER_QUESTION_CHIP_LABEL_WORD_MAX) {
153
+ ctx.addIssue({
154
+ code: z.ZodIssueCode.custom,
155
+ message: `option "${opt.label}" has ${words.length} words; chip labels must be ≤ ${ASK_USER_QUESTION_CHIP_LABEL_WORD_MAX} words`,
156
+ });
157
+ }
158
+ }
159
+ })
160
+ .describe('2-5 mutually-exclusive options. Each label ≤ 5 words. UI inserts "Skip — use defaults" as a final option when defaults are present.'),
161
+ });
162
+ export const askUserQuestionChipsSchema = z.strictObject({
163
+ questions: z
164
+ .array(askUserQuestionChipsQuestionSchema)
165
+ .min(1)
166
+ .max(ASK_USER_QUESTION_CHIPS_MAX)
167
+ .describe('Bundle of 1-3 short clarifier questions rendered side-by-side as chips.'),
168
+ });
169
+ /**
170
+ * Dispatch the structured tool: validate args via Zod, then route
171
+ * through the shared `askUser` primitive so abort/timeout/non-TTY
172
+ * envelope behaviour is identical to the legacy form.
173
+ *
174
+ * The bridge surface is the same `AskUserBridge` signature — the
175
+ * structured form just gives the Ink modal richer metadata to render
176
+ * (header chip + per-option description). The bridge sees the legacy
177
+ * `{question, options: string[]}` shape because all production bridges
178
+ * (Ink modal + non-TTY envelope emitter) already consume that shape.
179
+ * Per-option descriptions and the header chip are surfaced separately
180
+ * via `enrich` — the modal layer reads them off the dispatched payload
181
+ * stash, NOT off the bridge input, so structured callers can layer on
182
+ * top of the legacy interface without touching the modal contract.
183
+ *
184
+ * Return contract:
185
+ * - Interactive + bridge present + operator picks N options →
186
+ * `[ask_user_question:answered] <labels joined by ", ">`.
187
+ * - Interactive + bridge present + operator cancels →
188
+ * `[ask_user_question:cancelled]`.
189
+ * - Interactive + bridge present + timeout →
190
+ * `[ask_user_question:timeout]`.
191
+ * - Non-TTY or no bridge → `[user_input_required]<json>[/...]`
192
+ * envelope identical to the legacy form. Includes `header` +
193
+ * structured options so a scripted caller can parse the full shape.
194
+ */
195
+ export async function dispatchAskUserQuestion(ctx, rawArgs) {
196
+ const parsed = askUserQuestionSchema.parse(rawArgs);
197
+ // Schema-level guard against the "Other" leak: the prompt rules tell
198
+ // the model NEVER to include "Other" in `options`, but we still reject
199
+ // it defensively in case a future model misreads the spec. The Ink
200
+ // modal auto-appends "Other" itself; a model-supplied duplicate would
201
+ // render two "Other" rows.
202
+ for (const opt of parsed.options) {
203
+ const trimmed = opt.label.trim().toLowerCase();
204
+ if (trimmed === 'other' || trimmed === 'другое') {
205
+ throw new Error('ask_user_question: do NOT include "Other" in options — UI auto-adds it');
206
+ }
207
+ }
208
+ const legacyOptions = parsed.options.map((opt) => opt.label);
209
+ const result = await askUser(ctx, {
210
+ question: parsed.question,
211
+ options: legacyOptions,
212
+ multiSelect: parsed.multiSelect ?? false,
213
+ });
214
+ if (result.answers && result.answers.length > 0) {
215
+ return {
216
+ answers: result.answers,
217
+ envelope: `[ask_user_question:answered] ${result.answers.join(', ')}`,
218
+ };
219
+ }
220
+ // Non-TTY / cancelled / timeout. Re-wrap the envelope so callers can
221
+ // grep for the structured tool name even when the underlying primitive
222
+ // surfaced its legacy `[user_input_required]` envelope.
223
+ if (result.envelope.includes('"reason":"timeout"')) {
224
+ return { envelope: '[ask_user_question:timeout]' };
225
+ }
226
+ if (result.envelope.includes('"reason":"cancelled"')) {
227
+ return { envelope: '[ask_user_question:cancelled]' };
228
+ }
229
+ // Default to the legacy envelope verbatim — it is still
230
+ // grep-friendly and includes the structured payload above.
231
+ return { envelope: result.envelope };
232
+ }
233
+ /**
234
+ * JSON-Schema fragment surfaced to the model via the tool-bridge
235
+ * `parameters` field. Mirrors the Zod schema 1:1 — kept hand-written
236
+ * because the runtime engine wires OpenAI-compatible JSON Schema and
237
+ * the Zod-to-JSON-Schema converter pulls in a transitive dep we have
238
+ * not greenlit. If the Zod schema above changes, mirror the change here.
239
+ */
240
+ export const askUserQuestionJsonSchema = {
241
+ type: 'object',
242
+ additionalProperties: false,
243
+ required: ['question', 'header', 'options'],
244
+ properties: {
245
+ question: {
246
+ type: 'string',
247
+ minLength: ASK_USER_QUESTION_MIN,
248
+ maxLength: ASK_USER_QUESTION_MAX,
249
+ description: 'The complete question. Must end with "?". Plain English, no jargon.',
250
+ },
251
+ header: {
252
+ type: 'string',
253
+ minLength: ASK_USER_QUESTION_HEADER_MIN,
254
+ maxLength: ASK_USER_QUESTION_HEADER_MAX,
255
+ description: 'Short chip label (max 12 chars). E.g. "Auth method".',
256
+ },
257
+ options: {
258
+ type: 'array',
259
+ minItems: ASK_USER_QUESTION_OPTIONS_MIN,
260
+ maxItems: ASK_USER_QUESTION_OPTIONS_MAX,
261
+ items: {
262
+ type: 'object',
263
+ additionalProperties: false,
264
+ required: ['label', 'description'],
265
+ properties: {
266
+ label: {
267
+ type: 'string',
268
+ minLength: ASK_USER_QUESTION_OPTION_LABEL_MIN,
269
+ maxLength: ASK_USER_QUESTION_OPTION_LABEL_MAX,
270
+ description: 'Display text. Concise (1-5 words).',
271
+ },
272
+ description: {
273
+ type: 'string',
274
+ minLength: ASK_USER_QUESTION_OPTION_DESC_MIN,
275
+ maxLength: ASK_USER_QUESTION_OPTION_DESC_MAX,
276
+ description: 'What this option means / implications.',
277
+ },
278
+ },
279
+ },
280
+ description: '2-4 mutually-exclusive options. NEVER add "Other" — UI auto-adds.',
281
+ },
282
+ multiSelect: {
283
+ type: 'boolean',
284
+ description: 'Allow multiple selections. Default false.',
285
+ },
286
+ },
287
+ };
288
+ //# 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