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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (409) hide show
  1. package/CHANGELOG.md +132 -0
  2. package/LICENSE +1 -1
  3. package/assets/pugi-prozr2-mascot.ansi +9 -0
  4. package/bin/run.js +33 -1
  5. package/dist/commands/deploy.js +40 -40
  6. package/dist/commands/flatten.js +191 -0
  7. package/dist/commands/jobs-watch.js +201 -0
  8. package/dist/commands/jobs.js +42 -27
  9. package/dist/commands/smoke.js +133 -0
  10. package/dist/core/agent-progress/cleanup.js +134 -0
  11. package/dist/core/agent-progress/schema.js +144 -0
  12. package/dist/core/agent-progress/writer.js +101 -0
  13. package/dist/core/agents/adaptive-router.js +330 -0
  14. package/dist/core/agents/query-decomposer.js +297 -0
  15. package/dist/core/agents/registry.js +3 -3
  16. package/dist/core/approvals/shortcut-resolver.js +98 -0
  17. package/dist/core/artifact-chain/dispatcher.js +148 -0
  18. package/dist/core/artifact-chain/exporter.js +164 -0
  19. package/dist/core/artifact-chain/state.js +243 -0
  20. package/dist/core/artifact-chain/steps.js +169 -0
  21. package/dist/core/ask-user/question.js +92 -0
  22. package/dist/core/audit/audit-trail.js +275 -0
  23. package/dist/core/auth/ensure-authenticated.js +129 -0
  24. package/dist/core/auth/env-provider.js +238 -0
  25. package/dist/core/auto-open-browser.js +4 -4
  26. package/dist/core/auto-update/channels.js +122 -0
  27. package/dist/core/auto-update/checker.js +241 -0
  28. package/dist/core/auto-update/state.js +235 -0
  29. package/dist/core/bare-mode/index.js +107 -0
  30. package/dist/core/bash/redirect.js +281 -0
  31. package/dist/core/bash-classifier.js +436 -40
  32. package/dist/core/checkpoint/resumer.js +149 -0
  33. package/dist/core/checkpoint/rewinder.js +291 -0
  34. package/dist/core/checkpoints/shadow-git.js +670 -0
  35. package/dist/core/citations/parser.js +109 -0
  36. package/dist/core/classifier/yolo-classifier.js +88 -0
  37. package/dist/core/codegraph/decision-store.js +248 -0
  38. package/dist/core/codegraph/detect-repo.js +459 -0
  39. package/dist/core/codegraph/install.js +134 -0
  40. package/dist/core/codegraph/offer-hook.js +220 -0
  41. package/dist/core/compact/auto-trigger.js +96 -0
  42. package/dist/core/compact/buffer-rewriter.js +115 -0
  43. package/dist/core/compact/summarizer.js +208 -0
  44. package/dist/core/compact/token-counter.js +108 -0
  45. package/dist/core/consensus/anvil-fanout.js +25 -25
  46. package/dist/core/consensus/diff-capture.js +121 -12
  47. package/dist/core/consensus/rubric.js +21 -21
  48. package/dist/core/context/builder.js +6 -6
  49. package/dist/core/context/compaction-events.js +8 -8
  50. package/dist/core/context/compaction.js +31 -31
  51. package/dist/core/context/index.js +15 -8
  52. package/dist/core/context/invariants.js +51 -51
  53. package/dist/core/context/markdown-loader.js +28 -10
  54. package/dist/core/context/markdown-traverse.js +255 -0
  55. package/dist/core/context/pugiignore.js +41 -41
  56. package/dist/core/context/repo-skeleton.js +37 -37
  57. package/dist/core/context/tool-eviction.js +55 -0
  58. package/dist/core/context/watcher.js +32 -32
  59. package/dist/core/context/working-set.js +23 -23
  60. package/dist/core/coordinator/agent-tools.js +77 -0
  61. package/dist/core/coordinator/agent-toolset.js +65 -0
  62. package/dist/core/coordinator/fsm.js +73 -0
  63. package/dist/core/coordinator/mode-fsm.js +70 -0
  64. package/dist/core/cost/rate-card.js +129 -0
  65. package/dist/core/cost/tracker.js +221 -0
  66. package/dist/core/credentials.js +13 -13
  67. package/dist/core/cron/scheduler.js +138 -0
  68. package/dist/core/denial-tracking/index.js +8 -0
  69. package/dist/core/denial-tracking/state.js +264 -0
  70. package/dist/core/diagnostics/probe-runner.js +93 -0
  71. package/dist/core/diagnostics/probes/api.js +46 -0
  72. package/dist/core/diagnostics/probes/auth.js +93 -0
  73. package/dist/core/diagnostics/probes/bare-mode.js +42 -0
  74. package/dist/core/diagnostics/probes/cli-version.js +127 -0
  75. package/dist/core/diagnostics/probes/config.js +72 -0
  76. package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
  77. package/dist/core/diagnostics/probes/disk.js +81 -0
  78. package/dist/core/diagnostics/probes/engine-live.js +46 -0
  79. package/dist/core/diagnostics/probes/git.js +65 -0
  80. package/dist/core/diagnostics/probes/hooks.js +118 -0
  81. package/dist/core/diagnostics/probes/mcp.js +75 -0
  82. package/dist/core/diagnostics/probes/node.js +59 -0
  83. package/dist/core/diagnostics/probes/pnpm.js +36 -0
  84. package/dist/core/diagnostics/probes/pugi-md.js +89 -0
  85. package/dist/core/diagnostics/probes/sandbox.js +40 -0
  86. package/dist/core/diagnostics/probes/session.js +74 -0
  87. package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
  88. package/dist/core/diagnostics/probes/workspace.js +63 -0
  89. package/dist/core/diagnostics/types.js +70 -0
  90. package/dist/core/dispatch/cache-cleanup.js +197 -0
  91. package/dist/core/dispatch/cache-handoff.js +295 -0
  92. package/dist/core/edits/apply-patch-layer-e.js +189 -0
  93. package/dist/core/edits/dispatch.js +333 -7
  94. package/dist/core/edits/format-detector.js +260 -0
  95. package/dist/core/edits/format-matrix.js +26 -0
  96. package/dist/core/edits/fuzzy-ladder.js +650 -0
  97. package/dist/core/edits/index.js +5 -1
  98. package/dist/core/edits/journal.js +199 -0
  99. package/dist/core/edits/layer-a-apply.js +15 -15
  100. package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
  101. package/dist/core/edits/layer-b-apply.js +9 -9
  102. package/dist/core/edits/layer-c-apply.js +6 -6
  103. package/dist/core/edits/layer-d-ast.js +557 -14
  104. package/dist/core/edits/marker-parser.js +12 -12
  105. package/dist/core/edits/security-gate.js +27 -27
  106. package/dist/core/edits/verify-hook.js +273 -0
  107. package/dist/core/edits/worktree.js +29 -29
  108. package/dist/core/engine/anvil-client.js +214 -26
  109. package/dist/core/engine/auto-compact.js +179 -0
  110. package/dist/core/engine/budgets.js +186 -0
  111. package/dist/core/engine/context-prefix.js +155 -0
  112. package/dist/core/engine/index.js +1 -1
  113. package/dist/core/engine/intensity.js +158 -0
  114. package/dist/core/engine/intent.js +260 -0
  115. package/dist/core/engine/native-pugi.js +1295 -227
  116. package/dist/core/engine/prompts.js +129 -19
  117. package/dist/core/engine/strip-internal-fields.js +124 -0
  118. package/dist/core/engine/tool-bridge.js +1731 -59
  119. package/dist/core/evaluation/golden-dataset.js +293 -0
  120. package/dist/core/feedback/queue.js +177 -0
  121. package/dist/core/feedback/submitter.js +145 -0
  122. package/dist/core/file-cache.js +113 -1
  123. package/dist/core/flatten/flatten-repo.js +439 -0
  124. package/dist/core/format/osc8-link.js +28 -0
  125. package/dist/core/hook-chains.js +392 -0
  126. package/dist/core/hooks/citation-verify-hook.js +138 -0
  127. package/dist/core/hooks/citation-verify.js +112 -0
  128. package/dist/core/hooks/events.js +46 -0
  129. package/dist/core/hooks/index.js +15 -0
  130. package/dist/core/hooks/registry.js +216 -0
  131. package/dist/core/hooks/runner.js +236 -0
  132. package/dist/core/hooks/v2/event-emitter.js +115 -0
  133. package/dist/core/hooks/v2/executor.js +282 -0
  134. package/dist/core/hooks/v2/index.js +25 -0
  135. package/dist/core/hooks/v2/lifecycle.js +104 -0
  136. package/dist/core/hooks/v2/loader.js +216 -0
  137. package/dist/core/hooks/v2/matcher.js +125 -0
  138. package/dist/core/hooks/v2/trust.js +143 -0
  139. package/dist/core/hooks/v2/types.js +86 -0
  140. package/dist/core/hooks/worktree-events.js +158 -0
  141. package/dist/core/image/renderer.js +71 -0
  142. package/dist/core/init/detector.js +582 -0
  143. package/dist/core/init/template-renderer.js +242 -0
  144. package/dist/core/jobs/registry.js +18 -18
  145. package/dist/core/ledger/results-tsv.js +142 -0
  146. package/dist/core/log-discipline/stdout-redirect.js +51 -0
  147. package/dist/core/lsp/cache.js +105 -0
  148. package/dist/core/lsp/client.js +551 -41
  149. package/dist/core/lsp/language-detect.js +66 -0
  150. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  151. package/dist/core/lsp/server-detect.js +173 -0
  152. package/dist/core/lsp/symbol-cache.js +162 -0
  153. package/dist/core/lsp/symbol-tools.js +664 -0
  154. package/dist/core/mcp/client.js +97 -28
  155. package/dist/core/mcp/http-server.js +553 -0
  156. package/dist/core/mcp/orchestrator-tools.js +662 -0
  157. package/dist/core/mcp/permission.js +190 -0
  158. package/dist/core/mcp/registry.js +39 -17
  159. package/dist/core/mcp/server-tools.js +219 -0
  160. package/dist/core/mcp/server.js +397 -0
  161. package/dist/core/mcp/trust.js +10 -10
  162. package/dist/core/memory/dual-write.js +416 -0
  163. package/dist/core/memory/passive-extract.js +130 -0
  164. package/dist/core/memory/phase1-kinds.js +20 -0
  165. package/dist/core/memory/secret-scanner.js +304 -0
  166. package/dist/core/memory-sync/queue.js +170 -0
  167. package/dist/core/metrics/extract.js +113 -0
  168. package/dist/core/modes/roo-modes.js +68 -0
  169. package/dist/core/onboarding/ensure-initialized.js +133 -0
  170. package/dist/core/onboarding/marker.js +111 -0
  171. package/dist/core/onboarding/telemetry-state.js +108 -0
  172. package/dist/core/output-style/presets.js +176 -0
  173. package/dist/core/output-style/state.js +185 -0
  174. package/dist/core/path-security.js +287 -5
  175. package/dist/core/permission.js +82 -22
  176. package/dist/core/permissions/auto-classifier.js +124 -0
  177. package/dist/core/permissions/bash-parser.js +371 -0
  178. package/dist/core/permissions/circuit-breaker.js +83 -0
  179. package/dist/core/permissions/constrained-edit.js +91 -0
  180. package/dist/core/permissions/gate.js +278 -0
  181. package/dist/core/permissions/index.js +20 -0
  182. package/dist/core/permissions/mode.js +174 -0
  183. package/dist/core/permissions/network-egress.js +137 -0
  184. package/dist/core/permissions/state.js +241 -0
  185. package/dist/core/permissions/tool-class.js +93 -0
  186. package/dist/core/plan-mode/ui-state.js +51 -0
  187. package/dist/core/plans/plan-artifact.js +721 -0
  188. package/dist/core/policy-limits/etag-store.js +122 -0
  189. package/dist/core/prd-check/parser.js +215 -0
  190. package/dist/core/prd-check/reporter.js +127 -0
  191. package/dist/core/prd-check/session-review.js +557 -0
  192. package/dist/core/prd-check/verifiers.js +223 -0
  193. package/dist/core/prompt-cache/client-cache.js +99 -0
  194. package/dist/core/prompts/assembly.js +29 -0
  195. package/dist/core/prompts/registry.js +364 -0
  196. package/dist/core/pugi-md/cc-compat-rules.js +735 -0
  197. package/dist/core/pugi-md/context-injector.js +76 -0
  198. package/dist/core/pugi-md/walk-up.js +207 -0
  199. package/dist/core/python/uv-installer.js +270 -0
  200. package/dist/core/python/uv-resolver.js +83 -0
  201. package/dist/core/rate-limit/narrator.js +146 -0
  202. package/dist/core/recipes/cli-types.js +20 -0
  203. package/dist/core/recipes/loader.js +103 -0
  204. package/dist/core/recipes/runner.js +345 -0
  205. package/dist/core/recipes/schema.js +587 -0
  206. package/dist/core/release-notes/parser.js +241 -0
  207. package/dist/core/release-notes/state.js +116 -0
  208. package/dist/core/repl/ask.js +37 -37
  209. package/dist/core/repl/cancellation.js +26 -26
  210. package/dist/core/repl/cap-warning.js +4 -4
  211. package/dist/core/repl/clipboard-read.js +11 -11
  212. package/dist/core/repl/dispatch-fsm.js +12 -12
  213. package/dist/core/repl/history-search.js +15 -15
  214. package/dist/core/repl/history.js +28 -18
  215. package/dist/core/repl/kill-ring.js +5 -5
  216. package/dist/core/repl/model-pricing.js +135 -0
  217. package/dist/core/repl/privacy-banner.js +22 -22
  218. package/dist/core/repl/session.js +2148 -217
  219. package/dist/core/repl/slash-commands.js +501 -41
  220. package/dist/core/repl/store/index.js +1 -1
  221. package/dist/core/repl/store/jsonl-log.js +22 -22
  222. package/dist/core/repl/store/lockfile.js +10 -10
  223. package/dist/core/repl/store/session-store.js +136 -107
  224. package/dist/core/repl/store/types.js +15 -15
  225. package/dist/core/repl/store/uuid-v7.js +12 -12
  226. package/dist/core/repl/workspace-context.js +43 -21
  227. package/dist/core/repo-map/build.js +125 -0
  228. package/dist/core/repo-map/cache.js +185 -0
  229. package/dist/core/repo-map/extractor.js +254 -0
  230. package/dist/core/repo-map/formatter.js +145 -0
  231. package/dist/core/repo-map/page-rank.js +105 -0
  232. package/dist/core/repo-map/scanner.js +211 -0
  233. package/dist/core/retry-budget/budget.js +284 -0
  234. package/dist/core/retry-budget/index.js +5 -0
  235. package/dist/core/retry-budget/retry-cap.js +74 -0
  236. package/dist/core/routing/lead-worker.js +43 -0
  237. package/dist/core/routing/pre-flight-estimator.js +108 -0
  238. package/dist/core/runs/run-tree.js +103 -0
  239. package/dist/core/security/injection-scanner.js +367 -0
  240. package/dist/core/security/output-filter.js +418 -0
  241. package/dist/core/session/env-file.js +105 -0
  242. package/dist/core/session/section-budgets.js +140 -0
  243. package/dist/core/session.js +92 -0
  244. package/dist/core/settings.js +324 -5
  245. package/dist/core/share/formatter.js +271 -0
  246. package/dist/core/share/redactor.js +221 -0
  247. package/dist/core/share/uploader.js +267 -0
  248. package/dist/core/skills/defaults.js +30 -30
  249. package/dist/core/skills/loader.js +22 -22
  250. package/dist/core/skills/sources.js +27 -27
  251. package/dist/core/smoke/headless-driver.js +174 -0
  252. package/dist/core/smoke/orchestrator.js +194 -0
  253. package/dist/core/smoke/runner.js +238 -0
  254. package/dist/core/smoke/scenario-parser.js +316 -0
  255. package/dist/core/statusline.js +99 -0
  256. package/dist/core/subagents/dispatcher-real.js +600 -0
  257. package/dist/core/subagents/dispatcher.js +132 -43
  258. package/dist/core/subagents/index.js +19 -6
  259. package/dist/core/subagents/isolation-matrix.js +213 -0
  260. package/dist/core/subagents/spawn.js +19 -4
  261. package/dist/core/telemetry/emitter.js +229 -0
  262. package/dist/core/telemetry/queue.js +251 -0
  263. package/dist/core/theme/context.js +91 -0
  264. package/dist/core/theme/presets.js +228 -0
  265. package/dist/core/theme/state.js +181 -0
  266. package/dist/core/todos/invariant.js +10 -0
  267. package/dist/core/todos/state.js +177 -0
  268. package/dist/core/tool-schema/compressor.js +89 -0
  269. package/dist/core/transport/version-interceptor.js +166 -0
  270. package/dist/core/trust.js +2 -2
  271. package/dist/core/tui/thinking-block.js +64 -0
  272. package/dist/core/vim/keymap.js +288 -0
  273. package/dist/core/vim/state.js +92 -0
  274. package/dist/core/watch-markers/marker-watcher.js +133 -0
  275. package/dist/core/worktree/include-parser.js +249 -0
  276. package/dist/core/worktree-manager/cleanup.js +123 -0
  277. package/dist/core/worktree-manager/manager.js +303 -0
  278. package/dist/index.js +36 -0
  279. package/dist/runtime/bootstrap.js +190 -0
  280. package/dist/runtime/cli.js +4185 -549
  281. package/dist/runtime/commands/agents.js +31 -31
  282. package/dist/runtime/commands/budget.js +5 -5
  283. package/dist/runtime/commands/cancel.js +231 -0
  284. package/dist/runtime/commands/chain.js +489 -0
  285. package/dist/runtime/commands/codegraph-status.js +227 -0
  286. package/dist/runtime/commands/compact.js +297 -0
  287. package/dist/runtime/commands/config.js +73 -39
  288. package/dist/runtime/commands/cost.js +199 -0
  289. package/dist/runtime/commands/delegate.js +27 -4
  290. package/dist/runtime/commands/dispatch.js +126 -0
  291. package/dist/runtime/commands/doctor.js +579 -0
  292. package/dist/runtime/commands/feedback.js +184 -0
  293. package/dist/runtime/commands/hooks.js +187 -0
  294. package/dist/runtime/commands/init.js +254 -0
  295. package/dist/runtime/commands/lsp.js +200 -38
  296. package/dist/runtime/commands/mcp.js +879 -0
  297. package/dist/runtime/commands/memory.js +582 -0
  298. package/dist/runtime/commands/model.js +237 -0
  299. package/dist/runtime/commands/onboarding.js +275 -0
  300. package/dist/runtime/commands/patch.js +12 -12
  301. package/dist/runtime/commands/permissions.js +112 -0
  302. package/dist/runtime/commands/plan.js +143 -0
  303. package/dist/runtime/commands/prd-check.js +285 -0
  304. package/dist/runtime/commands/privacy.js +17 -17
  305. package/dist/runtime/commands/recipe.js +325 -0
  306. package/dist/runtime/commands/redo-blob-store.js +92 -0
  307. package/dist/runtime/commands/redo.js +361 -0
  308. package/dist/runtime/commands/release-notes.js +229 -0
  309. package/dist/runtime/commands/repo-map.js +95 -0
  310. package/dist/runtime/commands/report.js +299 -0
  311. package/dist/runtime/commands/resume.js +118 -0
  312. package/dist/runtime/commands/review-consensus.js +68 -53
  313. package/dist/runtime/commands/rewind.js +333 -0
  314. package/dist/runtime/commands/roster.js +14 -14
  315. package/dist/runtime/commands/sessions.js +163 -0
  316. package/dist/runtime/commands/share.js +316 -0
  317. package/dist/runtime/commands/skills.js +31 -31
  318. package/dist/runtime/commands/status.js +186 -0
  319. package/dist/runtime/commands/stickers.js +82 -0
  320. package/dist/runtime/commands/style.js +194 -0
  321. package/dist/runtime/commands/theme.js +196 -0
  322. package/dist/runtime/commands/undo.js +54 -22
  323. package/dist/runtime/commands/update.js +289 -0
  324. package/dist/runtime/commands/vim.js +140 -0
  325. package/dist/runtime/commands/worktree.js +8 -8
  326. package/dist/runtime/commands/worktrees.js +155 -0
  327. package/dist/runtime/headless-repl.js +195 -0
  328. package/dist/runtime/headless.js +543 -0
  329. package/dist/runtime/load-hooks-or-exit.js +71 -0
  330. package/dist/runtime/plan-decompose.js +22 -22
  331. package/dist/runtime/sigint-guard.js +272 -0
  332. package/dist/runtime/update-check.js +28 -28
  333. package/dist/runtime/version.js +65 -0
  334. package/dist/runtime/worktree-bootstrap.js +579 -0
  335. package/dist/skills/bundled/batch.js +617 -0
  336. package/dist/skills/bundled/index.js +45 -0
  337. package/dist/skills/bundled/loop.js +358 -0
  338. package/dist/skills/bundled/remember.js +383 -0
  339. package/dist/skills/bundled/simplify.js +289 -0
  340. package/dist/skills/bundled/skillify.js +373 -0
  341. package/dist/skills/bundled/stuck.js +558 -0
  342. package/dist/skills/bundled/verify.js +439 -0
  343. package/dist/testing/vcr.js +486 -0
  344. package/dist/tools/agent-tool.js +229 -0
  345. package/dist/tools/apply-patch.js +89 -28
  346. package/dist/tools/ask-user-question.js +337 -0
  347. package/dist/tools/ask-user.js +115 -0
  348. package/dist/tools/bash.js +624 -46
  349. package/dist/tools/brief.js +224 -0
  350. package/dist/tools/enter-worktree.js +250 -0
  351. package/dist/tools/exit-worktree.js +147 -0
  352. package/dist/tools/file-tools.js +161 -44
  353. package/dist/tools/lsp-tools.js +377 -1
  354. package/dist/tools/mcp-tool.js +260 -0
  355. package/dist/tools/multi-edit.js +361 -0
  356. package/dist/tools/powershell.js +268 -0
  357. package/dist/tools/registry.js +86 -4
  358. package/dist/tools/skill-tool.js +96 -0
  359. package/dist/tools/sleep.js +99 -0
  360. package/dist/tools/synthetic-output.js +133 -0
  361. package/dist/tools/tasks.js +208 -0
  362. package/dist/tools/todo-write.js +184 -0
  363. package/dist/tools/verify-plan-execution.js +295 -0
  364. package/dist/tools/web-fetch-injection-scanner.js +207 -0
  365. package/dist/tools/web-fetch.js +195 -10
  366. package/dist/tools/web-search.js +458 -0
  367. package/dist/tui/agent-progress-card.js +111 -0
  368. package/dist/tui/agent-tree.js +11 -1
  369. package/dist/tui/ask-modal.js +14 -14
  370. package/dist/tui/ask-user-question-chips.js +315 -0
  371. package/dist/tui/ask-user-question-prompt.js +203 -0
  372. package/dist/tui/compact-banner.js +81 -0
  373. package/dist/tui/conversation-pane.js +85 -11
  374. package/dist/tui/cost-table.js +111 -0
  375. package/dist/tui/device-flow.js +2 -2
  376. package/dist/tui/doctor-table.js +46 -0
  377. package/dist/tui/feedback-prompt.js +156 -0
  378. package/dist/tui/input-box.js +247 -32
  379. package/dist/tui/login-picker.js +3 -3
  380. package/dist/tui/markdown-render.js +6 -6
  381. package/dist/tui/onboarding-wizard.js +240 -0
  382. package/dist/tui/permissions-picker.js +86 -0
  383. package/dist/tui/render.js +36 -1
  384. package/dist/tui/repl-render.js +176 -25
  385. package/dist/tui/repl-splash-art.js +16 -16
  386. package/dist/tui/repl-splash-mascot.js +48 -24
  387. package/dist/tui/repl-splash.js +22 -22
  388. package/dist/tui/repl.js +125 -45
  389. package/dist/tui/slash-palette.js +6 -6
  390. package/dist/tui/splash.js +2 -2
  391. package/dist/tui/status-bar.js +109 -31
  392. package/dist/tui/status-table.js +7 -0
  393. package/dist/tui/stickers-art.js +136 -0
  394. package/dist/tui/style-table.js +28 -0
  395. package/dist/tui/theme-table.js +29 -0
  396. package/dist/tui/thinking-spinner.js +123 -0
  397. package/dist/tui/tool-stream-pane.js +53 -4
  398. package/dist/tui/update-banner.js +27 -2
  399. package/dist/tui/vim-input.js +267 -0
  400. package/dist/tui/welcome-banner.js +107 -0
  401. package/dist/tui/welcome-data.js +293 -0
  402. package/dist/tui/workspace-context.js +2 -2
  403. package/package.json +31 -16
  404. package/test/scenarios/codegen-create-file.scenario.txt +13 -0
  405. package/test/scenarios/compact-force.scenario.txt +12 -0
  406. package/test/scenarios/identity.scenario.txt +12 -0
  407. package/test/scenarios/persona-handoff.scenario.txt +12 -0
  408. package/test/scenarios/walkback.scenario.txt +12 -0
  409. package/dist/core/engine/compaction-hook.js +0 -154
@@ -0,0 +1,189 @@
1
+ const BEGIN_PATCH = '*** Begin Patch';
2
+ const END_PATCH = '*** End Patch';
3
+ const END_FILE = '*** End File';
4
+ const ADD_FILE_PREFIX = '*** Add File: ';
5
+ const UPDATE_FILE_PREFIX = '*** Update File: ';
6
+ const DELETE_FILE_PREFIX = '*** Delete File: ';
7
+ export function parseApplyPatch(envelope) {
8
+ const ops = [];
9
+ const errors = [];
10
+ const lines = envelope.replace(/\r\n/g, '\n').replace(/\r/g, '\n').split('\n');
11
+ if (envelope.trim().length === 0) {
12
+ return { ops, errors };
13
+ }
14
+ let foundBlock = false;
15
+ let index = 0;
16
+ while (index < lines.length) {
17
+ const line = lines[index];
18
+ if (line === undefined) {
19
+ index += 1;
20
+ continue;
21
+ }
22
+ if (line === BEGIN_PATCH) {
23
+ foundBlock = true;
24
+ const parsed = parseBlock(lines, index + 1);
25
+ ops.push(...parsed.ops);
26
+ if (parsed.error !== null) {
27
+ errors.push(parsed.error);
28
+ }
29
+ index = parsed.nextIndex;
30
+ continue;
31
+ }
32
+ if (line.trim().length > 0) {
33
+ errors.push(`Unexpected content outside patch block at line ${index + 1}: ${line}`);
34
+ }
35
+ index += 1;
36
+ }
37
+ if (!foundBlock) {
38
+ errors.push('No patch block found: expected *** Begin Patch');
39
+ }
40
+ return { ops, errors };
41
+ }
42
+ function parseBlock(lines, startIndex) {
43
+ const ops = [];
44
+ const blockErrors = [];
45
+ let pending = null;
46
+ let index = startIndex;
47
+ const flushPending = () => {
48
+ if (pending === null) {
49
+ return;
50
+ }
51
+ if (pending.kind === 'add') {
52
+ ops.push({
53
+ kind: 'add',
54
+ path: pending.path,
55
+ content: pending.contentLines.join('\n'),
56
+ });
57
+ }
58
+ else if (pending.kind === 'update') {
59
+ ops.push({
60
+ kind: 'update',
61
+ path: pending.path,
62
+ oldContent: pending.oldLines.join('\n'),
63
+ newContent: pending.newLines.join('\n'),
64
+ });
65
+ }
66
+ else {
67
+ ops.push({
68
+ kind: 'delete',
69
+ path: pending.path,
70
+ });
71
+ }
72
+ pending = null;
73
+ };
74
+ while (index < lines.length) {
75
+ const line = lines[index];
76
+ if (line === undefined) {
77
+ index += 1;
78
+ continue;
79
+ }
80
+ if (line === END_PATCH) {
81
+ flushPending();
82
+ return {
83
+ ops,
84
+ error: blockErrors.length > 0 ? blockErrors.join('; ') : null,
85
+ nextIndex: index + 1,
86
+ };
87
+ }
88
+ if (line === BEGIN_PATCH) {
89
+ flushPending();
90
+ blockErrors.push(`Missing ${END_PATCH} before new patch block at line ${index + 1}`);
91
+ return {
92
+ ops,
93
+ error: blockErrors.join('; '),
94
+ nextIndex: index,
95
+ };
96
+ }
97
+ if (line === END_FILE) {
98
+ flushPending();
99
+ index += 1;
100
+ continue;
101
+ }
102
+ if (line.startsWith(ADD_FILE_PREFIX)) {
103
+ flushPending();
104
+ const path = parsePath(line.slice(ADD_FILE_PREFIX.length));
105
+ if (path.length === 0) {
106
+ blockErrors.push(`Missing add file path at line ${index + 1}`);
107
+ }
108
+ pending = { kind: 'add', path, contentLines: [] };
109
+ index += 1;
110
+ continue;
111
+ }
112
+ if (line.startsWith(UPDATE_FILE_PREFIX)) {
113
+ flushPending();
114
+ const path = parsePath(line.slice(UPDATE_FILE_PREFIX.length));
115
+ if (path.length === 0) {
116
+ blockErrors.push(`Missing update file path at line ${index + 1}`);
117
+ }
118
+ pending = { kind: 'update', path, oldLines: [], newLines: [] };
119
+ index += 1;
120
+ continue;
121
+ }
122
+ if (line.startsWith(DELETE_FILE_PREFIX)) {
123
+ flushPending();
124
+ const path = parsePath(line.slice(DELETE_FILE_PREFIX.length));
125
+ if (path.length === 0) {
126
+ blockErrors.push(`Missing delete file path at line ${index + 1}`);
127
+ }
128
+ pending = { kind: 'delete', path };
129
+ index += 1;
130
+ continue;
131
+ }
132
+ if (pending === null) {
133
+ if (line.trim().length > 0) {
134
+ blockErrors.push(`Unexpected line without active file operation at line ${index + 1}: ${line}`);
135
+ }
136
+ index += 1;
137
+ continue;
138
+ }
139
+ if (pending.kind === 'add') {
140
+ if (line.startsWith('+')) {
141
+ pending.contentLines.push(line.slice(1));
142
+ }
143
+ else if (line.trim().length > 0) {
144
+ blockErrors.push(`Unexpected add line at line ${index + 1}: ${line}`);
145
+ }
146
+ index += 1;
147
+ continue;
148
+ }
149
+ if (pending.kind === 'update') {
150
+ if (line.startsWith('-')) {
151
+ pending.oldLines.push(line.slice(1));
152
+ }
153
+ else if (line.startsWith('+')) {
154
+ pending.newLines.push(line.slice(1));
155
+ }
156
+ else if (line.startsWith('@@')) {
157
+ // Context marker; accepted but not included in structured content.
158
+ }
159
+ else if (line.trim().length > 0) {
160
+ // Unchanged context line; accepted but not included in structured content.
161
+ }
162
+ index += 1;
163
+ continue;
164
+ }
165
+ if (line.trim().length > 0) {
166
+ blockErrors.push(`Unexpected delete content at line ${index + 1}: ${line}`);
167
+ }
168
+ index += 1;
169
+ }
170
+ flushPending();
171
+ blockErrors.push(`Missing ${END_PATCH}`);
172
+ return {
173
+ ops,
174
+ error: blockErrors.join('; '),
175
+ nextIndex: index,
176
+ };
177
+ }
178
+ function parsePath(rawPath) {
179
+ const path = rawPath.trim();
180
+ if (path.length >= 2) {
181
+ const first = path[0];
182
+ const last = path[path.length - 1];
183
+ if ((first === '"' && last === '"') || (first === "'" && last === "'")) {
184
+ return path.slice(1, -1);
185
+ }
186
+ }
187
+ return path;
188
+ }
189
+ //# sourceMappingURL=apply-patch-layer-e.js.map
@@ -1,15 +1,16 @@
1
1
  /**
2
- * Diff dispatch — α6.6 escalation Phase 1.
2
+ * Diff dispatch — escalation Phase 1, β1b Pl8 transactional layer
3
+ * .
3
4
  *
4
5
  * Reads a raw model response containing one or more SEARCH/REPLACE
5
6
  * envelopes, normalises them through `marker-parser`, and routes each
6
7
  * parsed edit to the correct applicator:
7
8
  *
8
- * - `layer-a` → applyLayerA
9
- * - `layer-b` → applyLayerB
10
- * - `layer-c` → applyLayerC
11
- * - `layer-d` → throws LayerDDeferredError, surfaced as a clean
12
- * dispatch failure (Layer D ships in α6.6b)
9
+ * - `layer-a` → applyLayerA
10
+ * - `layer-b` → applyLayerB
11
+ * - `layer-c` → applyLayerC
12
+ * - `layer-d` → throws LayerDDeferredError, surfaced as a clean
13
+ * dispatch failure (Layer D ships in )
13
14
  *
14
15
  * Per-edit results are aggregated into `DispatchResult[]` so callers
15
16
  * can render the full apply transcript even when some edits failed.
@@ -21,16 +22,39 @@
21
22
  * the writeFile. A crash between the two leaves a recoverable trail —
22
23
  * the operator (or `pugi resume`) sees the intent and can re-attempt.
23
24
  *
25
+ * β1b Pl8 — transactional rollback: when the caller supplies
26
+ * `transactional: { sessionId, taskId, workspaceRoot }`, the dispatcher
27
+ * wraps the multi-file edits in a session journal (see
28
+ * `core/edits/journal.ts`). On non-zero exit / budget kill / partial
29
+ * fail it runs `rollbackDispatch()`:
30
+ *
31
+ * - tracked files that EXISTED before → `git restore -- <file>`
32
+ * - newly created files (existed=false) → `fs.unlink`
33
+ * - untracked files that EXISTED before → restore from in-memory
34
+ * pre-content snapshot (sha256_before validated)
35
+ *
36
+ * Pattern mirrors `tools/apply-patch.ts::rollbackFiles` (PR). The
37
+ * journal is the durability layer that lets a process crash recover
38
+ * across PIDs; the in-memory snapshot covers the single-process case
39
+ * where the journal write itself failed.
40
+ *
24
41
  * The dispatcher is intentionally side-effect-light: no logging, no
25
42
  * stdout writes, no exit-code mutation. The CLI integration layer in
26
43
  * `cli.ts` owns operator-facing rendering; the dispatcher returns
27
44
  * structured data and lets the caller decide UX.
28
45
  */
46
+ import { spawnSync } from 'node:child_process';
47
+ import { readFileSync, rmSync, writeFileSync } from 'node:fs';
48
+ import { resolve, sep } from 'node:path';
49
+ import { commitCheckpoint, formatCheckpointMessage, initShadowRepo, ShadowGitUnavailableError, } from '../checkpoints/shadow-git.js';
50
+ import { detectEditFormat } from './format-detector.js';
29
51
  import { LayerDDeferredError, applyLayerD } from './layer-d-ast.js';
30
52
  import { applyLayerA } from './layer-a-apply.js';
53
+ import { applyLayerAFuzzy } from './layer-a-fuzzy-apply.js';
31
54
  import { applyLayerB } from './layer-b-apply.js';
32
55
  import { applyLayerC } from './layer-c-apply.js';
33
56
  import { MarkerParseError, parseMarkers, } from './marker-parser.js';
57
+ import { appendEntry, snapshotForDispatch, } from './journal.js';
34
58
  /**
35
59
  * Parse `raw` into edits and apply each in order. Aggregate results,
36
60
  * preserving order. Never throws — parse failures surface as a single
@@ -39,6 +63,15 @@ import { MarkerParseError, parseMarkers, } from './marker-parser.js';
39
63
  */
40
64
  export async function dispatchEdit(raw, opts) {
41
65
  const family = resolveFamily(opts.modelTag);
66
+ // PUGI-79 — resolve the preferred-layer chain ONCE per dispatch.
67
+ // Caller supplied wins; otherwise the detector derives from modelTag.
68
+ // The chain is informational here (the parser still routes per
69
+ // envelope); a future fuzzy-ladder change can read it to influence
70
+ // retry ordering without revisiting the dispatcher's escalation
71
+ // contract.
72
+ const preferredChain = resolvePreferredChain(opts);
73
+ if (preferredChain)
74
+ opts.onPreferredLayer?.(preferredChain);
42
75
  let parsed;
43
76
  try {
44
77
  parsed = parseMarkers(raw, family);
@@ -52,6 +85,7 @@ export async function dispatchEdit(raw, opts) {
52
85
  bytesWritten: 0,
53
86
  reason: 'marker_parse_error',
54
87
  detail: `${error.message}${error.atLine ? ` (line ${error.atLine})` : ''} — modelHint=${error.modelHint}`,
88
+ ...(preferredChain ? { expectedLayer: preferredChain.primary } : {}),
55
89
  };
56
90
  opts.onResult?.(result);
57
91
  return [result];
@@ -65,16 +99,286 @@ export async function dispatchEdit(raw, opts) {
65
99
  // render "no edits proposed".
66
100
  return [];
67
101
  }
102
+ // β1b Pl8 — transactional path. When enabled, snapshot every target
103
+ // file BEFORE the first applicator runs so we can roll back if a
104
+ // later edit fails. Snapshot also drives the journal entry so a
105
+ // post-crash recovery can replay rollback in a fresh process.
106
+ //
107
+ // The journal write itself is best-effort: a disk-full / EACCES
108
+ // failure must NOT block the dispatch. The in-memory snapshot
109
+ // still carries every pre-existing file's content, so single-
110
+ // process rollback degrades cleanly. The operator sees the
111
+ // journal-failed warning via the session events mirror (caller's
112
+ // responsibility to emit).
113
+ let snapshot = null;
114
+ let preContent = null;
115
+ if (opts.transactional && !opts.dryRun) {
116
+ const targets = Array.from(new Set(parsed.map((e) => editFile(e))));
117
+ snapshot = snapshotForDispatch(opts.transactional.workspaceRoot, targets);
118
+ preContent = new Map();
119
+ for (const e of snapshot) {
120
+ if (!e.existed)
121
+ continue;
122
+ const abs = resolve(opts.transactional.workspaceRoot, e.path);
123
+ try {
124
+ preContent.set(e.path, readFileSync(abs));
125
+ }
126
+ catch {
127
+ /* file vanished between snapshot + read — treat as not-snapshotted */
128
+ }
129
+ }
130
+ appendEntry(opts.transactional.workspaceRoot, opts.transactional.sessionId, {
131
+ ts: Date.now(),
132
+ taskId: opts.transactional.taskId,
133
+ files: snapshot,
134
+ });
135
+ }
136
+ // Per-task shadow git checkpoint hook (backlog). Init the
137
+ // shadow once per dispatch, then commit after every successful
138
+ // edit. The init is best-effort — a missing `git` binary or a
139
+ // chmod failure surfaces through `onCheckpointError` but does not
140
+ // disable the dispatch.
141
+ let checkpointStep = opts.checkpoint?.stepStart && Number.isFinite(opts.checkpoint.stepStart)
142
+ ? Math.max(1, Math.floor(opts.checkpoint.stepStart))
143
+ : 1;
144
+ let checkpointEnabled = false;
145
+ if (opts.checkpoint && !opts.dryRun) {
146
+ try {
147
+ initShadowRepo(opts.cwd, opts.checkpoint.taskId);
148
+ checkpointEnabled = true;
149
+ }
150
+ catch (error) {
151
+ const err = error instanceof Error ? error : new Error(String(error));
152
+ opts.checkpoint.onCheckpointError?.(err, '');
153
+ }
154
+ }
155
+ const expectedLayer = preferredChain?.primary;
68
156
  const out = [];
157
+ let crashError = null;
69
158
  for (const edit of parsed) {
70
159
  const intent = makeIntent(edit);
71
160
  opts.onIntent?.(intent);
72
- const result = await applyOne(edit, opts);
161
+ let result;
162
+ try {
163
+ result = await applyOne(edit, opts);
164
+ }
165
+ catch (error) {
166
+ // applyOne does not throw today (errors are returned as
167
+ // `ok: false`), but a future applicator that does throw —
168
+ // or a budget-kill that arrives mid-write — needs deterministic
169
+ // rollback. Catch + record + break so the rollback below runs.
170
+ crashError = error;
171
+ const msg = error instanceof Error ? error.message : String(error);
172
+ result = {
173
+ layer: 'layer-a',
174
+ file: editFile(edit),
175
+ ok: false,
176
+ bytesWritten: 0,
177
+ reason: 'apply_error',
178
+ detail: `dispatch threw: ${msg}`,
179
+ };
180
+ }
181
+ if (expectedLayer !== undefined) {
182
+ // PUGI-79: stamp the expected layer on every result so
183
+ // observability surfaces can compute drift (expected vs actual)
184
+ // per family. The actual `layer` field is unchanged — the
185
+ // hint is informational, not a routing override.
186
+ result = { ...result, expectedLayer };
187
+ }
73
188
  out.push(result);
74
189
  opts.onResult?.(result);
190
+ if (result.ok && checkpointEnabled && opts.checkpoint && !opts.dryRun) {
191
+ // Best-effort shadow commit. Failures here MUST NOT block the
192
+ // dispatch — the operator already has the edit applied; the
193
+ // checkpoint is a customer-trust safety net, not a hard gate.
194
+ try {
195
+ const absPath = result.absPath ?? resolve(opts.cwd, result.file);
196
+ const message = formatCheckpointMessage({
197
+ taskId: opts.checkpoint.taskId,
198
+ step: checkpointStep,
199
+ toolName: result.layer,
200
+ cwd: opts.cwd,
201
+ absPath,
202
+ });
203
+ commitCheckpoint(opts.cwd, opts.checkpoint.taskId, message);
204
+ }
205
+ catch (error) {
206
+ const err = error instanceof Error ? error : new Error(String(error));
207
+ // Disable the shadow hook for the rest of this dispatch if
208
+ // the git binary is unavailable — repeated probes burn CPU
209
+ // and add no signal beyond the first failure.
210
+ if (err instanceof ShadowGitUnavailableError) {
211
+ checkpointEnabled = false;
212
+ }
213
+ opts.checkpoint.onCheckpointError?.(err, result.file);
214
+ }
215
+ checkpointStep += 1;
216
+ }
217
+ if (!result.ok && opts.transactional && snapshot && preContent) {
218
+ // Rollback every snapshotted file then break out — partial
219
+ // success is unacceptable in transactional mode.
220
+ const rollback = rollbackDispatch(opts.transactional.workspaceRoot, snapshot, preContent);
221
+ if (!rollback.ok) {
222
+ // Surface the rollback failure as an additional synthetic
223
+ // result so the caller can render the operator-facing
224
+ // message without losing the original failure context.
225
+ const failure = {
226
+ layer: 'layer-a',
227
+ file: '',
228
+ ok: false,
229
+ bytesWritten: 0,
230
+ reason: 'rollback_failed',
231
+ detail: rollback.detail,
232
+ };
233
+ out.push(failure);
234
+ opts.onResult?.(failure);
235
+ }
236
+ if (crashError) {
237
+ // Re-throw post-rollback so the caller learns the dispatch
238
+ // crashed (vs returned ok: false). Rollback already completed
239
+ // so the workspace is consistent.
240
+ throw crashError;
241
+ }
242
+ break;
243
+ }
75
244
  }
76
245
  return out;
77
246
  }
247
+ /**
248
+ * Workspace-relative path for a parsed edit, regardless of layer.
249
+ * Hoisted because the snapshot + intent both need the same answer.
250
+ */
251
+ function editFile(edit) {
252
+ switch (edit.kind) {
253
+ case 'layer-a':
254
+ case 'layer-b':
255
+ case 'layer-c':
256
+ case 'layer-d':
257
+ return edit.edit.file;
258
+ }
259
+ }
260
+ /**
261
+ * Roll the workspace back to the pre-dispatch state captured in
262
+ * `snapshot`. Mirrors `tools/apply-patch.ts::rollbackFiles`:
263
+ *
264
+ * - existed-before + git-tracked → `git restore -- <file>` (cheap
265
+ * + atomic against the index).
266
+ * - existed-before + untracked → restore from the in-memory
267
+ * preContent buffer (sha256_before validates the snapshot still
268
+ * matches what we hold; if not we punt with `partial_rollback`).
269
+ * - newly created → fs.unlink (force).
270
+ *
271
+ * Best-effort: every per-file failure is collected into the detail
272
+ * string so the operator can manually fix the residual state. The
273
+ * dispatcher does not abort on the first error so an unrelated
274
+ * permission glitch on one file doesn't strand the others.
275
+ *
276
+ * Exported for the spec suite + a future operator command
277
+ * (`pugi resume --rollback <taskId>` ships in β2).
278
+ */
279
+ export function rollbackDispatch(workspaceRoot, snapshot, preContent) {
280
+ if (snapshot.length === 0)
281
+ return { ok: true };
282
+ // Filter to workspace-internal paths only. A snapshot entry that
283
+ // escaped the workspace would already have aborted upstream; the
284
+ // filter is belt + braces against a future bug.
285
+ const safe = snapshot.filter((e) => {
286
+ const abs = resolve(workspaceRoot, e.path);
287
+ return abs === workspaceRoot || abs.startsWith(workspaceRoot + sep);
288
+ });
289
+ const failures = [];
290
+ const tracked = listTrackedFiles(workspaceRoot, safe.map((e) => e.path));
291
+ for (const entry of safe) {
292
+ const abs = resolve(workspaceRoot, entry.path);
293
+ if (!entry.existed) {
294
+ try {
295
+ // β1b r1: `recursive: true` so rollback handles the case where
296
+ // the dispatcher created an intermediate directory (e.g. a new
297
+ // `src/feature/` tree). Without it the unlink fails on a dir
298
+ // and the journal-replay leaves an orphan workspace path.
299
+ rmSync(abs, { force: true, recursive: true });
300
+ }
301
+ catch (error) {
302
+ failures.push(`${entry.path}: unlink failed: ${error instanceof Error ? error.message : String(error)}`);
303
+ }
304
+ continue;
305
+ }
306
+ if (tracked.has(entry.path)) {
307
+ const result = spawnSync('git', ['restore', '--', entry.path], {
308
+ cwd: workspaceRoot,
309
+ encoding: 'utf8',
310
+ });
311
+ if (result.status !== 0) {
312
+ failures.push(`${entry.path}: git restore failed: ${(result.stderr ?? '').trim() || 'non-zero exit'}`);
313
+ }
314
+ continue;
315
+ }
316
+ // Untracked-but-existed: write back from memory.
317
+ const buf = preContent.get(entry.path);
318
+ if (!buf) {
319
+ failures.push(`${entry.path}: pre-content snapshot missing (partial rollback)`);
320
+ continue;
321
+ }
322
+ try {
323
+ writeFileSync(abs, buf);
324
+ }
325
+ catch (error) {
326
+ failures.push(`${entry.path}: rewrite failed: ${error instanceof Error ? error.message : String(error)}`);
327
+ }
328
+ }
329
+ if (failures.length === 0)
330
+ return { ok: true };
331
+ return { ok: false, detail: failures.join('; ') };
332
+ }
333
+ /**
334
+ * Ask git which of `paths` are tracked. A single `git ls-files
335
+ * --error-unmatch` call would fail-fast on the first untracked path,
336
+ * so we use `git ls-files -- <paths...>` which lists only tracked
337
+ * matches. Returns a Set of workspace-relative paths.
338
+ *
339
+ * Pure-stdlib fallback when git is unavailable: returns an empty
340
+ * Set — every "existed" entry then routes to the untracked-restore
341
+ * path via the in-memory preContent map. Slower per file but still
342
+ * correct.
343
+ */
344
+ function listTrackedFiles(cwd, paths) {
345
+ if (paths.length === 0)
346
+ return new Set();
347
+ const result = spawnSync('git', ['ls-files', '--', ...paths], {
348
+ cwd,
349
+ encoding: 'utf8',
350
+ });
351
+ if (result.status !== 0)
352
+ return new Set();
353
+ const out = new Set();
354
+ for (const line of (result.stdout ?? '').split('\n')) {
355
+ const trimmed = line.trim();
356
+ if (trimmed.length > 0)
357
+ out.add(trimmed);
358
+ }
359
+ return out;
360
+ }
361
+ /**
362
+ * PUGI-79 helper — resolve the preferred-layer chain for this
363
+ * dispatch. Caller-supplied `preferredLayer` wins; otherwise the
364
+ * detector derives from `modelTag`. Returns null when neither path
365
+ * yields a chain (no caller hint AND no model tag) — the dispatcher
366
+ * skips the observability hook in that case so we do not emit a
367
+ * misleading wildcard reading.
368
+ */
369
+ function resolvePreferredChain(opts) {
370
+ if (opts.preferredLayer) {
371
+ // Defensive clone so caller mutations after dispatchEdit returns
372
+ // cannot poison subsequent calls that share the chain reference.
373
+ return {
374
+ primary: opts.preferredLayer.primary,
375
+ fallback: [...opts.preferredLayer.fallback],
376
+ };
377
+ }
378
+ if (opts.modelTag)
379
+ return detectEditFormat(opts.modelTag);
380
+ return null;
381
+ }
78
382
  /**
79
383
  * Public helper exposed for the marker parser tests + CLI surface that
80
384
  * may want to know the resolved family without re-running the auto
@@ -125,6 +429,28 @@ async function applyOne(edit, opts) {
125
429
  switch (edit.kind) {
126
430
  case 'layer-a': {
127
431
  const r = await applyLayerA(edit.edit, applyOpts);
432
+ // Backlog — Layer A.5 escalation. The fuzzy ladder is the
433
+ // ONLY retry path: every other Layer A failure mode (ambiguous,
434
+ // file_missing, identical_replacement, security gate denial)
435
+ // bubbles up untouched because the ladder can't fix them.
436
+ // ambiguous in particular MUST stay loud — a fuzzy retry on top
437
+ // of an ambiguous strict match would just pick one of the
438
+ // strict candidates silently, defeating the disambiguation
439
+ // contract.
440
+ if (!r.ok && r.reason === 'no_match' && opts.fuzzy === true) {
441
+ const fuzzy = await applyLayerAFuzzy(edit.edit, {
442
+ cwd: opts.cwd,
443
+ dryRun: opts.dryRun,
444
+ minRatio: opts.fuzzyMinRatio,
445
+ lengthFlex: opts.fuzzyLengthFlex,
446
+ });
447
+ const base = toResult('layer-a-fuzzy', edit.edit.file, fuzzy);
448
+ return {
449
+ ...base,
450
+ ...(fuzzy.fuzzyTier ? { fuzzyTier: fuzzy.fuzzyTier } : {}),
451
+ ...(fuzzy.fuzzyScore !== undefined ? { fuzzyScore: fuzzy.fuzzyScore } : {}),
452
+ };
453
+ }
128
454
  return toResult('layer-a', edit.edit.file, r);
129
455
  }
130
456
  case 'layer-b': {