@wahack/pi-coding-agent 15.11.0

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 (1152) hide show
  1. package/CHANGELOG.md +10031 -0
  2. package/README.md +36 -0
  3. package/examples/README.md +21 -0
  4. package/examples/custom-tools/README.md +104 -0
  5. package/examples/custom-tools/hello/index.ts +20 -0
  6. package/examples/extensions/README.md +142 -0
  7. package/examples/extensions/api-demo.ts +79 -0
  8. package/examples/extensions/chalk-logger.ts +25 -0
  9. package/examples/extensions/hello.ts +31 -0
  10. package/examples/extensions/pirate.ts +43 -0
  11. package/examples/extensions/plan-mode.ts +549 -0
  12. package/examples/extensions/reload-runtime.ts +38 -0
  13. package/examples/extensions/thinking-note.ts +13 -0
  14. package/examples/extensions/tools.ts +145 -0
  15. package/examples/extensions/with-deps/index.ts +36 -0
  16. package/examples/extensions/with-deps/package-lock.json +31 -0
  17. package/examples/extensions/with-deps/package.json +17 -0
  18. package/examples/hooks/README.md +56 -0
  19. package/examples/hooks/auto-commit-on-exit.ts +48 -0
  20. package/examples/hooks/confirm-destructive.ts +58 -0
  21. package/examples/hooks/custom-compaction.ts +115 -0
  22. package/examples/hooks/dirty-repo-guard.ts +51 -0
  23. package/examples/hooks/file-trigger.ts +40 -0
  24. package/examples/hooks/git-checkpoint.ts +52 -0
  25. package/examples/hooks/handoff.ts +149 -0
  26. package/examples/hooks/permission-gate.ts +33 -0
  27. package/examples/hooks/protected-paths.ts +29 -0
  28. package/examples/hooks/qna.ts +118 -0
  29. package/examples/hooks/status-line.ts +39 -0
  30. package/examples/sdk/01-minimal.ts +21 -0
  31. package/examples/sdk/02-custom-model.ts +49 -0
  32. package/examples/sdk/03-custom-prompt.ts +46 -0
  33. package/examples/sdk/04-skills.ts +43 -0
  34. package/examples/sdk/06-extensions.ts +82 -0
  35. package/examples/sdk/06-hooks.ts +61 -0
  36. package/examples/sdk/07-context-files.ts +35 -0
  37. package/examples/sdk/08-prompt-templates.ts +41 -0
  38. package/examples/sdk/08-slash-commands.ts +46 -0
  39. package/examples/sdk/09-api-keys-and-oauth.ts +54 -0
  40. package/examples/sdk/11-sessions.ts +47 -0
  41. package/examples/sdk/12-redis-sessions.ts +54 -0
  42. package/examples/sdk/13-sql-sessions.ts +61 -0
  43. package/examples/sdk/README.md +172 -0
  44. package/package.json +554 -0
  45. package/scripts/build-binary.ts +100 -0
  46. package/scripts/bundle-dist.ts +90 -0
  47. package/scripts/format-prompts.ts +68 -0
  48. package/scripts/generate-docs-index.ts +40 -0
  49. package/scripts/generate-template.ts +33 -0
  50. package/scripts/omp +42 -0
  51. package/scripts/omp.ts +19 -0
  52. package/src/async/index.ts +1 -0
  53. package/src/async/job-manager.ts +625 -0
  54. package/src/auto-thinking/classifier.ts +185 -0
  55. package/src/autoresearch/command-resume.md +14 -0
  56. package/src/autoresearch/dashboard.ts +436 -0
  57. package/src/autoresearch/git.ts +319 -0
  58. package/src/autoresearch/helpers.ts +218 -0
  59. package/src/autoresearch/index.ts +536 -0
  60. package/src/autoresearch/prompt-setup.md +43 -0
  61. package/src/autoresearch/prompt.md +103 -0
  62. package/src/autoresearch/resume-message.md +10 -0
  63. package/src/autoresearch/state.ts +273 -0
  64. package/src/autoresearch/storage.ts +699 -0
  65. package/src/autoresearch/tools/init-experiment.ts +272 -0
  66. package/src/autoresearch/tools/log-experiment.ts +524 -0
  67. package/src/autoresearch/tools/run-experiment.ts +407 -0
  68. package/src/autoresearch/tools/update-notes.ts +109 -0
  69. package/src/autoresearch/types.ts +168 -0
  70. package/src/bun-imports.d.ts +28 -0
  71. package/src/capability/context-file.ts +44 -0
  72. package/src/capability/extension-module.ts +34 -0
  73. package/src/capability/extension.ts +47 -0
  74. package/src/capability/fs.ts +117 -0
  75. package/src/capability/hook.ts +40 -0
  76. package/src/capability/index.ts +436 -0
  77. package/src/capability/instruction.ts +37 -0
  78. package/src/capability/mcp.ts +74 -0
  79. package/src/capability/prompt.ts +35 -0
  80. package/src/capability/rule-buckets.ts +66 -0
  81. package/src/capability/rule.ts +261 -0
  82. package/src/capability/settings.ts +34 -0
  83. package/src/capability/skill.ts +63 -0
  84. package/src/capability/slash-command.ts +40 -0
  85. package/src/capability/ssh.ts +41 -0
  86. package/src/capability/system-prompt.ts +34 -0
  87. package/src/capability/tool.ts +38 -0
  88. package/src/capability/types.ts +168 -0
  89. package/src/cli/agents-cli.ts +138 -0
  90. package/src/cli/args.ts +340 -0
  91. package/src/cli/auth-broker-cli.ts +895 -0
  92. package/src/cli/auth-gateway-cli.ts +611 -0
  93. package/src/cli/classify-install-target.ts +76 -0
  94. package/src/cli/claude-trace-cli.ts +795 -0
  95. package/src/cli/commands/init-xdg.ts +27 -0
  96. package/src/cli/completion-gen.ts +550 -0
  97. package/src/cli/config-cli.ts +418 -0
  98. package/src/cli/dry-balance-cli.ts +856 -0
  99. package/src/cli/extension-flags.ts +48 -0
  100. package/src/cli/file-processor.ts +133 -0
  101. package/src/cli/gallery-cli.ts +230 -0
  102. package/src/cli/gallery-fixtures/agentic.ts +407 -0
  103. package/src/cli/gallery-fixtures/codeintel.ts +187 -0
  104. package/src/cli/gallery-fixtures/edit.ts +194 -0
  105. package/src/cli/gallery-fixtures/fs.ts +220 -0
  106. package/src/cli/gallery-fixtures/index.ts +40 -0
  107. package/src/cli/gallery-fixtures/interaction.ts +49 -0
  108. package/src/cli/gallery-fixtures/memory.ts +81 -0
  109. package/src/cli/gallery-fixtures/misc.ts +250 -0
  110. package/src/cli/gallery-fixtures/search.ts +213 -0
  111. package/src/cli/gallery-fixtures/shell.ts +167 -0
  112. package/src/cli/gallery-fixtures/types.ts +57 -0
  113. package/src/cli/gallery-fixtures/web.ts +158 -0
  114. package/src/cli/gallery-screenshot.ts +279 -0
  115. package/src/cli/grep-cli.ts +160 -0
  116. package/src/cli/grievances-cli.ts +256 -0
  117. package/src/cli/initial-message.ts +58 -0
  118. package/src/cli/list-models.ts +194 -0
  119. package/src/cli/plugin-cli.ts +996 -0
  120. package/src/cli/read-cli.ts +57 -0
  121. package/src/cli/session-picker.ts +79 -0
  122. package/src/cli/setup-cli.ts +231 -0
  123. package/src/cli/shell-cli.ts +176 -0
  124. package/src/cli/ssh-cli.ts +179 -0
  125. package/src/cli/startup-cwd.ts +68 -0
  126. package/src/cli/stats-cli.ts +238 -0
  127. package/src/cli/tiny-models-cli.ts +127 -0
  128. package/src/cli/update-cli.ts +611 -0
  129. package/src/cli/usage-cli.ts +603 -0
  130. package/src/cli/web-search-cli.ts +132 -0
  131. package/src/cli/worktree-cli.ts +291 -0
  132. package/src/cli-commands.ts +79 -0
  133. package/src/cli.ts +200 -0
  134. package/src/commands/acp.ts +24 -0
  135. package/src/commands/agents.ts +57 -0
  136. package/src/commands/auth-broker.ts +99 -0
  137. package/src/commands/auth-gateway.ts +69 -0
  138. package/src/commands/commit.ts +46 -0
  139. package/src/commands/complete.ts +66 -0
  140. package/src/commands/completions.ts +60 -0
  141. package/src/commands/config.ts +51 -0
  142. package/src/commands/dry-balance.ts +43 -0
  143. package/src/commands/gallery.ts +52 -0
  144. package/src/commands/grep.ts +48 -0
  145. package/src/commands/grievances.ts +51 -0
  146. package/src/commands/install.ts +107 -0
  147. package/src/commands/launch.ts +169 -0
  148. package/src/commands/plugin.ts +78 -0
  149. package/src/commands/read.ts +38 -0
  150. package/src/commands/setup.ts +67 -0
  151. package/src/commands/shell.ts +29 -0
  152. package/src/commands/ssh.ts +60 -0
  153. package/src/commands/stats.ts +29 -0
  154. package/src/commands/tiny-models.ts +36 -0
  155. package/src/commands/update.ts +21 -0
  156. package/src/commands/usage.ts +35 -0
  157. package/src/commands/web-search.ts +42 -0
  158. package/src/commands/worktree.ts +56 -0
  159. package/src/commit/agentic/agent.ts +317 -0
  160. package/src/commit/agentic/fallback.ts +96 -0
  161. package/src/commit/agentic/index.ts +355 -0
  162. package/src/commit/agentic/prompts/analyze-file.md +22 -0
  163. package/src/commit/agentic/prompts/session-user.md +25 -0
  164. package/src/commit/agentic/prompts/split-confirm.md +1 -0
  165. package/src/commit/agentic/prompts/system.md +38 -0
  166. package/src/commit/agentic/state.ts +60 -0
  167. package/src/commit/agentic/tools/analyze-file.ts +146 -0
  168. package/src/commit/agentic/tools/git-file-diff.ts +191 -0
  169. package/src/commit/agentic/tools/git-hunk.ts +50 -0
  170. package/src/commit/agentic/tools/git-overview.ts +81 -0
  171. package/src/commit/agentic/tools/index.ts +54 -0
  172. package/src/commit/agentic/tools/propose-changelog.ts +144 -0
  173. package/src/commit/agentic/tools/propose-commit.ts +109 -0
  174. package/src/commit/agentic/tools/recent-commits.ts +81 -0
  175. package/src/commit/agentic/tools/schemas.ts +23 -0
  176. package/src/commit/agentic/tools/split-commit.ts +245 -0
  177. package/src/commit/agentic/topo-sort.ts +44 -0
  178. package/src/commit/agentic/trivial.ts +51 -0
  179. package/src/commit/agentic/validation.ts +183 -0
  180. package/src/commit/analysis/conventional.ts +64 -0
  181. package/src/commit/analysis/index.ts +4 -0
  182. package/src/commit/analysis/scope.ts +242 -0
  183. package/src/commit/analysis/summary.ts +105 -0
  184. package/src/commit/analysis/validation.ts +66 -0
  185. package/src/commit/changelog/detect.ts +40 -0
  186. package/src/commit/changelog/generate.ts +97 -0
  187. package/src/commit/changelog/index.ts +234 -0
  188. package/src/commit/changelog/parse.ts +44 -0
  189. package/src/commit/cli.ts +85 -0
  190. package/src/commit/git/diff.ts +148 -0
  191. package/src/commit/index.ts +5 -0
  192. package/src/commit/map-reduce/index.ts +69 -0
  193. package/src/commit/map-reduce/map-phase.ts +193 -0
  194. package/src/commit/map-reduce/reduce-phase.ts +49 -0
  195. package/src/commit/map-reduce/utils.ts +9 -0
  196. package/src/commit/message.ts +11 -0
  197. package/src/commit/model-selection.ts +92 -0
  198. package/src/commit/pipeline.ts +243 -0
  199. package/src/commit/prompts/analysis-system.md +148 -0
  200. package/src/commit/prompts/analysis-user.md +38 -0
  201. package/src/commit/prompts/changelog-system.md +50 -0
  202. package/src/commit/prompts/changelog-user.md +18 -0
  203. package/src/commit/prompts/file-observer-system.md +24 -0
  204. package/src/commit/prompts/file-observer-user.md +8 -0
  205. package/src/commit/prompts/reduce-system.md +50 -0
  206. package/src/commit/prompts/reduce-user.md +17 -0
  207. package/src/commit/prompts/summary-retry.md +3 -0
  208. package/src/commit/prompts/summary-system.md +38 -0
  209. package/src/commit/prompts/summary-user.md +13 -0
  210. package/src/commit/prompts/types-description.md +2 -0
  211. package/src/commit/shared-llm.ts +77 -0
  212. package/src/commit/types.ts +118 -0
  213. package/src/commit/utils/exclusions.ts +42 -0
  214. package/src/commit/utils.ts +58 -0
  215. package/src/config/api-key-resolver.ts +60 -0
  216. package/src/config/append-only-context-mode.ts +31 -0
  217. package/src/config/config-file.ts +317 -0
  218. package/src/config/file-lock.ts +164 -0
  219. package/src/config/keybindings.ts +628 -0
  220. package/src/config/mcp-schema.json +230 -0
  221. package/src/config/model-discovery.ts +554 -0
  222. package/src/config/model-registry.ts +2090 -0
  223. package/src/config/model-resolver.ts +1502 -0
  224. package/src/config/model-roles.ts +74 -0
  225. package/src/config/models-config-schema.ts +226 -0
  226. package/src/config/models-config.ts +129 -0
  227. package/src/config/prompt-templates.ts +185 -0
  228. package/src/config/resolve-config-value.ts +94 -0
  229. package/src/config/settings-schema.ts +3530 -0
  230. package/src/config/settings.ts +1178 -0
  231. package/src/config.ts +242 -0
  232. package/src/cursor.ts +340 -0
  233. package/src/dap/client.ts +760 -0
  234. package/src/dap/config.ts +189 -0
  235. package/src/dap/defaults.json +212 -0
  236. package/src/dap/index.ts +4 -0
  237. package/src/dap/session.ts +1441 -0
  238. package/src/dap/types.ts +610 -0
  239. package/src/debug/index.ts +515 -0
  240. package/src/debug/log-formatting.ts +58 -0
  241. package/src/debug/log-viewer.ts +908 -0
  242. package/src/debug/profiler.ts +162 -0
  243. package/src/debug/protocol-probe.ts +267 -0
  244. package/src/debug/raw-sse-buffer.ts +273 -0
  245. package/src/debug/raw-sse.ts +292 -0
  246. package/src/debug/report-bundle.ts +374 -0
  247. package/src/debug/system-info.ts +111 -0
  248. package/src/debug/terminal-info.ts +124 -0
  249. package/src/discovery/agents-md.ts +67 -0
  250. package/src/discovery/agents.ts +230 -0
  251. package/src/discovery/at-imports.ts +273 -0
  252. package/src/discovery/builtin-defaults.ts +39 -0
  253. package/src/discovery/builtin-rules/index.ts +54 -0
  254. package/src/discovery/builtin-rules/rs-box-leak.md +48 -0
  255. package/src/discovery/builtin-rules/rs-future-prelude.md +23 -0
  256. package/src/discovery/builtin-rules/rs-lazylock.md +51 -0
  257. package/src/discovery/builtin-rules/rs-match-ergonomics.md +67 -0
  258. package/src/discovery/builtin-rules/rs-parking-lot.md +44 -0
  259. package/src/discovery/builtin-rules/rs-result-type.md +19 -0
  260. package/src/discovery/builtin-rules/ts-bare-catch.md +38 -0
  261. package/src/discovery/builtin-rules/ts-import-type.md +42 -0
  262. package/src/discovery/builtin-rules/ts-no-any.md +56 -0
  263. package/src/discovery/builtin-rules/ts-no-deprecated-leftovers.md +44 -0
  264. package/src/discovery/builtin-rules/ts-no-dynamic-import.md +39 -0
  265. package/src/discovery/builtin-rules/ts-no-return-type.md +45 -0
  266. package/src/discovery/builtin-rules/ts-no-test-timers.md +55 -0
  267. package/src/discovery/builtin-rules/ts-no-tiny-functions.md +51 -0
  268. package/src/discovery/builtin-rules/ts-promise-with-resolvers.md +65 -0
  269. package/src/discovery/builtin-rules/ts-redundant-clear-guard.md +75 -0
  270. package/src/discovery/builtin-rules/ts-set-map.md +28 -0
  271. package/src/discovery/builtin.ts +906 -0
  272. package/src/discovery/claude-plugins.ts +386 -0
  273. package/src/discovery/claude.ts +584 -0
  274. package/src/discovery/cline.ts +83 -0
  275. package/src/discovery/codex.ts +522 -0
  276. package/src/discovery/cursor.ts +220 -0
  277. package/src/discovery/gemini.ts +383 -0
  278. package/src/discovery/github.ts +154 -0
  279. package/src/discovery/helpers.ts +1016 -0
  280. package/src/discovery/index.ts +81 -0
  281. package/src/discovery/mcp-json.ts +171 -0
  282. package/src/discovery/omp-extension-roots.ts +190 -0
  283. package/src/discovery/omp-plugins.ts +383 -0
  284. package/src/discovery/opencode.ts +398 -0
  285. package/src/discovery/plugin-dir-roots.ts +28 -0
  286. package/src/discovery/ssh.ts +153 -0
  287. package/src/discovery/substitute-plugin-root.ts +29 -0
  288. package/src/discovery/vscode.ts +105 -0
  289. package/src/discovery/windsurf.ts +147 -0
  290. package/src/edit/apply-patch/index.ts +87 -0
  291. package/src/edit/apply-patch/parser.ts +174 -0
  292. package/src/edit/diff.ts +999 -0
  293. package/src/edit/file-snapshot-store.ts +91 -0
  294. package/src/edit/hashline/block-resolver.ts +33 -0
  295. package/src/edit/hashline/diff.ts +290 -0
  296. package/src/edit/hashline/execute.ts +242 -0
  297. package/src/edit/hashline/filesystem.ts +130 -0
  298. package/src/edit/hashline/index.ts +5 -0
  299. package/src/edit/hashline/noop-loop-guard.ts +99 -0
  300. package/src/edit/hashline/params.ts +18 -0
  301. package/src/edit/index.ts +571 -0
  302. package/src/edit/modes/apply-patch.lark +19 -0
  303. package/src/edit/modes/apply-patch.ts +53 -0
  304. package/src/edit/modes/patch.ts +1891 -0
  305. package/src/edit/modes/replace.ts +1137 -0
  306. package/src/edit/normalize.ts +345 -0
  307. package/src/edit/notebook.ts +242 -0
  308. package/src/edit/read-file.ts +25 -0
  309. package/src/edit/renderer.ts +769 -0
  310. package/src/edit/streaming.ts +517 -0
  311. package/src/eval/__tests__/agent-bridge.test.ts +708 -0
  312. package/src/eval/__tests__/bridge-timeout.test.ts +64 -0
  313. package/src/eval/__tests__/budget-bridge.test.ts +69 -0
  314. package/src/eval/__tests__/completion-bridge.test.ts +412 -0
  315. package/src/eval/__tests__/helpers-local-roots.test.ts +58 -0
  316. package/src/eval/__tests__/idle-timeout.test.ts +80 -0
  317. package/src/eval/__tests__/js-context-manager.test.ts +241 -0
  318. package/src/eval/__tests__/kernel-spawn.test.ts +103 -0
  319. package/src/eval/agent-bridge.ts +319 -0
  320. package/src/eval/backend.ts +71 -0
  321. package/src/eval/bridge-timeout.ts +44 -0
  322. package/src/eval/budget-bridge.ts +48 -0
  323. package/src/eval/completion-bridge.ts +207 -0
  324. package/src/eval/concurrency-bridge.ts +34 -0
  325. package/src/eval/idle-timeout.ts +91 -0
  326. package/src/eval/index.ts +4 -0
  327. package/src/eval/js/context-manager.ts +502 -0
  328. package/src/eval/js/executor.ts +173 -0
  329. package/src/eval/js/index.ts +51 -0
  330. package/src/eval/js/shared/helpers.ts +283 -0
  331. package/src/eval/js/shared/indirect-eval.ts +30 -0
  332. package/src/eval/js/shared/local-module-loader.ts +342 -0
  333. package/src/eval/js/shared/prelude.ts +2 -0
  334. package/src/eval/js/shared/prelude.txt +246 -0
  335. package/src/eval/js/shared/rewrite-imports.ts +532 -0
  336. package/src/eval/js/shared/runtime.ts +352 -0
  337. package/src/eval/js/shared/types.ts +18 -0
  338. package/src/eval/js/tool-bridge.ts +162 -0
  339. package/src/eval/js/worker-core.ts +132 -0
  340. package/src/eval/js/worker-entry.ts +30 -0
  341. package/src/eval/js/worker-protocol.ts +47 -0
  342. package/src/eval/py/__tests__/prelude.test.ts +19 -0
  343. package/src/eval/py/display.ts +71 -0
  344. package/src/eval/py/executor.ts +742 -0
  345. package/src/eval/py/index.ts +68 -0
  346. package/src/eval/py/kernel.ts +748 -0
  347. package/src/eval/py/prelude.py +658 -0
  348. package/src/eval/py/prelude.ts +3 -0
  349. package/src/eval/py/runner.py +1133 -0
  350. package/src/eval/py/runtime.ts +276 -0
  351. package/src/eval/py/spawn-options.ts +126 -0
  352. package/src/eval/py/tool-bridge.ts +182 -0
  353. package/src/eval/session-id.ts +8 -0
  354. package/src/eval/types.ts +48 -0
  355. package/src/exa/index.ts +2 -0
  356. package/src/exa/mcp-client.ts +370 -0
  357. package/src/exa/types.ts +69 -0
  358. package/src/exec/bash-executor.ts +419 -0
  359. package/src/exec/exec.ts +53 -0
  360. package/src/exec/non-interactive-env.ts +48 -0
  361. package/src/export/custom-share.ts +65 -0
  362. package/src/export/html/index.ts +164 -0
  363. package/src/export/html/template.css +1051 -0
  364. package/src/export/html/template.generated.ts +2 -0
  365. package/src/export/html/template.html +46 -0
  366. package/src/export/html/template.js +2271 -0
  367. package/src/export/html/template.macro.ts +25 -0
  368. package/src/export/html/vendor/highlight.min.js +1213 -0
  369. package/src/export/html/vendor/marked.min.js +6 -0
  370. package/src/export/ttsr.ts +583 -0
  371. package/src/extensibility/custom-commands/bundled/ci-green/index.ts +54 -0
  372. package/src/extensibility/custom-commands/bundled/review/index.ts +489 -0
  373. package/src/extensibility/custom-commands/index.ts +2 -0
  374. package/src/extensibility/custom-commands/loader.ts +238 -0
  375. package/src/extensibility/custom-commands/types.ts +113 -0
  376. package/src/extensibility/custom-tools/index.ts +7 -0
  377. package/src/extensibility/custom-tools/loader.ts +269 -0
  378. package/src/extensibility/custom-tools/types.ts +270 -0
  379. package/src/extensibility/custom-tools/wrapper.ts +47 -0
  380. package/src/extensibility/extensions/compact-handler.ts +40 -0
  381. package/src/extensibility/extensions/get-commands-handler.ts +78 -0
  382. package/src/extensibility/extensions/index.ts +16 -0
  383. package/src/extensibility/extensions/loader.ts +572 -0
  384. package/src/extensibility/extensions/runner.ts +922 -0
  385. package/src/extensibility/extensions/types.ts +1322 -0
  386. package/src/extensibility/extensions/wrapper.ts +223 -0
  387. package/src/extensibility/hooks/index.ts +5 -0
  388. package/src/extensibility/hooks/loader.ts +257 -0
  389. package/src/extensibility/hooks/runner.ts +425 -0
  390. package/src/extensibility/hooks/tool-wrapper.ts +107 -0
  391. package/src/extensibility/hooks/types.ts +606 -0
  392. package/src/extensibility/legacy-pi-ai-shim.ts +24 -0
  393. package/src/extensibility/legacy-pi-coding-agent-shim.ts +15 -0
  394. package/src/extensibility/plugins/doctor.ts +65 -0
  395. package/src/extensibility/plugins/git-url.ts +367 -0
  396. package/src/extensibility/plugins/index.ts +9 -0
  397. package/src/extensibility/plugins/installer.ts +192 -0
  398. package/src/extensibility/plugins/legacy-pi-compat.ts +682 -0
  399. package/src/extensibility/plugins/loader.ts +313 -0
  400. package/src/extensibility/plugins/manager.ts +827 -0
  401. package/src/extensibility/plugins/marketplace/cache.ts +136 -0
  402. package/src/extensibility/plugins/marketplace/fetcher.ts +317 -0
  403. package/src/extensibility/plugins/marketplace/index.ts +6 -0
  404. package/src/extensibility/plugins/marketplace/manager.ts +770 -0
  405. package/src/extensibility/plugins/marketplace/registry.ts +196 -0
  406. package/src/extensibility/plugins/marketplace/source-resolver.ts +147 -0
  407. package/src/extensibility/plugins/marketplace/types.ts +191 -0
  408. package/src/extensibility/plugins/marketplace-auto-update.ts +49 -0
  409. package/src/extensibility/plugins/parser.ts +105 -0
  410. package/src/extensibility/plugins/types.ts +194 -0
  411. package/src/extensibility/shared-events.ts +343 -0
  412. package/src/extensibility/skills.ts +312 -0
  413. package/src/extensibility/slash-commands.ts +227 -0
  414. package/src/extensibility/tool-proxy.ts +25 -0
  415. package/src/extensibility/typebox.ts +418 -0
  416. package/src/extensibility/utils.ts +44 -0
  417. package/src/goals/index.ts +3 -0
  418. package/src/goals/runtime.ts +528 -0
  419. package/src/goals/state.ts +37 -0
  420. package/src/goals/tools/goal-tool.ts +251 -0
  421. package/src/hindsight/backend.ts +354 -0
  422. package/src/hindsight/bank.ts +156 -0
  423. package/src/hindsight/client.ts +598 -0
  424. package/src/hindsight/config.ts +175 -0
  425. package/src/hindsight/content.ts +210 -0
  426. package/src/hindsight/index.ts +8 -0
  427. package/src/hindsight/mental-models.ts +429 -0
  428. package/src/hindsight/seeds.json +32 -0
  429. package/src/hindsight/state.ts +488 -0
  430. package/src/hindsight/transcript.ts +71 -0
  431. package/src/index.ts +59 -0
  432. package/src/internal-urls/agent-protocol.ts +146 -0
  433. package/src/internal-urls/artifact-protocol.ts +107 -0
  434. package/src/internal-urls/docs-index.generated.ts +106 -0
  435. package/src/internal-urls/history-protocol.ts +113 -0
  436. package/src/internal-urls/index.ts +25 -0
  437. package/src/internal-urls/issue-pr-protocol.ts +584 -0
  438. package/src/internal-urls/json-query.ts +126 -0
  439. package/src/internal-urls/local-protocol.ts +287 -0
  440. package/src/internal-urls/mcp-protocol.ts +151 -0
  441. package/src/internal-urls/memory-protocol.ts +169 -0
  442. package/src/internal-urls/omp-protocol.ts +93 -0
  443. package/src/internal-urls/parse.ts +72 -0
  444. package/src/internal-urls/registry-helpers.ts +25 -0
  445. package/src/internal-urls/router.ts +105 -0
  446. package/src/internal-urls/rule-protocol.ts +45 -0
  447. package/src/internal-urls/skill-protocol.ts +96 -0
  448. package/src/internal-urls/types.ts +152 -0
  449. package/src/internal-urls/vault-protocol.ts +936 -0
  450. package/src/irc/bus.ts +292 -0
  451. package/src/lib/xai-http.ts +124 -0
  452. package/src/lsp/client.ts +1193 -0
  453. package/src/lsp/clients/biome-client.ts +264 -0
  454. package/src/lsp/clients/index.ts +50 -0
  455. package/src/lsp/clients/lsp-linter-client.ts +93 -0
  456. package/src/lsp/clients/swiftlint-client.ts +120 -0
  457. package/src/lsp/config.ts +502 -0
  458. package/src/lsp/defaults.json +493 -0
  459. package/src/lsp/diagnostics-ledger.ts +51 -0
  460. package/src/lsp/edits.ts +267 -0
  461. package/src/lsp/index.ts +2477 -0
  462. package/src/lsp/lspmux.ts +233 -0
  463. package/src/lsp/render.ts +694 -0
  464. package/src/lsp/startup-events.ts +13 -0
  465. package/src/lsp/types.ts +455 -0
  466. package/src/lsp/utils.ts +718 -0
  467. package/src/main.ts +1325 -0
  468. package/src/mcp/client.ts +484 -0
  469. package/src/mcp/config-writer.ts +225 -0
  470. package/src/mcp/config.ts +365 -0
  471. package/src/mcp/index.ts +29 -0
  472. package/src/mcp/json-rpc.ts +122 -0
  473. package/src/mcp/loader.ts +124 -0
  474. package/src/mcp/manager.ts +1275 -0
  475. package/src/mcp/oauth-discovery.ts +442 -0
  476. package/src/mcp/oauth-flow.ts +442 -0
  477. package/src/mcp/render.ts +132 -0
  478. package/src/mcp/smithery-auth.ts +104 -0
  479. package/src/mcp/smithery-connect.ts +145 -0
  480. package/src/mcp/smithery-registry.ts +477 -0
  481. package/src/mcp/timeout.ts +59 -0
  482. package/src/mcp/tool-bridge.ts +426 -0
  483. package/src/mcp/tool-cache.ts +117 -0
  484. package/src/mcp/transports/http.ts +519 -0
  485. package/src/mcp/transports/index.ts +6 -0
  486. package/src/mcp/transports/stdio.ts +528 -0
  487. package/src/mcp/types.ts +423 -0
  488. package/src/memories/index.ts +1150 -0
  489. package/src/memories/storage.ts +577 -0
  490. package/src/memory-backend/index.ts +18 -0
  491. package/src/memory-backend/local-backend.ts +39 -0
  492. package/src/memory-backend/off-backend.ts +25 -0
  493. package/src/memory-backend/resolve.ts +25 -0
  494. package/src/memory-backend/runtime.ts +66 -0
  495. package/src/memory-backend/types.ts +166 -0
  496. package/src/mnemopi/backend.ts +547 -0
  497. package/src/mnemopi/config.ts +160 -0
  498. package/src/mnemopi/index.ts +3 -0
  499. package/src/mnemopi/state.ts +584 -0
  500. package/src/modes/acp/acp-agent.ts +2407 -0
  501. package/src/modes/acp/acp-client-bridge.ts +154 -0
  502. package/src/modes/acp/acp-event-mapper.ts +929 -0
  503. package/src/modes/acp/acp-mode.ts +23 -0
  504. package/src/modes/acp/index.ts +2 -0
  505. package/src/modes/acp/terminal-auth.ts +37 -0
  506. package/src/modes/components/agent-dashboard.ts +1206 -0
  507. package/src/modes/components/agent-hub.ts +1071 -0
  508. package/src/modes/components/assistant-message.ts +307 -0
  509. package/src/modes/components/bash-execution.ts +220 -0
  510. package/src/modes/components/bordered-loader.ts +41 -0
  511. package/src/modes/components/branch-summary-message.ts +45 -0
  512. package/src/modes/components/btw-panel.ts +104 -0
  513. package/src/modes/components/chat-block.ts +111 -0
  514. package/src/modes/components/compaction-summary-message.ts +87 -0
  515. package/src/modes/components/copy-selector.ts +206 -0
  516. package/src/modes/components/countdown-timer.ts +75 -0
  517. package/src/modes/components/custom-editor.ts +398 -0
  518. package/src/modes/components/custom-message.ts +63 -0
  519. package/src/modes/components/diff.ts +277 -0
  520. package/src/modes/components/dynamic-border.ts +34 -0
  521. package/src/modes/components/error-banner.ts +33 -0
  522. package/src/modes/components/eval-execution.ts +158 -0
  523. package/src/modes/components/execution-shared.ts +101 -0
  524. package/src/modes/components/extensions/extension-dashboard.ts +399 -0
  525. package/src/modes/components/extensions/extension-list.ts +502 -0
  526. package/src/modes/components/extensions/index.ts +9 -0
  527. package/src/modes/components/extensions/inspector-panel.ts +317 -0
  528. package/src/modes/components/extensions/state-manager.ts +627 -0
  529. package/src/modes/components/extensions/types.ts +186 -0
  530. package/src/modes/components/footer.ts +274 -0
  531. package/src/modes/components/history-search.ts +280 -0
  532. package/src/modes/components/hook-editor.ts +167 -0
  533. package/src/modes/components/hook-input.ts +87 -0
  534. package/src/modes/components/hook-message.ts +66 -0
  535. package/src/modes/components/hook-selector.ts +660 -0
  536. package/src/modes/components/index.ts +38 -0
  537. package/src/modes/components/keybinding-hints.ts +65 -0
  538. package/src/modes/components/late-diagnostics-message.ts +60 -0
  539. package/src/modes/components/login-dialog.ts +164 -0
  540. package/src/modes/components/mcp-add-wizard.ts +1340 -0
  541. package/src/modes/components/message-frame.ts +88 -0
  542. package/src/modes/components/model-selector.ts +1271 -0
  543. package/src/modes/components/oauth-selector.ts +368 -0
  544. package/src/modes/components/omfg-panel.ts +141 -0
  545. package/src/modes/components/overlay-box.ts +108 -0
  546. package/src/modes/components/plan-review-overlay.ts +820 -0
  547. package/src/modes/components/plan-toc.ts +138 -0
  548. package/src/modes/components/plugin-selector.ts +95 -0
  549. package/src/modes/components/plugin-settings.ts +722 -0
  550. package/src/modes/components/queue-mode-selector.ts +56 -0
  551. package/src/modes/components/read-tool-group.ts +670 -0
  552. package/src/modes/components/segment-track.ts +52 -0
  553. package/src/modes/components/session-selector.ts +625 -0
  554. package/src/modes/components/settings-defs.ts +189 -0
  555. package/src/modes/components/settings-selector.ts +651 -0
  556. package/src/modes/components/show-images-selector.ts +45 -0
  557. package/src/modes/components/skill-message.ts +89 -0
  558. package/src/modes/components/status-line/component.ts +869 -0
  559. package/src/modes/components/status-line/context-thresholds.ts +79 -0
  560. package/src/modes/components/status-line/git-utils.ts +42 -0
  561. package/src/modes/components/status-line/index.ts +5 -0
  562. package/src/modes/components/status-line/presets.ts +106 -0
  563. package/src/modes/components/status-line/segments.ts +584 -0
  564. package/src/modes/components/status-line/separators.ts +55 -0
  565. package/src/modes/components/status-line/token-rate.ts +66 -0
  566. package/src/modes/components/status-line/types.ts +108 -0
  567. package/src/modes/components/theme-selector.ts +63 -0
  568. package/src/modes/components/thinking-selector.ts +52 -0
  569. package/src/modes/components/tiny-title-download-progress.ts +90 -0
  570. package/src/modes/components/tips.txt +19 -0
  571. package/src/modes/components/todo-reminder.ts +38 -0
  572. package/src/modes/components/tool-execution.ts +1024 -0
  573. package/src/modes/components/transcript-container.ts +608 -0
  574. package/src/modes/components/tree-selector.ts +978 -0
  575. package/src/modes/components/ttsr-notification.ts +122 -0
  576. package/src/modes/components/user-message-selector.ts +227 -0
  577. package/src/modes/components/user-message.ts +66 -0
  578. package/src/modes/components/visual-truncate.ts +63 -0
  579. package/src/modes/components/welcome.ts +493 -0
  580. package/src/modes/controllers/btw-controller.ts +105 -0
  581. package/src/modes/controllers/command-controller-shared.ts +109 -0
  582. package/src/modes/controllers/command-controller.ts +1566 -0
  583. package/src/modes/controllers/event-controller.ts +1054 -0
  584. package/src/modes/controllers/extension-ui-controller.ts +886 -0
  585. package/src/modes/controllers/input-controller.ts +1073 -0
  586. package/src/modes/controllers/mcp-command-controller.ts +2017 -0
  587. package/src/modes/controllers/omfg-controller.ts +283 -0
  588. package/src/modes/controllers/omfg-rule.ts +647 -0
  589. package/src/modes/controllers/selector-controller.ts +1108 -0
  590. package/src/modes/controllers/ssh-command-controller.ts +384 -0
  591. package/src/modes/controllers/streaming-reveal.ts +279 -0
  592. package/src/modes/controllers/tan-command-controller.ts +173 -0
  593. package/src/modes/controllers/todo-command-controller.ts +485 -0
  594. package/src/modes/data/emojis.json +1 -0
  595. package/src/modes/emoji-autocomplete.ts +285 -0
  596. package/src/modes/gradient-highlight.ts +87 -0
  597. package/src/modes/image-references.ts +117 -0
  598. package/src/modes/index.ts +17 -0
  599. package/src/modes/interactive-mode.ts +3370 -0
  600. package/src/modes/internal-url-autocomplete.ts +143 -0
  601. package/src/modes/loop-limit.ts +140 -0
  602. package/src/modes/magic-keywords.ts +20 -0
  603. package/src/modes/markdown-prose.ts +247 -0
  604. package/src/modes/oauth-manual-input.ts +69 -0
  605. package/src/modes/orchestrate.ts +42 -0
  606. package/src/modes/print-mode.ts +126 -0
  607. package/src/modes/prompt-action-autocomplete.ts +260 -0
  608. package/src/modes/rpc/host-tools.ts +186 -0
  609. package/src/modes/rpc/host-uris.ts +235 -0
  610. package/src/modes/rpc/rpc-client.ts +963 -0
  611. package/src/modes/rpc/rpc-mode.ts +947 -0
  612. package/src/modes/rpc/rpc-subagents.ts +265 -0
  613. package/src/modes/rpc/rpc-types.ts +458 -0
  614. package/src/modes/runtime-init.ts +116 -0
  615. package/src/modes/session-observer-registry.ts +146 -0
  616. package/src/modes/setup-version.ts +11 -0
  617. package/src/modes/setup-wizard/index.ts +99 -0
  618. package/src/modes/setup-wizard/lazy.ts +16 -0
  619. package/src/modes/setup-wizard/scenes/glyph.ts +96 -0
  620. package/src/modes/setup-wizard/scenes/outro.ts +35 -0
  621. package/src/modes/setup-wizard/scenes/providers.ts +69 -0
  622. package/src/modes/setup-wizard/scenes/sign-in.ts +205 -0
  623. package/src/modes/setup-wizard/scenes/splash.ts +201 -0
  624. package/src/modes/setup-wizard/scenes/theme.ts +299 -0
  625. package/src/modes/setup-wizard/scenes/types.ts +48 -0
  626. package/src/modes/setup-wizard/scenes/web-search.ts +129 -0
  627. package/src/modes/setup-wizard/wizard-overlay.ts +275 -0
  628. package/src/modes/shared.ts +47 -0
  629. package/src/modes/theme/dark.json +95 -0
  630. package/src/modes/theme/defaults/alabaster.json +93 -0
  631. package/src/modes/theme/defaults/amethyst.json +96 -0
  632. package/src/modes/theme/defaults/anthracite.json +93 -0
  633. package/src/modes/theme/defaults/basalt.json +91 -0
  634. package/src/modes/theme/defaults/birch.json +95 -0
  635. package/src/modes/theme/defaults/dark-abyss.json +91 -0
  636. package/src/modes/theme/defaults/dark-arctic.json +104 -0
  637. package/src/modes/theme/defaults/dark-aurora.json +95 -0
  638. package/src/modes/theme/defaults/dark-catppuccin.json +107 -0
  639. package/src/modes/theme/defaults/dark-cavern.json +91 -0
  640. package/src/modes/theme/defaults/dark-copper.json +95 -0
  641. package/src/modes/theme/defaults/dark-cosmos.json +90 -0
  642. package/src/modes/theme/defaults/dark-cyberpunk.json +102 -0
  643. package/src/modes/theme/defaults/dark-dracula.json +98 -0
  644. package/src/modes/theme/defaults/dark-eclipse.json +91 -0
  645. package/src/modes/theme/defaults/dark-ember.json +95 -0
  646. package/src/modes/theme/defaults/dark-equinox.json +90 -0
  647. package/src/modes/theme/defaults/dark-forest.json +96 -0
  648. package/src/modes/theme/defaults/dark-github.json +105 -0
  649. package/src/modes/theme/defaults/dark-gruvbox.json +112 -0
  650. package/src/modes/theme/defaults/dark-lavender.json +95 -0
  651. package/src/modes/theme/defaults/dark-lunar.json +89 -0
  652. package/src/modes/theme/defaults/dark-midnight.json +95 -0
  653. package/src/modes/theme/defaults/dark-monochrome.json +94 -0
  654. package/src/modes/theme/defaults/dark-monokai.json +98 -0
  655. package/src/modes/theme/defaults/dark-nebula.json +90 -0
  656. package/src/modes/theme/defaults/dark-nord.json +97 -0
  657. package/src/modes/theme/defaults/dark-ocean.json +101 -0
  658. package/src/modes/theme/defaults/dark-one.json +100 -0
  659. package/src/modes/theme/defaults/dark-poimandres.json +142 -0
  660. package/src/modes/theme/defaults/dark-rainforest.json +91 -0
  661. package/src/modes/theme/defaults/dark-reef.json +91 -0
  662. package/src/modes/theme/defaults/dark-retro.json +92 -0
  663. package/src/modes/theme/defaults/dark-rose-pine.json +96 -0
  664. package/src/modes/theme/defaults/dark-sakura.json +95 -0
  665. package/src/modes/theme/defaults/dark-slate.json +95 -0
  666. package/src/modes/theme/defaults/dark-solarized.json +97 -0
  667. package/src/modes/theme/defaults/dark-solstice.json +90 -0
  668. package/src/modes/theme/defaults/dark-starfall.json +91 -0
  669. package/src/modes/theme/defaults/dark-sunset.json +99 -0
  670. package/src/modes/theme/defaults/dark-swamp.json +90 -0
  671. package/src/modes/theme/defaults/dark-synthwave.json +103 -0
  672. package/src/modes/theme/defaults/dark-taiga.json +91 -0
  673. package/src/modes/theme/defaults/dark-terminal.json +95 -0
  674. package/src/modes/theme/defaults/dark-tokyo-night.json +101 -0
  675. package/src/modes/theme/defaults/dark-tundra.json +91 -0
  676. package/src/modes/theme/defaults/dark-twilight.json +91 -0
  677. package/src/modes/theme/defaults/dark-volcanic.json +91 -0
  678. package/src/modes/theme/defaults/graphite.json +92 -0
  679. package/src/modes/theme/defaults/index.ts +199 -0
  680. package/src/modes/theme/defaults/light-arctic.json +107 -0
  681. package/src/modes/theme/defaults/light-aurora-day.json +91 -0
  682. package/src/modes/theme/defaults/light-canyon.json +91 -0
  683. package/src/modes/theme/defaults/light-catppuccin.json +106 -0
  684. package/src/modes/theme/defaults/light-cirrus.json +90 -0
  685. package/src/modes/theme/defaults/light-coral.json +95 -0
  686. package/src/modes/theme/defaults/light-cyberpunk.json +96 -0
  687. package/src/modes/theme/defaults/light-dawn.json +90 -0
  688. package/src/modes/theme/defaults/light-dunes.json +91 -0
  689. package/src/modes/theme/defaults/light-eucalyptus.json +95 -0
  690. package/src/modes/theme/defaults/light-forest.json +100 -0
  691. package/src/modes/theme/defaults/light-frost.json +95 -0
  692. package/src/modes/theme/defaults/light-github.json +115 -0
  693. package/src/modes/theme/defaults/light-glacier.json +91 -0
  694. package/src/modes/theme/defaults/light-gruvbox.json +108 -0
  695. package/src/modes/theme/defaults/light-haze.json +90 -0
  696. package/src/modes/theme/defaults/light-honeycomb.json +95 -0
  697. package/src/modes/theme/defaults/light-lagoon.json +91 -0
  698. package/src/modes/theme/defaults/light-lavender.json +95 -0
  699. package/src/modes/theme/defaults/light-meadow.json +91 -0
  700. package/src/modes/theme/defaults/light-mint.json +95 -0
  701. package/src/modes/theme/defaults/light-monochrome.json +101 -0
  702. package/src/modes/theme/defaults/light-ocean.json +99 -0
  703. package/src/modes/theme/defaults/light-one.json +99 -0
  704. package/src/modes/theme/defaults/light-opal.json +91 -0
  705. package/src/modes/theme/defaults/light-orchard.json +91 -0
  706. package/src/modes/theme/defaults/light-paper.json +95 -0
  707. package/src/modes/theme/defaults/light-poimandres.json +142 -0
  708. package/src/modes/theme/defaults/light-prism.json +90 -0
  709. package/src/modes/theme/defaults/light-retro.json +98 -0
  710. package/src/modes/theme/defaults/light-sand.json +95 -0
  711. package/src/modes/theme/defaults/light-savanna.json +91 -0
  712. package/src/modes/theme/defaults/light-solarized.json +102 -0
  713. package/src/modes/theme/defaults/light-soleil.json +90 -0
  714. package/src/modes/theme/defaults/light-sunset.json +99 -0
  715. package/src/modes/theme/defaults/light-synthwave.json +98 -0
  716. package/src/modes/theme/defaults/light-tokyo-night.json +111 -0
  717. package/src/modes/theme/defaults/light-wetland.json +91 -0
  718. package/src/modes/theme/defaults/light-zenith.json +89 -0
  719. package/src/modes/theme/defaults/limestone.json +94 -0
  720. package/src/modes/theme/defaults/mahogany.json +97 -0
  721. package/src/modes/theme/defaults/marble.json +93 -0
  722. package/src/modes/theme/defaults/obsidian.json +91 -0
  723. package/src/modes/theme/defaults/onyx.json +91 -0
  724. package/src/modes/theme/defaults/pearl.json +93 -0
  725. package/src/modes/theme/defaults/porcelain.json +91 -0
  726. package/src/modes/theme/defaults/quartz.json +96 -0
  727. package/src/modes/theme/defaults/sandstone.json +95 -0
  728. package/src/modes/theme/defaults/titanium.json +90 -0
  729. package/src/modes/theme/light.json +93 -0
  730. package/src/modes/theme/mermaid-cache.ts +29 -0
  731. package/src/modes/theme/shimmer.ts +235 -0
  732. package/src/modes/theme/theme-schema.json +459 -0
  733. package/src/modes/theme/theme.ts +2676 -0
  734. package/src/modes/turn-budget.ts +31 -0
  735. package/src/modes/types.ts +359 -0
  736. package/src/modes/ultrathink.ts +41 -0
  737. package/src/modes/utils/context-usage.ts +339 -0
  738. package/src/modes/utils/copy-targets.ts +360 -0
  739. package/src/modes/utils/hotkeys-markdown.ts +61 -0
  740. package/src/modes/utils/keybinding-matchers.ts +51 -0
  741. package/src/modes/utils/tools-markdown.ts +27 -0
  742. package/src/modes/utils/ui-helpers.ts +801 -0
  743. package/src/modes/workflow.ts +42 -0
  744. package/src/plan-mode/approved-plan.ts +186 -0
  745. package/src/plan-mode/plan-handoff.ts +37 -0
  746. package/src/plan-mode/plan-protection.ts +31 -0
  747. package/src/plan-mode/state.ts +6 -0
  748. package/src/priority.json +41 -0
  749. package/src/prompts/agents/designer.md +66 -0
  750. package/src/prompts/agents/explore.md +58 -0
  751. package/src/prompts/agents/frontmatter.md +11 -0
  752. package/src/prompts/agents/init.md +33 -0
  753. package/src/prompts/agents/librarian.md +119 -0
  754. package/src/prompts/agents/oracle.md +55 -0
  755. package/src/prompts/agents/plan.md +48 -0
  756. package/src/prompts/agents/reviewer.md +140 -0
  757. package/src/prompts/agents/task.md +16 -0
  758. package/src/prompts/ci-green-request.md +36 -0
  759. package/src/prompts/dry-balance-bench.md +8 -0
  760. package/src/prompts/goals/goal-budget-limit.md +16 -0
  761. package/src/prompts/goals/goal-continuation.md +28 -0
  762. package/src/prompts/goals/goal-mode-active.md +23 -0
  763. package/src/prompts/memories/consolidation.md +30 -0
  764. package/src/prompts/memories/read-path.md +11 -0
  765. package/src/prompts/memories/stage_one_input.md +6 -0
  766. package/src/prompts/memories/stage_one_system.md +21 -0
  767. package/src/prompts/review-custom-request.md +22 -0
  768. package/src/prompts/review-headless-request.md +16 -0
  769. package/src/prompts/review-request.md +69 -0
  770. package/src/prompts/steering/user-interjection.md +10 -0
  771. package/src/prompts/system/agent-creation-architect.md +50 -0
  772. package/src/prompts/system/agent-creation-user.md +6 -0
  773. package/src/prompts/system/auto-continue.md +1 -0
  774. package/src/prompts/system/auto-thinking-difficulty-local.md +14 -0
  775. package/src/prompts/system/auto-thinking-difficulty.md +12 -0
  776. package/src/prompts/system/background-tan-dispatch.md +8 -0
  777. package/src/prompts/system/btw-user.md +8 -0
  778. package/src/prompts/system/commit-message-system.md +14 -0
  779. package/src/prompts/system/custom-system-prompt.md +64 -0
  780. package/src/prompts/system/eager-todo.md +13 -0
  781. package/src/prompts/system/empty-stop-retry.md +6 -0
  782. package/src/prompts/system/irc-incoming.md +7 -0
  783. package/src/prompts/system/manual-continue.md +7 -0
  784. package/src/prompts/system/memory-consolidation-system.md +8 -0
  785. package/src/prompts/system/memory-extraction-system.md +26 -0
  786. package/src/prompts/system/omfg-user.md +50 -0
  787. package/src/prompts/system/orchestrate-notice.md +40 -0
  788. package/src/prompts/system/plan-mode-active.md +109 -0
  789. package/src/prompts/system/plan-mode-approved.md +25 -0
  790. package/src/prompts/system/plan-mode-compact-instructions.md +16 -0
  791. package/src/prompts/system/plan-mode-reference.md +11 -0
  792. package/src/prompts/system/plan-mode-subagent.md +33 -0
  793. package/src/prompts/system/plan-mode-tool-decision-reminder.md +9 -0
  794. package/src/prompts/system/project-prompt.md +52 -0
  795. package/src/prompts/system/subagent-system-prompt.md +64 -0
  796. package/src/prompts/system/subagent-user-prompt.md +3 -0
  797. package/src/prompts/system/subagent-yield-reminder.md +12 -0
  798. package/src/prompts/system/system-prompt.md +258 -0
  799. package/src/prompts/system/tiny-title-system.md +8 -0
  800. package/src/prompts/system/title-system.md +16 -0
  801. package/src/prompts/system/ttsr-interrupt.md +7 -0
  802. package/src/prompts/system/ttsr-tool-reminder.md +5 -0
  803. package/src/prompts/system/ultrathink-notice.md +3 -0
  804. package/src/prompts/system/web-search.md +25 -0
  805. package/src/prompts/system/workflow-notice.md +70 -0
  806. package/src/prompts/tools/apply-patch.md +65 -0
  807. package/src/prompts/tools/ask.md +30 -0
  808. package/src/prompts/tools/ast-edit.md +39 -0
  809. package/src/prompts/tools/ast-grep.md +42 -0
  810. package/src/prompts/tools/async-result.md +8 -0
  811. package/src/prompts/tools/bash.md +46 -0
  812. package/src/prompts/tools/browser.md +73 -0
  813. package/src/prompts/tools/checkpoint.md +16 -0
  814. package/src/prompts/tools/debug.md +34 -0
  815. package/src/prompts/tools/eval.md +92 -0
  816. package/src/prompts/tools/find.md +36 -0
  817. package/src/prompts/tools/github.md +21 -0
  818. package/src/prompts/tools/goal.md +18 -0
  819. package/src/prompts/tools/image-gen.md +7 -0
  820. package/src/prompts/tools/inspect-image-system.md +20 -0
  821. package/src/prompts/tools/inspect-image.md +32 -0
  822. package/src/prompts/tools/irc.md +59 -0
  823. package/src/prompts/tools/job.md +19 -0
  824. package/src/prompts/tools/lsp-late-diagnostic.md +8 -0
  825. package/src/prompts/tools/lsp.md +42 -0
  826. package/src/prompts/tools/memory-edit.md +8 -0
  827. package/src/prompts/tools/patch.md +70 -0
  828. package/src/prompts/tools/read.md +84 -0
  829. package/src/prompts/tools/recall.md +5 -0
  830. package/src/prompts/tools/reflect.md +5 -0
  831. package/src/prompts/tools/render-mermaid.md +9 -0
  832. package/src/prompts/tools/replace.md +30 -0
  833. package/src/prompts/tools/resolve.md +9 -0
  834. package/src/prompts/tools/retain.md +6 -0
  835. package/src/prompts/tools/rewind.md +13 -0
  836. package/src/prompts/tools/search-tool-bm25.md +32 -0
  837. package/src/prompts/tools/search.md +24 -0
  838. package/src/prompts/tools/ssh.md +31 -0
  839. package/src/prompts/tools/task-summary.md +17 -0
  840. package/src/prompts/tools/task.md +88 -0
  841. package/src/prompts/tools/todo.md +62 -0
  842. package/src/prompts/tools/web-search.md +10 -0
  843. package/src/prompts/tools/write.md +14 -0
  844. package/src/registry/agent-lifecycle.ts +218 -0
  845. package/src/registry/agent-registry.ts +151 -0
  846. package/src/sdk.ts +2558 -0
  847. package/src/secrets/index.ts +123 -0
  848. package/src/secrets/obfuscator.ts +298 -0
  849. package/src/secrets/regex.ts +21 -0
  850. package/src/session/agent-session.ts +10121 -0
  851. package/src/session/agent-storage.ts +455 -0
  852. package/src/session/artifacts.ts +135 -0
  853. package/src/session/auth-broker-config.ts +131 -0
  854. package/src/session/auth-storage.ts +29 -0
  855. package/src/session/blob-store.ts +255 -0
  856. package/src/session/client-bridge.ts +85 -0
  857. package/src/session/history-storage.ts +348 -0
  858. package/src/session/indexed-session-storage.ts +430 -0
  859. package/src/session/messages.ts +541 -0
  860. package/src/session/redis-session-storage.ts +170 -0
  861. package/src/session/session-dump-format.ts +209 -0
  862. package/src/session/session-history-format.ts +246 -0
  863. package/src/session/session-manager.ts +3676 -0
  864. package/src/session/session-storage.ts +529 -0
  865. package/src/session/shake-types.ts +43 -0
  866. package/src/session/sql-session-storage.ts +314 -0
  867. package/src/session/streaming-output.ts +1330 -0
  868. package/src/session/tool-choice-queue.ts +213 -0
  869. package/src/session/yield-queue.ts +173 -0
  870. package/src/slash-commands/acp-builtins.ts +70 -0
  871. package/src/slash-commands/builtin-registry.ts +1798 -0
  872. package/src/slash-commands/helpers/context-report.ts +39 -0
  873. package/src/slash-commands/helpers/format.ts +46 -0
  874. package/src/slash-commands/helpers/marketplace-manager.ts +25 -0
  875. package/src/slash-commands/helpers/mcp.ts +532 -0
  876. package/src/slash-commands/helpers/parse.ts +85 -0
  877. package/src/slash-commands/helpers/ssh.ts +195 -0
  878. package/src/slash-commands/helpers/stats-dashboard.ts +85 -0
  879. package/src/slash-commands/helpers/todo.ts +279 -0
  880. package/src/slash-commands/helpers/usage-report.ts +95 -0
  881. package/src/slash-commands/marketplace-install-parser.ts +99 -0
  882. package/src/slash-commands/types.ts +135 -0
  883. package/src/ssh/config-writer.ts +183 -0
  884. package/src/ssh/connection-manager.ts +509 -0
  885. package/src/ssh/ssh-executor.ts +189 -0
  886. package/src/ssh/sshfs-mount.ts +140 -0
  887. package/src/ssh/utils.ts +8 -0
  888. package/src/stt/downloader.ts +71 -0
  889. package/src/stt/index.ts +3 -0
  890. package/src/stt/recorder.ts +351 -0
  891. package/src/stt/setup.ts +52 -0
  892. package/src/stt/stt-controller.ts +160 -0
  893. package/src/stt/transcribe.py +70 -0
  894. package/src/stt/transcriber.ts +91 -0
  895. package/src/stubs/natives/index.ts +814 -0
  896. package/src/stubs/natives/package.json +7 -0
  897. package/src/stubs/tui/index.ts +282 -0
  898. package/src/stubs/tui/package.json +7 -0
  899. package/src/system-prompt.ts +611 -0
  900. package/src/task/agents.ts +167 -0
  901. package/src/task/commands.ts +132 -0
  902. package/src/task/discovery.ts +122 -0
  903. package/src/task/executor.ts +2133 -0
  904. package/src/task/index.ts +1419 -0
  905. package/src/task/name-generator.ts +1577 -0
  906. package/src/task/omp-command.ts +26 -0
  907. package/src/task/output-manager.ts +88 -0
  908. package/src/task/parallel.ts +116 -0
  909. package/src/task/render.ts +1381 -0
  910. package/src/task/repair-args.ts +129 -0
  911. package/src/task/subprocess-tool-registry.ts +88 -0
  912. package/src/task/types.ts +336 -0
  913. package/src/task/worktree.ts +514 -0
  914. package/src/telemetry-export.ts +144 -0
  915. package/src/thinking.ts +167 -0
  916. package/src/tiny/compiled-runtime.ts +179 -0
  917. package/src/tiny/device.ts +111 -0
  918. package/src/tiny/dtype.ts +101 -0
  919. package/src/tiny/models.ts +242 -0
  920. package/src/tiny/text.ts +165 -0
  921. package/src/tiny/title-client.ts +543 -0
  922. package/src/tiny/title-protocol.ts +56 -0
  923. package/src/tiny/worker.ts +568 -0
  924. package/src/tool-discovery/mode.ts +24 -0
  925. package/src/tool-discovery/tool-index.ts +256 -0
  926. package/src/tools/approval.ts +189 -0
  927. package/src/tools/archive-reader.ts +721 -0
  928. package/src/tools/ask.ts +928 -0
  929. package/src/tools/ast-edit.ts +642 -0
  930. package/src/tools/ast-grep.ts +452 -0
  931. package/src/tools/auto-generated-guard.ts +322 -0
  932. package/src/tools/bash-command-fixup.ts +37 -0
  933. package/src/tools/bash-interactive.ts +408 -0
  934. package/src/tools/bash-interceptor.ts +67 -0
  935. package/src/tools/bash-pty-selection.ts +14 -0
  936. package/src/tools/bash-skill-urls.ts +248 -0
  937. package/src/tools/bash.ts +1386 -0
  938. package/src/tools/browser/attach.ts +175 -0
  939. package/src/tools/browser/launch.ts +660 -0
  940. package/src/tools/browser/readable.ts +112 -0
  941. package/src/tools/browser/registry.ts +197 -0
  942. package/src/tools/browser/render.ts +216 -0
  943. package/src/tools/browser/tab-protocol.ts +105 -0
  944. package/src/tools/browser/tab-supervisor.ts +628 -0
  945. package/src/tools/browser/tab-worker-entry.ts +21 -0
  946. package/src/tools/browser/tab-worker.ts +1226 -0
  947. package/src/tools/browser.ts +343 -0
  948. package/src/tools/checkpoint.ts +136 -0
  949. package/src/tools/conflict-detect.ts +718 -0
  950. package/src/tools/context.ts +39 -0
  951. package/src/tools/debug.ts +1067 -0
  952. package/src/tools/eval-backends.ts +27 -0
  953. package/src/tools/eval-render.ts +752 -0
  954. package/src/tools/eval.ts +577 -0
  955. package/src/tools/fetch.ts +1926 -0
  956. package/src/tools/file-recorder.ts +35 -0
  957. package/src/tools/find.ts +609 -0
  958. package/src/tools/fs-cache-invalidation.ts +28 -0
  959. package/src/tools/gh-cache-invalidation.ts +255 -0
  960. package/src/tools/gh-format.ts +12 -0
  961. package/src/tools/gh-renderer.ts +481 -0
  962. package/src/tools/gh.ts +3720 -0
  963. package/src/tools/github-cache.ts +637 -0
  964. package/src/tools/grouped-file-output.ts +210 -0
  965. package/src/tools/image-gen.ts +1517 -0
  966. package/src/tools/index.ts +599 -0
  967. package/src/tools/inspect-image-renderer.ts +132 -0
  968. package/src/tools/inspect-image.ts +174 -0
  969. package/src/tools/irc.ts +723 -0
  970. package/src/tools/job.ts +557 -0
  971. package/src/tools/json-tree.ts +243 -0
  972. package/src/tools/jtd-to-json-schema.ts +219 -0
  973. package/src/tools/jtd-to-typescript.ts +136 -0
  974. package/src/tools/jtd-utils.ts +102 -0
  975. package/src/tools/list-limit.ts +40 -0
  976. package/src/tools/match-line-format.ts +20 -0
  977. package/src/tools/memory-edit.ts +59 -0
  978. package/src/tools/memory-recall.ts +100 -0
  979. package/src/tools/memory-reflect.ts +88 -0
  980. package/src/tools/memory-render.ts +202 -0
  981. package/src/tools/memory-retain.ts +91 -0
  982. package/src/tools/output-meta.ts +754 -0
  983. package/src/tools/output-schema-validator.ts +132 -0
  984. package/src/tools/path-utils.ts +1054 -0
  985. package/src/tools/plan-mode-guard.ts +108 -0
  986. package/src/tools/puppeteer/00_stealth_tampering.txt +63 -0
  987. package/src/tools/puppeteer/01_stealth_activity.txt +20 -0
  988. package/src/tools/puppeteer/02_stealth_hairline.txt +11 -0
  989. package/src/tools/puppeteer/03_stealth_botd.txt +384 -0
  990. package/src/tools/puppeteer/04_stealth_iframe.txt +81 -0
  991. package/src/tools/puppeteer/05_stealth_webgl.txt +75 -0
  992. package/src/tools/puppeteer/06_stealth_screen.txt +72 -0
  993. package/src/tools/puppeteer/07_stealth_fonts.txt +97 -0
  994. package/src/tools/puppeteer/08_stealth_audio.txt +51 -0
  995. package/src/tools/puppeteer/09_stealth_locale.txt +46 -0
  996. package/src/tools/puppeteer/10_stealth_plugins.txt +206 -0
  997. package/src/tools/puppeteer/11_stealth_hardware.txt +8 -0
  998. package/src/tools/puppeteer/12_stealth_codecs.txt +40 -0
  999. package/src/tools/puppeteer/13_stealth_worker.txt +74 -0
  1000. package/src/tools/read.ts +2929 -0
  1001. package/src/tools/render-mermaid.ts +69 -0
  1002. package/src/tools/render-utils.ts +838 -0
  1003. package/src/tools/renderers.ts +77 -0
  1004. package/src/tools/report-tool-issue.ts +534 -0
  1005. package/src/tools/resolve.ts +276 -0
  1006. package/src/tools/review.ts +253 -0
  1007. package/src/tools/search-tool-bm25.ts +351 -0
  1008. package/src/tools/search.ts +1580 -0
  1009. package/src/tools/sqlite-reader.ts +828 -0
  1010. package/src/tools/ssh.ts +349 -0
  1011. package/src/tools/todo.ts +982 -0
  1012. package/src/tools/tool-errors.ts +62 -0
  1013. package/src/tools/tool-result.ts +94 -0
  1014. package/src/tools/tool-timeouts.ts +30 -0
  1015. package/src/tools/tts.ts +133 -0
  1016. package/src/tools/write.ts +1217 -0
  1017. package/src/tools/yield.ts +269 -0
  1018. package/src/tui/code-cell.ts +216 -0
  1019. package/src/tui/file-list.ts +55 -0
  1020. package/src/tui/hyperlink.ts +175 -0
  1021. package/src/tui/index.ts +12 -0
  1022. package/src/tui/output-block.ts +240 -0
  1023. package/src/tui/status-line.ts +54 -0
  1024. package/src/tui/tree-list.ts +84 -0
  1025. package/src/tui/types.ts +15 -0
  1026. package/src/tui/utils.ts +103 -0
  1027. package/src/utils/block-context.ts +312 -0
  1028. package/src/utils/changelog.ts +132 -0
  1029. package/src/utils/clipboard.ts +193 -0
  1030. package/src/utils/command-args.ts +76 -0
  1031. package/src/utils/commit-message-generator.ts +151 -0
  1032. package/src/utils/edit-mode.ts +41 -0
  1033. package/src/utils/enhanced-paste.ts +230 -0
  1034. package/src/utils/event-bus.ts +33 -0
  1035. package/src/utils/external-editor.ts +65 -0
  1036. package/src/utils/file-display-mode.ts +45 -0
  1037. package/src/utils/file-mentions.ts +281 -0
  1038. package/src/utils/git.ts +1833 -0
  1039. package/src/utils/image-loading.ts +132 -0
  1040. package/src/utils/image-resize.ts +309 -0
  1041. package/src/utils/jj.ts +248 -0
  1042. package/src/utils/lang-from-path.ts +239 -0
  1043. package/src/utils/markit.ts +89 -0
  1044. package/src/utils/open.ts +55 -0
  1045. package/src/utils/session-color.ts +68 -0
  1046. package/src/utils/shell-snapshot.ts +187 -0
  1047. package/src/utils/sixel.ts +69 -0
  1048. package/src/utils/title-generator.ts +373 -0
  1049. package/src/utils/tool-choice.ts +33 -0
  1050. package/src/utils/tools-manager.ts +363 -0
  1051. package/src/web/kagi.ts +305 -0
  1052. package/src/web/parallel.ts +353 -0
  1053. package/src/web/scrapers/artifacthub.ts +207 -0
  1054. package/src/web/scrapers/arxiv.ts +83 -0
  1055. package/src/web/scrapers/aur.ts +162 -0
  1056. package/src/web/scrapers/biorxiv.ts +133 -0
  1057. package/src/web/scrapers/bluesky.ts +262 -0
  1058. package/src/web/scrapers/brew.ts +172 -0
  1059. package/src/web/scrapers/cheatsh.ts +68 -0
  1060. package/src/web/scrapers/chocolatey.ts +196 -0
  1061. package/src/web/scrapers/choosealicense.ts +95 -0
  1062. package/src/web/scrapers/cisa-kev.ts +87 -0
  1063. package/src/web/scrapers/clojars.ts +154 -0
  1064. package/src/web/scrapers/coingecko.ts +177 -0
  1065. package/src/web/scrapers/crates-io.ts +97 -0
  1066. package/src/web/scrapers/crossref.ts +136 -0
  1067. package/src/web/scrapers/devto.ts +147 -0
  1068. package/src/web/scrapers/discogs.ts +306 -0
  1069. package/src/web/scrapers/discourse.ts +197 -0
  1070. package/src/web/scrapers/dockerhub.ts +138 -0
  1071. package/src/web/scrapers/docs-rs.ts +653 -0
  1072. package/src/web/scrapers/fdroid.ts +134 -0
  1073. package/src/web/scrapers/firefox-addons.ts +191 -0
  1074. package/src/web/scrapers/flathub.ts +223 -0
  1075. package/src/web/scrapers/github-gist.ts +58 -0
  1076. package/src/web/scrapers/github.ts +704 -0
  1077. package/src/web/scrapers/gitlab.ts +401 -0
  1078. package/src/web/scrapers/go-pkg.ts +266 -0
  1079. package/src/web/scrapers/hackage.ts +140 -0
  1080. package/src/web/scrapers/hackernews.ts +189 -0
  1081. package/src/web/scrapers/hex.ts +105 -0
  1082. package/src/web/scrapers/huggingface.ts +321 -0
  1083. package/src/web/scrapers/iacr.ts +89 -0
  1084. package/src/web/scrapers/index.ts +252 -0
  1085. package/src/web/scrapers/jetbrains-marketplace.ts +159 -0
  1086. package/src/web/scrapers/lemmy.ts +203 -0
  1087. package/src/web/scrapers/lobsters.ts +175 -0
  1088. package/src/web/scrapers/mastodon.ts +292 -0
  1089. package/src/web/scrapers/maven.ts +138 -0
  1090. package/src/web/scrapers/mdn.ts +173 -0
  1091. package/src/web/scrapers/metacpan.ts +222 -0
  1092. package/src/web/scrapers/musicbrainz.ts +250 -0
  1093. package/src/web/scrapers/npm.ts +98 -0
  1094. package/src/web/scrapers/nuget.ts +183 -0
  1095. package/src/web/scrapers/nvd.ts +222 -0
  1096. package/src/web/scrapers/ollama.ts +239 -0
  1097. package/src/web/scrapers/open-vsx.ts +106 -0
  1098. package/src/web/scrapers/opencorporates.ts +292 -0
  1099. package/src/web/scrapers/openlibrary.ts +336 -0
  1100. package/src/web/scrapers/orcid.ts +286 -0
  1101. package/src/web/scrapers/osv.ts +176 -0
  1102. package/src/web/scrapers/packagist.ts +160 -0
  1103. package/src/web/scrapers/pub-dev.ts +143 -0
  1104. package/src/web/scrapers/pubmed.ts +211 -0
  1105. package/src/web/scrapers/pypi.ts +112 -0
  1106. package/src/web/scrapers/rawg.ts +110 -0
  1107. package/src/web/scrapers/readthedocs.ts +120 -0
  1108. package/src/web/scrapers/reddit.ts +95 -0
  1109. package/src/web/scrapers/repology.ts +251 -0
  1110. package/src/web/scrapers/rfc.ts +201 -0
  1111. package/src/web/scrapers/rubygems.ts +103 -0
  1112. package/src/web/scrapers/searchcode.ts +189 -0
  1113. package/src/web/scrapers/sec-edgar.ts +261 -0
  1114. package/src/web/scrapers/semantic-scholar.ts +171 -0
  1115. package/src/web/scrapers/snapcraft.ts +187 -0
  1116. package/src/web/scrapers/sourcegraph.ts +336 -0
  1117. package/src/web/scrapers/spdx.ts +108 -0
  1118. package/src/web/scrapers/spotify.ts +198 -0
  1119. package/src/web/scrapers/stackoverflow.ts +120 -0
  1120. package/src/web/scrapers/terraform.ts +277 -0
  1121. package/src/web/scrapers/tldr.ts +47 -0
  1122. package/src/web/scrapers/twitter.ts +94 -0
  1123. package/src/web/scrapers/types.ts +397 -0
  1124. package/src/web/scrapers/utils.ts +109 -0
  1125. package/src/web/scrapers/vimeo.ts +133 -0
  1126. package/src/web/scrapers/vscode-marketplace.ts +187 -0
  1127. package/src/web/scrapers/w3c.ts +156 -0
  1128. package/src/web/scrapers/wikidata.ts +344 -0
  1129. package/src/web/scrapers/wikipedia.ts +84 -0
  1130. package/src/web/scrapers/youtube.ts +325 -0
  1131. package/src/web/search/index.ts +292 -0
  1132. package/src/web/search/provider.ts +157 -0
  1133. package/src/web/search/providers/anthropic.ts +318 -0
  1134. package/src/web/search/providers/base.ts +89 -0
  1135. package/src/web/search/providers/brave.ts +152 -0
  1136. package/src/web/search/providers/codex.ts +591 -0
  1137. package/src/web/search/providers/exa.ts +400 -0
  1138. package/src/web/search/providers/gemini.ts +460 -0
  1139. package/src/web/search/providers/jina.ts +111 -0
  1140. package/src/web/search/providers/kagi.ts +86 -0
  1141. package/src/web/search/providers/kimi.ts +196 -0
  1142. package/src/web/search/providers/parallel.ts +225 -0
  1143. package/src/web/search/providers/perplexity.ts +730 -0
  1144. package/src/web/search/providers/searxng.ts +313 -0
  1145. package/src/web/search/providers/synthetic.ts +114 -0
  1146. package/src/web/search/providers/tavily.ts +176 -0
  1147. package/src/web/search/providers/utils.ts +128 -0
  1148. package/src/web/search/providers/zai.ts +333 -0
  1149. package/src/web/search/render.ts +262 -0
  1150. package/src/web/search/types.ts +482 -0
  1151. package/src/web/search/utils.ts +17 -0
  1152. package/src/workspace-tree.ts +286 -0
@@ -0,0 +1,1891 @@
1
+ /**
2
+ * Patch application logic for the edit tool.
3
+ *
4
+ * Applies parsed diff hunks to file content using fuzzy matching
5
+ * for robust handling of whitespace and formatting differences.
6
+ */
7
+
8
+ import * as fs from "node:fs";
9
+ import * as path from "node:path";
10
+ import type { AgentToolResult } from "@oh-my-pi/pi-agent-core";
11
+ import { isEnoent } from "@oh-my-pi/pi-utils";
12
+ import * as z from "zod/v4";
13
+ import {
14
+ type FileDiagnosticsResult,
15
+ flushLspWritethroughBatch,
16
+ type WritethroughCallback,
17
+ type WritethroughDeferredHandle,
18
+ } from "../../lsp";
19
+ import type { ToolSession } from "../../tools";
20
+ import { assertEditableFile } from "../../tools/auto-generated-guard";
21
+ import {
22
+ invalidateFsScanAfterDelete,
23
+ invalidateFsScanAfterRename,
24
+ invalidateFsScanAfterWrite,
25
+ } from "../../tools/fs-cache-invalidation";
26
+ import { outputMeta } from "../../tools/output-meta";
27
+ import { resolveToCwd } from "../../tools/path-utils";
28
+ import { enforcePlanModeWrite, resolvePlanPath } from "../../tools/plan-mode-guard";
29
+ import { ToolError } from "../../tools/tool-errors";
30
+ import {
31
+ ApplyPatchError,
32
+ type DiffHunk,
33
+ generateUnifiedDiffString,
34
+ normalizeCreateContent,
35
+ parseDiffHunks,
36
+ } from "../diff";
37
+ import {
38
+ adjustIndentation,
39
+ convertLeadingTabsToSpaces,
40
+ countLeadingWhitespace,
41
+ detectLineEnding,
42
+ getLeadingWhitespace,
43
+ normalizeForFuzzy,
44
+ normalizeToLF,
45
+ restoreLineEndings,
46
+ stripBom,
47
+ } from "../normalize";
48
+ import { readEditFileText, serializeEditFileText } from "../read-file";
49
+ import type { EditToolDetails, LspBatchRequest } from "../renderer";
50
+ import {
51
+ type ContextLineResult,
52
+ DEFAULT_FUZZY_THRESHOLD,
53
+ findClosestSequenceMatch,
54
+ findContextLine,
55
+ findMatch,
56
+ type SequenceSearchResult,
57
+ seekSequence,
58
+ } from "./replace";
59
+
60
+ export type Operation = "create" | "delete" | "update";
61
+
62
+ export interface PatchInput {
63
+ path: string;
64
+ op: Operation;
65
+ rename?: string;
66
+ diff?: string;
67
+ }
68
+
69
+ export interface FileSystem {
70
+ exists(path: string): Promise<boolean>;
71
+ read(path: string): Promise<string>;
72
+ readBinary?: (path: string) => Promise<Uint8Array>;
73
+ write(path: string, content: string): Promise<void>;
74
+ delete(path: string): Promise<void>;
75
+ mkdir(path: string): Promise<void>;
76
+ }
77
+
78
+ interface FileChange {
79
+ type: Operation;
80
+ path: string;
81
+ newPath?: string;
82
+ oldContent?: string;
83
+ newContent?: string;
84
+ }
85
+
86
+ export interface ApplyPatchResult {
87
+ change: FileChange;
88
+ warnings?: string[];
89
+ }
90
+
91
+ export interface ApplyPatchOptions {
92
+ cwd: string;
93
+ dryRun?: boolean;
94
+ fuzzyThreshold?: number;
95
+ allowFuzzy?: boolean;
96
+ fs?: FileSystem;
97
+ }
98
+
99
+ // ═══════════════════════════════════════════════════════════════════════════
100
+ // Default File System
101
+ // ═══════════════════════════════════════════════════════════════════════════
102
+
103
+ /** Default filesystem implementation using Bun APIs */
104
+ export const defaultFileSystem: FileSystem = {
105
+ async exists(path: string): Promise<boolean> {
106
+ return Bun.file(path).exists();
107
+ },
108
+ async read(path: string): Promise<string> {
109
+ return readEditFileText(path, path);
110
+ },
111
+ async readBinary(path: string): Promise<Uint8Array> {
112
+ return fs.promises.readFile(path);
113
+ },
114
+ async write(path: string, content: string): Promise<void> {
115
+ await Bun.write(path, await serializeEditFileText(path, path, content));
116
+ },
117
+ async delete(path: string): Promise<void> {
118
+ await fs.promises.unlink(path);
119
+ },
120
+ async mkdir(path: string): Promise<void> {
121
+ await fs.promises.mkdir(path, { recursive: true });
122
+ },
123
+ };
124
+
125
+ // ═══════════════════════════════════════════════════════════════════════════
126
+ // Internal Types
127
+ // ═══════════════════════════════════════════════════════════════════════════
128
+
129
+ interface Replacement {
130
+ startIndex: number;
131
+ oldLen: number;
132
+ newLines: string[];
133
+ }
134
+
135
+ type HunkVariantKind = "trim-common" | "dedupe-shared" | "collapse-repeated" | "single-line";
136
+
137
+ interface HunkVariant {
138
+ oldLines: string[];
139
+ newLines: string[];
140
+ kind: HunkVariantKind;
141
+ }
142
+
143
+ function isBlankLine(line: string): boolean {
144
+ return line.trim().length === 0;
145
+ }
146
+
147
+ function areEqualLines(left: string[], right: string[]): boolean {
148
+ if (left.length !== right.length) return false;
149
+ for (let i = 0; i < left.length; i++) {
150
+ if (left[i] !== right[i]) return false;
151
+ }
152
+ return true;
153
+ }
154
+
155
+ function areEqualTrimmedLines(left: string[], right: string[]): boolean {
156
+ if (left.length !== right.length) return false;
157
+ for (let i = 0; i < left.length; i++) {
158
+ if (left[i].trim() !== right[i].trim()) return false;
159
+ }
160
+ return true;
161
+ }
162
+
163
+ function getIndentChar(lines: string[]): string {
164
+ for (const line of lines) {
165
+ const ws = getLeadingWhitespace(line);
166
+ if (ws.length > 0) return ws[0];
167
+ }
168
+ return " ";
169
+ }
170
+
171
+ function collectIndentDeltas(oldLines: string[], actualLines: string[]): number[] {
172
+ const deltas: number[] = [];
173
+ const lineCount = Math.min(oldLines.length, actualLines.length);
174
+ for (let i = 0; i < lineCount; i++) {
175
+ const oldLine = oldLines[i];
176
+ const actualLine = actualLines[i];
177
+ if (isBlankLine(oldLine) || isBlankLine(actualLine)) continue;
178
+ deltas.push(countLeadingWhitespace(actualLine) - countLeadingWhitespace(oldLine));
179
+ }
180
+ return deltas;
181
+ }
182
+
183
+ function applyIndentDelta(lines: string[], delta: number, indentChar: string): string[] {
184
+ return lines.map(line => {
185
+ if (isBlankLine(line)) return line;
186
+ if (delta > 0) return indentChar.repeat(delta) + line;
187
+ const toRemove = Math.min(-delta, countLeadingWhitespace(line));
188
+ return line.slice(toRemove);
189
+ });
190
+ }
191
+
192
+ function canConvertTabsToSpaces(oldLines: string[], actualLines: string[], spacesPerTab: number): boolean {
193
+ const lineCount = Math.min(oldLines.length, actualLines.length);
194
+ for (let i = 0; i < lineCount; i++) {
195
+ const oldLine = oldLines[i];
196
+ const actualLine = actualLines[i];
197
+ if (isBlankLine(oldLine) || isBlankLine(actualLine)) continue;
198
+ const oldIndent = getLeadingWhitespace(oldLine);
199
+ const actualIndent = getLeadingWhitespace(actualLine);
200
+ if (oldIndent.length === 0) continue;
201
+ if (actualIndent.length !== oldIndent.length * spacesPerTab) {
202
+ return false;
203
+ }
204
+ }
205
+ return true;
206
+ }
207
+
208
+ // ═══════════════════════════════════════════════════════════════════════════
209
+ // Replacement Computation
210
+ // ═══════════════════════════════════════════════════════════════════════════
211
+
212
+ /** Adjust indentation of newLines to match the delta between patternLines and actualLines */
213
+ function adjustLinesIndentation(patternLines: string[], actualLines: string[], newLines: string[]): string[] {
214
+ if (patternLines.length === 0 || actualLines.length === 0 || newLines.length === 0) {
215
+ return newLines;
216
+ }
217
+
218
+ // If pattern already matches actual exactly (including indentation), preserve agent's intended changes
219
+ if (areEqualLines(patternLines, actualLines)) {
220
+ return newLines;
221
+ }
222
+
223
+ // If the patch is purely an indentation change (same trimmed content), apply exactly as specified
224
+ if (areEqualTrimmedLines(patternLines, newLines)) {
225
+ return newLines;
226
+ }
227
+
228
+ // Detect indent character from actual content
229
+ const indentChar = getIndentChar(actualLines);
230
+
231
+ let patternTabOnly = true;
232
+ let actualSpaceOnly = true;
233
+ let patternSpaceOnly = true;
234
+ let actualTabOnly = true;
235
+ let patternMixed = false;
236
+ let actualMixed = false;
237
+
238
+ for (const line of patternLines) {
239
+ if (line.trim().length === 0) continue;
240
+ const ws = getLeadingWhitespace(line);
241
+ if (ws.includes(" ")) patternTabOnly = false;
242
+ if (ws.includes("\t")) patternSpaceOnly = false;
243
+ if (ws.includes(" ") && ws.includes("\t")) patternMixed = true;
244
+ }
245
+
246
+ for (const line of actualLines) {
247
+ if (line.trim().length === 0) continue;
248
+ const ws = getLeadingWhitespace(line);
249
+ if (ws.includes("\t")) actualSpaceOnly = false;
250
+ if (ws.includes(" ")) actualTabOnly = false;
251
+ if (ws.includes(" ") && ws.includes("\t")) actualMixed = true;
252
+ }
253
+
254
+ if (!patternMixed && !actualMixed && patternTabOnly && actualSpaceOnly) {
255
+ let ratio: number | undefined;
256
+ const lineCount = Math.min(patternLines.length, actualLines.length);
257
+ let consistent = true;
258
+ for (let i = 0; i < lineCount; i++) {
259
+ const patternLine = patternLines[i];
260
+ const actualLine = actualLines[i];
261
+ if (patternLine.trim().length === 0 || actualLine.trim().length === 0) continue;
262
+ const patternIndent = countLeadingWhitespace(patternLine);
263
+ const actualIndent = countLeadingWhitespace(actualLine);
264
+ if (patternIndent === 0) continue;
265
+ if (actualIndent % patternIndent !== 0) {
266
+ consistent = false;
267
+ break;
268
+ }
269
+ const nextRatio = actualIndent / patternIndent;
270
+ if (!ratio) {
271
+ ratio = nextRatio;
272
+ } else if (ratio !== nextRatio) {
273
+ consistent = false;
274
+ break;
275
+ }
276
+ }
277
+
278
+ if (consistent && ratio && canConvertTabsToSpaces(patternLines, actualLines, ratio)) {
279
+ return convertLeadingTabsToSpaces(newLines.join("\n"), ratio).split("\n");
280
+ }
281
+ }
282
+
283
+ // Reverse: pattern uses spaces, actual uses tabs — infer spaces = tabs * width + offset
284
+ // Collect (tabs, spaces) pairs from matched lines to solve for the model's tab rendering.
285
+ // With one data point: spaces = tabs * width (offset=0).
286
+ // With two+: solve ax + b via pairs with distinct tab counts.
287
+ if (!patternMixed && !actualMixed && patternSpaceOnly && actualTabOnly) {
288
+ const samples = new Map<number, number>(); // tabs -> spaces
289
+ const lineCount = Math.min(patternLines.length, actualLines.length);
290
+ let consistent = true;
291
+ for (let i = 0; i < lineCount; i++) {
292
+ const patternLine = patternLines[i];
293
+ const actualLine = actualLines[i];
294
+ if (patternLine.trim().length === 0 || actualLine.trim().length === 0) continue;
295
+ const spaces = countLeadingWhitespace(patternLine);
296
+ const tabs = countLeadingWhitespace(actualLine);
297
+ if (tabs === 0) continue;
298
+ const existing = samples.get(tabs);
299
+ if (existing !== undefined && existing !== spaces) {
300
+ consistent = false;
301
+ break;
302
+ }
303
+ samples.set(tabs, spaces);
304
+ }
305
+
306
+ if (consistent && samples.size > 0) {
307
+ let tabWidth: number | undefined;
308
+ let offset = 0;
309
+
310
+ if (samples.size === 1) {
311
+ // One level: assume offset=0, width = spaces / tabs
312
+ const [[tabs, spaces]] = samples;
313
+ if (spaces % tabs === 0) {
314
+ tabWidth = spaces / tabs;
315
+ }
316
+ } else {
317
+ // Two+ levels: solve via any two distinct pairs
318
+ // spaces = tabs * width + offset => width = (s2 - s1) / (t2 - t1)
319
+ const entries = [...samples.entries()];
320
+ const [t1, s1] = entries[0];
321
+ const [t2, s2] = entries[1];
322
+ if (t1 !== t2) {
323
+ const w = (s2 - s1) / (t2 - t1);
324
+ if (w > 0 && Number.isInteger(w)) {
325
+ const b = s1 - t1 * w;
326
+ // Validate all samples against this model
327
+ let valid = true;
328
+ for (const [t, s] of samples) {
329
+ if (t * w + b !== s) {
330
+ valid = false;
331
+ break;
332
+ }
333
+ }
334
+ if (valid) {
335
+ tabWidth = w;
336
+ offset = b;
337
+ }
338
+ }
339
+ }
340
+ }
341
+
342
+ if (tabWidth !== undefined && tabWidth > 0) {
343
+ const converted = newLines.map(line => {
344
+ if (line.trim().length === 0) return line;
345
+ const ws = countLeadingWhitespace(line);
346
+ if (ws === 0) return line;
347
+ // Reverse: tabs = (spaces - offset) / width
348
+ const adjusted = ws - offset;
349
+ if (adjusted >= 0 && adjusted % tabWidth! === 0) {
350
+ return "\t".repeat(adjusted / tabWidth!) + line.slice(ws);
351
+ }
352
+ // Partial tab — keep remainder as spaces
353
+ const tabCount = Math.floor(adjusted / tabWidth!);
354
+ const remainder = adjusted - tabCount * tabWidth!;
355
+ if (tabCount >= 0) {
356
+ return "\t".repeat(tabCount) + " ".repeat(remainder) + line.slice(ws);
357
+ }
358
+ return line;
359
+ });
360
+ return converted;
361
+ }
362
+ }
363
+ }
364
+
365
+ // Build a map from trimmed content to actual lines (by content, not position)
366
+ // This handles fuzzy matches where pattern and actual may not be positionally aligned
367
+ const contentToActualLines = new Map<string, string[]>();
368
+ for (const line of actualLines) {
369
+ const trimmed = line.trim();
370
+ if (trimmed.length === 0) continue;
371
+ const arr = contentToActualLines.get(trimmed);
372
+ if (arr) {
373
+ arr.push(line);
374
+ } else {
375
+ contentToActualLines.set(trimmed, [line]);
376
+ }
377
+ }
378
+
379
+ let patternMin = Infinity;
380
+ for (const line of patternLines) {
381
+ if (line.trim().length === 0) continue;
382
+ patternMin = Math.min(patternMin, countLeadingWhitespace(line));
383
+ }
384
+ if (patternMin === Infinity) {
385
+ patternMin = 0;
386
+ }
387
+
388
+ const deltas = collectIndentDeltas(patternLines, actualLines);
389
+ const delta = deltas.length > 0 && deltas.every(value => value === deltas[0]) ? deltas[0] : undefined;
390
+
391
+ // Track which actual lines we've used to handle duplicate content correctly
392
+ const usedActualLines = new Map<string, number>(); // trimmed content -> count used
393
+
394
+ return newLines.map(newLine => {
395
+ if (newLine.trim().length === 0) {
396
+ return newLine;
397
+ }
398
+
399
+ const trimmed = newLine.trim();
400
+ const matchingActualLines = contentToActualLines.get(trimmed);
401
+
402
+ // Check if this is a context line (same trimmed content exists in actual)
403
+ if (matchingActualLines && matchingActualLines.length > 0) {
404
+ if (matchingActualLines.length === 1) {
405
+ return matchingActualLines[0];
406
+ }
407
+ if (matchingActualLines.includes(newLine)) {
408
+ return newLine;
409
+ }
410
+ const usedCount = usedActualLines.get(trimmed) ?? 0;
411
+ if (usedCount < matchingActualLines.length) {
412
+ usedActualLines.set(trimmed, usedCount + 1);
413
+ // Use actual file content directly for context lines
414
+ return matchingActualLines[usedCount];
415
+ }
416
+ }
417
+
418
+ // This is a new/added line - apply consistent delta if safe
419
+ if (delta && delta !== 0) {
420
+ const newIndent = countLeadingWhitespace(newLine);
421
+ if (newIndent === patternMin) {
422
+ return applyIndentDelta([newLine], delta, indentChar)[0];
423
+ }
424
+ }
425
+ return newLine;
426
+ });
427
+ }
428
+
429
+ function trimCommonContext(oldLines: string[], newLines: string[]): HunkVariant | undefined {
430
+ let start = 0;
431
+ let endOld = oldLines.length;
432
+ let endNew = newLines.length;
433
+
434
+ while (start < endOld && start < endNew && oldLines[start] === newLines[start]) {
435
+ start++;
436
+ }
437
+
438
+ while (endOld > start && endNew > start && oldLines[endOld - 1] === newLines[endNew - 1]) {
439
+ endOld--;
440
+ endNew--;
441
+ }
442
+
443
+ if (start === 0 && endOld === oldLines.length && endNew === newLines.length) {
444
+ return undefined;
445
+ }
446
+
447
+ const trimmedOld = oldLines.slice(start, endOld);
448
+ const trimmedNew = newLines.slice(start, endNew);
449
+ if (trimmedOld.length === 0 && trimmedNew.length === 0) {
450
+ return undefined;
451
+ }
452
+ return { oldLines: trimmedOld, newLines: trimmedNew, kind: "trim-common" };
453
+ }
454
+
455
+ function collapseConsecutiveSharedLines(oldLines: string[], newLines: string[]): HunkVariant | undefined {
456
+ const shared = new Set(oldLines.filter(line => newLines.includes(line)));
457
+ const collapse = (lines: string[]): string[] => {
458
+ const out: string[] = [];
459
+ let i = 0;
460
+ while (i < lines.length) {
461
+ const line = lines[i];
462
+ out.push(line);
463
+ let j = i + 1;
464
+ while (j < lines.length && lines[j] === line && shared.has(line)) {
465
+ j++;
466
+ }
467
+ i = j;
468
+ }
469
+ return out;
470
+ };
471
+
472
+ const collapsedOld = collapse(oldLines);
473
+ const collapsedNew = collapse(newLines);
474
+ if (collapsedOld.length === oldLines.length && collapsedNew.length === newLines.length) {
475
+ return undefined;
476
+ }
477
+ return { oldLines: collapsedOld, newLines: collapsedNew, kind: "dedupe-shared" };
478
+ }
479
+
480
+ function collapseRepeatedBlocks(oldLines: string[], newLines: string[]): HunkVariant | undefined {
481
+ const shared = new Set(oldLines.filter(line => newLines.includes(line)));
482
+ const collapse = (lines: string[]): string[] => {
483
+ const output = [...lines];
484
+ let changed = false;
485
+ let i = 0;
486
+ while (i < output.length) {
487
+ let collapsed = false;
488
+ for (let size = Math.floor((output.length - i) / 2); size >= 2; size--) {
489
+ const first = output.slice(i, i + size);
490
+ const second = output.slice(i + size, i + size * 2);
491
+ if (first.length !== second.length || first.length === 0) continue;
492
+ if (!first.every(line => shared.has(line))) continue;
493
+ let same = true;
494
+ for (let idx = 0; idx < size; idx++) {
495
+ if (first[idx] !== second[idx]) {
496
+ same = false;
497
+ break;
498
+ }
499
+ }
500
+ if (same) {
501
+ output.splice(i + size, size);
502
+ changed = true;
503
+ collapsed = true;
504
+ break;
505
+ }
506
+ }
507
+ if (!collapsed) {
508
+ i++;
509
+ }
510
+ }
511
+ return changed ? output : lines;
512
+ };
513
+
514
+ const collapsedOld = collapse(oldLines);
515
+ const collapsedNew = collapse(newLines);
516
+ if (collapsedOld.length === oldLines.length && collapsedNew.length === newLines.length) {
517
+ return undefined;
518
+ }
519
+ return { oldLines: collapsedOld, newLines: collapsedNew, kind: "collapse-repeated" };
520
+ }
521
+
522
+ function reduceToSingleLineChange(oldLines: string[], newLines: string[]): HunkVariant | undefined {
523
+ if (oldLines.length !== newLines.length || oldLines.length === 0) return undefined;
524
+ let changedIndex: number | undefined;
525
+ for (let i = 0; i < oldLines.length; i++) {
526
+ if (oldLines[i] !== newLines[i]) {
527
+ if (changedIndex !== undefined) return undefined;
528
+ changedIndex = i;
529
+ }
530
+ }
531
+ if (changedIndex === undefined) return undefined;
532
+ return { oldLines: [oldLines[changedIndex]], newLines: [newLines[changedIndex]], kind: "single-line" };
533
+ }
534
+
535
+ function buildFallbackVariants(hunk: DiffHunk): HunkVariant[] {
536
+ const variants: HunkVariant[] = [];
537
+ const base: HunkVariant = { oldLines: hunk.oldLines, newLines: hunk.newLines, kind: "trim-common" };
538
+
539
+ const trimmed = trimCommonContext(base.oldLines, base.newLines);
540
+ if (trimmed) variants.push(trimmed);
541
+
542
+ const deduped = collapseConsecutiveSharedLines(
543
+ trimmed?.oldLines ?? base.oldLines,
544
+ trimmed?.newLines ?? base.newLines,
545
+ );
546
+ if (deduped) variants.push(deduped);
547
+
548
+ const collapsed = collapseRepeatedBlocks(
549
+ deduped?.oldLines ?? trimmed?.oldLines ?? base.oldLines,
550
+ deduped?.newLines ?? trimmed?.newLines ?? base.newLines,
551
+ );
552
+ if (collapsed) variants.push(collapsed);
553
+
554
+ const singleLine = reduceToSingleLineChange(trimmed?.oldLines ?? base.oldLines, trimmed?.newLines ?? base.newLines);
555
+ if (singleLine) variants.push(singleLine);
556
+
557
+ const seen = new Set<string>();
558
+ return variants.filter(variant => {
559
+ if (variant.oldLines.length === 0 && variant.newLines.length === 0) return false;
560
+ const key = `${variant.oldLines.join("\n")}||${variant.newLines.join("\n")}`;
561
+ if (seen.has(key)) return false;
562
+ seen.add(key);
563
+ return true;
564
+ });
565
+ }
566
+
567
+ function filterFallbackVariants(variants: HunkVariant[], allowAggressive: boolean): HunkVariant[] {
568
+ if (allowAggressive) return variants;
569
+ return variants.filter(variant => variant.kind !== "collapse-repeated" && variant.kind !== "single-line");
570
+ }
571
+
572
+ function findContextRelativeMatch(
573
+ lines: string[],
574
+ patternLine: string,
575
+ contextIndex: number,
576
+ preferSecondForwardMatch: boolean,
577
+ ): number | undefined {
578
+ const trimmed = patternLine.trim();
579
+ const forwardMatches: number[] = [];
580
+ for (let i = contextIndex + 1; i < lines.length; i++) {
581
+ if (lines[i].trim() === trimmed) {
582
+ forwardMatches.push(i);
583
+ }
584
+ }
585
+ if (forwardMatches.length > 0) {
586
+ if (preferSecondForwardMatch && forwardMatches.length > 1) {
587
+ return forwardMatches[1];
588
+ }
589
+ return forwardMatches[0];
590
+ }
591
+ for (let i = contextIndex - 1; i >= 0; i--) {
592
+ if (lines[i].trim() === trimmed) {
593
+ return i;
594
+ }
595
+ }
596
+ return undefined;
597
+ }
598
+
599
+ const AMBIGUITY_HINT_WINDOW = 200;
600
+ const MATCH_PREVIEW_CONTEXT = 2;
601
+ const MATCH_PREVIEW_MAX_LEN = 80;
602
+
603
+ function formatSequenceMatchPreview(lines: string[], startIdx: number): string {
604
+ const start = Math.max(0, startIdx - MATCH_PREVIEW_CONTEXT);
605
+ const end = Math.min(lines.length, startIdx + MATCH_PREVIEW_CONTEXT + 1);
606
+ const previewLines = lines.slice(start, end);
607
+ return previewLines
608
+ .map((line, i) => {
609
+ const num = start + i + 1;
610
+ const truncated = line.length > MATCH_PREVIEW_MAX_LEN ? `${line.slice(0, MATCH_PREVIEW_MAX_LEN - 1)}…` : line;
611
+ return ` ${num} | ${truncated}`;
612
+ })
613
+ .join("\n");
614
+ }
615
+
616
+ function formatSequenceMatchPreviews(
617
+ lines: string[],
618
+ matchIndices: number[] | undefined,
619
+ matchCount: number | undefined,
620
+ ): string | undefined {
621
+ if (!matchIndices || matchIndices.length === 0) return undefined;
622
+ const previews = matchIndices.map(index => formatSequenceMatchPreview(lines, index));
623
+ const moreMsg =
624
+ matchCount && matchCount > matchIndices.length ? ` (showing first ${matchIndices.length} of ${matchCount})` : "";
625
+ return `${previews.join("\n\n")}${moreMsg}`;
626
+ }
627
+
628
+ function chooseHintedMatch(
629
+ matchIndices: number[] | undefined,
630
+ hintIndex: number | undefined,
631
+ window: number,
632
+ ): number | undefined {
633
+ if (!matchIndices || matchIndices.length === 0 || hintIndex === undefined) return undefined;
634
+ const candidates = matchIndices.filter(index => Math.abs(index - hintIndex) <= window);
635
+ if (candidates.length === 1) return candidates[0];
636
+ return undefined;
637
+ }
638
+
639
+ /** Get hint index from hunk's line number */
640
+ function getHunkHintIndex(hunk: DiffHunk, currentIndex: number): number | undefined {
641
+ if (hunk.oldStartLine === undefined) return undefined;
642
+ const hintIndex = Math.max(0, hunk.oldStartLine - 1);
643
+ return hintIndex >= currentIndex ? hintIndex : undefined;
644
+ }
645
+
646
+ /**
647
+ * Find hierarchical context in file lines.
648
+ *
649
+ * Handles three formats:
650
+ * 1. Simple context: "function foo" - find this line
651
+ * 2. Hierarchical (newline): "class Foo\nmethod" - find class, then method after it
652
+ * 3. Hierarchical (space): "class Foo method" - try as literal first, then split and search
653
+ *
654
+ * @returns The result from finding the final (innermost) context, or undefined if not found
655
+ */
656
+ function findHierarchicalContext(
657
+ lines: string[],
658
+ context: string,
659
+ startFrom: number,
660
+ lineHint: number | undefined,
661
+ allowFuzzy: boolean,
662
+ ): ContextLineResult {
663
+ // Check for newline-separated hierarchical contexts (from nested @@ anchors)
664
+ if (context.includes("\n")) {
665
+ const parts = context
666
+ .split("\n")
667
+ .map(p => p.trim())
668
+ .filter(p => p.length > 0);
669
+ let currentStart = startFrom;
670
+
671
+ for (let i = 0; i < parts.length; i++) {
672
+ const part = parts[i];
673
+ const isLast = i === parts.length - 1;
674
+
675
+ const result = findContextLine(lines, part, currentStart, { allowFuzzy });
676
+
677
+ if (result.matchCount !== undefined && result.matchCount > 1) {
678
+ if (isLast && lineHint !== undefined) {
679
+ const hintStart = Math.max(0, lineHint - 1);
680
+ if (hintStart >= currentStart) {
681
+ const hintedResult = findContextLine(lines, part, hintStart, { allowFuzzy });
682
+ if (hintedResult.index !== undefined) {
683
+ return { ...hintedResult, matchCount: 1, matchIndices: [hintedResult.index] };
684
+ }
685
+ }
686
+ }
687
+ return {
688
+ index: undefined,
689
+ confidence: result.confidence,
690
+ matchCount: result.matchCount,
691
+ matchIndices: result.matchIndices,
692
+ strategy: result.strategy,
693
+ };
694
+ }
695
+
696
+ if (result.index === undefined) {
697
+ if (isLast && lineHint !== undefined) {
698
+ const hintStart = Math.max(0, lineHint - 1);
699
+ if (hintStart >= currentStart) {
700
+ const hintedResult = findContextLine(lines, part, hintStart, { allowFuzzy });
701
+ if (hintedResult.index !== undefined) {
702
+ return { ...hintedResult, matchCount: 1, matchIndices: [hintedResult.index] };
703
+ }
704
+ }
705
+ }
706
+ return { index: undefined, confidence: result.confidence };
707
+ }
708
+
709
+ if (isLast) {
710
+ return result;
711
+ }
712
+ currentStart = result.index + 1;
713
+ }
714
+ return { index: undefined, confidence: 0 };
715
+ }
716
+
717
+ // Try literal context first
718
+ const spaceParts = context.split(/\s+/).filter(p => p.length > 0);
719
+ const hasSignatureChars = /[(){}[\]]/.test(context);
720
+ if (!hasSignatureChars && spaceParts.length > 2) {
721
+ const outer = spaceParts.slice(0, -1).join(" ");
722
+ const inner = spaceParts[spaceParts.length - 1];
723
+ const outerResult = findContextLine(lines, outer, startFrom, { allowFuzzy });
724
+ if (outerResult.matchCount !== undefined && outerResult.matchCount > 1) {
725
+ return {
726
+ index: undefined,
727
+ confidence: outerResult.confidence,
728
+ matchCount: outerResult.matchCount,
729
+ matchIndices: outerResult.matchIndices,
730
+ strategy: outerResult.strategy,
731
+ };
732
+ }
733
+ if (outerResult.index !== undefined) {
734
+ const innerResult = findContextLine(lines, inner, outerResult.index + 1, { allowFuzzy });
735
+ if (innerResult.index !== undefined) {
736
+ return innerResult.matchCount && innerResult.matchCount > 1
737
+ ? { ...innerResult, matchCount: 1, matchIndices: [innerResult.index] }
738
+ : innerResult;
739
+ }
740
+ if (innerResult.matchCount !== undefined && innerResult.matchCount > 1) {
741
+ return {
742
+ ...innerResult,
743
+ matchCount: 1,
744
+ matchIndices: innerResult.index !== undefined ? [innerResult.index] : innerResult.matchIndices,
745
+ };
746
+ }
747
+ }
748
+ }
749
+
750
+ const result = findContextLine(lines, context, startFrom, { allowFuzzy });
751
+
752
+ // If line hint exists and result is ambiguous or missing, try from hint
753
+ if ((result.index === undefined || (result.matchCount ?? 0) > 1) && lineHint !== undefined) {
754
+ const hintStart = Math.max(0, lineHint - 1);
755
+ const hintedResult = findContextLine(lines, context, hintStart, { allowFuzzy });
756
+ if (hintedResult.index !== undefined) {
757
+ return { ...hintedResult, matchCount: 1, matchIndices: [hintedResult.index] };
758
+ }
759
+ }
760
+
761
+ // If found uniquely, return it
762
+ if (result.index !== undefined && (result.matchCount ?? 0) <= 1) {
763
+ return result;
764
+ }
765
+ if (result.matchCount !== undefined && result.matchCount > 1) {
766
+ return result;
767
+ }
768
+
769
+ // Try from beginning if not found from current position
770
+ if (result.index === undefined && startFrom !== 0) {
771
+ const fromStartResult = findContextLine(lines, context, 0, { allowFuzzy });
772
+ if (fromStartResult.index !== undefined && (fromStartResult.matchCount ?? 0) <= 1) {
773
+ return fromStartResult;
774
+ }
775
+ if (fromStartResult.matchCount !== undefined && fromStartResult.matchCount > 1) {
776
+ return fromStartResult;
777
+ }
778
+ }
779
+
780
+ // Fallback: try space-separated hierarchical matching
781
+ // e.g., "class PatchTool constructor" -> find "class PatchTool", then "constructor" after it
782
+ if (!hasSignatureChars && spaceParts.length > 1) {
783
+ const outer = spaceParts.slice(0, -1).join(" ");
784
+ const inner = spaceParts[spaceParts.length - 1];
785
+ const outerResult = findContextLine(lines, outer, startFrom, { allowFuzzy });
786
+
787
+ if (outerResult.matchCount !== undefined && outerResult.matchCount > 1) {
788
+ return {
789
+ index: undefined,
790
+ confidence: outerResult.confidence,
791
+ matchCount: outerResult.matchCount,
792
+ matchIndices: outerResult.matchIndices,
793
+ strategy: outerResult.strategy,
794
+ };
795
+ }
796
+
797
+ if (outerResult.index === undefined) {
798
+ return { index: undefined, confidence: outerResult.confidence };
799
+ }
800
+
801
+ const innerResult = findContextLine(lines, inner, outerResult.index + 1, { allowFuzzy });
802
+ if (innerResult.index !== undefined) {
803
+ return innerResult.matchCount && innerResult.matchCount > 1
804
+ ? { ...innerResult, matchCount: 1, matchIndices: [innerResult.index] }
805
+ : innerResult;
806
+ }
807
+ if (innerResult.matchCount !== undefined && innerResult.matchCount > 1) {
808
+ return {
809
+ ...innerResult,
810
+ matchCount: 1,
811
+ matchIndices: innerResult.index !== undefined ? [innerResult.index] : innerResult.matchIndices,
812
+ };
813
+ }
814
+ }
815
+
816
+ return result;
817
+ }
818
+
819
+ /** Find sequence with optional hint position, returning full search result */
820
+ function findSequenceWithHint(
821
+ lines: string[],
822
+ pattern: string[],
823
+ currentIndex: number,
824
+ hintIndex: number | undefined,
825
+ eof: boolean,
826
+ allowFuzzy: boolean,
827
+ ): SequenceSearchResult {
828
+ // Prefer content-based search starting from currentIndex
829
+ const primaryResult = seekSequence(lines, pattern, currentIndex, eof, { allowFuzzy });
830
+ if (
831
+ primaryResult.matchCount &&
832
+ primaryResult.matchCount > 1 &&
833
+ hintIndex !== undefined &&
834
+ hintIndex !== currentIndex
835
+ ) {
836
+ const hintedResult = seekSequence(lines, pattern, hintIndex, eof, { allowFuzzy });
837
+ if (hintedResult.index !== undefined && (hintedResult.matchCount ?? 1) <= 1) {
838
+ return hintedResult;
839
+ }
840
+ if (hintedResult.matchCount && hintedResult.matchCount > 1) {
841
+ return hintedResult;
842
+ }
843
+ }
844
+ if (primaryResult.index !== undefined || (primaryResult.matchCount && primaryResult.matchCount > 1)) {
845
+ return primaryResult;
846
+ }
847
+
848
+ // Use line hint as a secondary bias only if needed
849
+ if (hintIndex !== undefined && hintIndex !== currentIndex) {
850
+ const hintedResult = seekSequence(lines, pattern, hintIndex, eof, { allowFuzzy });
851
+ if (hintedResult.index !== undefined || (hintedResult.matchCount && hintedResult.matchCount > 1)) {
852
+ return hintedResult;
853
+ }
854
+ }
855
+
856
+ // Last resort: search from beginning (handles out-of-order hunks)
857
+ if (currentIndex !== 0) {
858
+ const fromStartResult = seekSequence(lines, pattern, 0, eof, { allowFuzzy });
859
+ if (fromStartResult.index !== undefined || (fromStartResult.matchCount && fromStartResult.matchCount > 1)) {
860
+ return fromStartResult;
861
+ }
862
+ }
863
+
864
+ return primaryResult;
865
+ }
866
+
867
+ function attemptSequenceFallback(
868
+ lines: string[],
869
+ hunk: DiffHunk,
870
+ currentIndex: number,
871
+ lineHint: number | undefined,
872
+ allowFuzzy: boolean,
873
+ allowAggressiveFallbacks: boolean,
874
+ ): number | undefined {
875
+ if (hunk.oldLines.length === 0) return undefined;
876
+ const matchHint = getHunkHintIndex(hunk, currentIndex);
877
+ const fallbackResult = findSequenceWithHint(
878
+ lines,
879
+ hunk.oldLines,
880
+ currentIndex,
881
+ matchHint ?? lineHint,
882
+ false,
883
+ allowFuzzy,
884
+ );
885
+ if (fallbackResult.index !== undefined && (fallbackResult.matchCount ?? 1) <= 1) {
886
+ const nextIndex = fallbackResult.index + 1;
887
+ if (nextIndex <= lines.length - hunk.oldLines.length) {
888
+ const secondMatch = seekSequence(lines, hunk.oldLines, nextIndex, false, { allowFuzzy });
889
+ if (secondMatch.index !== undefined) {
890
+ return undefined;
891
+ }
892
+ }
893
+ return fallbackResult.index;
894
+ }
895
+
896
+ for (const variant of filterFallbackVariants(buildFallbackVariants(hunk), allowAggressiveFallbacks)) {
897
+ if (variant.oldLines.length === 0) continue;
898
+ const variantResult = findSequenceWithHint(
899
+ lines,
900
+ variant.oldLines,
901
+ currentIndex,
902
+ matchHint ?? lineHint,
903
+ false,
904
+ allowFuzzy,
905
+ );
906
+ if (variantResult.index !== undefined && (variantResult.matchCount ?? 1) <= 1) {
907
+ return variantResult.index;
908
+ }
909
+ }
910
+ return undefined;
911
+ }
912
+
913
+ /**
914
+ * Apply a hunk using character-based fuzzy matching.
915
+ * Used when the hunk contains only -/+ lines without context.
916
+ */
917
+ function applyCharacterMatch(
918
+ originalContent: string,
919
+ path: string,
920
+ hunk: DiffHunk,
921
+ fuzzyThreshold: number,
922
+ allowFuzzy: boolean,
923
+ ): { content: string; warnings: string[] } {
924
+ const oldText = hunk.oldLines.join("\n");
925
+ const newText = hunk.newLines.join("\n");
926
+
927
+ const normalizedContent = normalizeToLF(originalContent);
928
+ const normalizedOldText = normalizeToLF(oldText);
929
+
930
+ let matchOutcome = findMatch(normalizedContent, normalizedOldText, {
931
+ allowFuzzy,
932
+ threshold: fuzzyThreshold,
933
+ });
934
+ if (!matchOutcome.match && allowFuzzy) {
935
+ const relaxedThreshold = Math.min(fuzzyThreshold, 0.92);
936
+ if (relaxedThreshold < fuzzyThreshold) {
937
+ const relaxedOutcome = findMatch(normalizedContent, normalizedOldText, {
938
+ allowFuzzy,
939
+ threshold: relaxedThreshold,
940
+ });
941
+ if (relaxedOutcome.match) {
942
+ matchOutcome = relaxedOutcome;
943
+ }
944
+ }
945
+ }
946
+
947
+ // Check for multiple exact occurrences
948
+ if (matchOutcome.occurrences && matchOutcome.occurrences > 1) {
949
+ const previews = matchOutcome.occurrencePreviews?.join("\n\n") ?? "";
950
+ const moreMsg = matchOutcome.occurrences > 5 ? ` (showing first 5 of ${matchOutcome.occurrences})` : "";
951
+ throw new ApplyPatchError(
952
+ `Found ${matchOutcome.occurrences} occurrences in ${path}${moreMsg}:\n\n${previews}\n\n` +
953
+ `Add more context lines to disambiguate.`,
954
+ );
955
+ }
956
+
957
+ if (matchOutcome.fuzzyMatches && matchOutcome.fuzzyMatches > 1) {
958
+ throw new ApplyPatchError(
959
+ `Found ${matchOutcome.fuzzyMatches} high-confidence matches in ${path}. ` +
960
+ `The text must be unique. Please provide more context to make it unique.`,
961
+ );
962
+ }
963
+
964
+ if (!matchOutcome.match) {
965
+ const closest = matchOutcome.closest;
966
+ if (closest) {
967
+ const similarity = Math.round(closest.confidence * 100);
968
+ throw new ApplyPatchError(
969
+ `Could not find a close enough match in ${path}. ` +
970
+ `Closest match (${similarity}% similar) at line ${closest.startLine}.`,
971
+ );
972
+ }
973
+ throw new ApplyPatchError(`Failed to find expected lines in ${path}:\n${oldText}`);
974
+ }
975
+
976
+ // Adjust indentation to match what was actually found
977
+ const adjustedNewText = adjustIndentation(normalizedOldText, matchOutcome.match.actualText, newText);
978
+
979
+ const warnings: string[] = [];
980
+ if (matchOutcome.dominantFuzzy && matchOutcome.match) {
981
+ const similarity = Math.round(matchOutcome.match.confidence * 100);
982
+ warnings.push(
983
+ `Dominant fuzzy match selected in ${path} near line ${matchOutcome.match.startLine} (${similarity}% similar).`,
984
+ );
985
+ }
986
+
987
+ // Apply the replacement
988
+ const before = normalizedContent.substring(0, matchOutcome.match.startIndex);
989
+ const after = normalizedContent.substring(matchOutcome.match.startIndex + matchOutcome.match.actualText.length);
990
+ return { content: before + adjustedNewText + after, warnings };
991
+ }
992
+
993
+ function applyTrailingNewlinePolicy(content: string, hadFinalNewline: boolean): string {
994
+ if (hadFinalNewline) {
995
+ return content.endsWith("\n") ? content : `${content}\n`;
996
+ }
997
+ return content.replace(/\n+$/u, "");
998
+ }
999
+
1000
+ async function readExistingPatchFile(fileSystem: FileSystem, absolutePath: string, path: string): Promise<string> {
1001
+ try {
1002
+ return await fileSystem.read(absolutePath);
1003
+ } catch (error) {
1004
+ if (isEnoent(error) || (error instanceof Error && error.message.startsWith("File not found:"))) {
1005
+ throw new ApplyPatchError(`File not found: ${path}`);
1006
+ }
1007
+ throw error;
1008
+ }
1009
+ }
1010
+
1011
+ /**
1012
+ * A prefix/substring strategy matched pattern lines that cover only part of
1013
+ * the corresponding file lines; replacing whole lines would silently drop the
1014
+ * uncovered text the model never saw. Allow the replacement only when every
1015
+ * discarded piece (normalized) survives somewhere in the hunk's new lines.
1016
+ */
1017
+ function assertPartialMatchPreservesDiscardedText(
1018
+ path: string,
1019
+ pattern: string[],
1020
+ matchedLines: string[],
1021
+ newLines: string[],
1022
+ matchStartIndex: number,
1023
+ ): void {
1024
+ let newLinesNorm: string | undefined;
1025
+ for (let j = 0; j < pattern.length; j++) {
1026
+ const lineNorm = normalizeForFuzzy(matchedLines[j]);
1027
+ const patternNorm = normalizeForFuzzy(pattern[j]);
1028
+ if (lineNorm === patternNorm) continue;
1029
+ const at = lineNorm.indexOf(patternNorm);
1030
+ if (at === -1) continue;
1031
+ const discardedParts = [lineNorm.slice(0, at).trim(), lineNorm.slice(at + patternNorm.length).trim()];
1032
+ for (const part of discardedParts) {
1033
+ if (part.length === 0) continue;
1034
+ newLinesNorm ??= newLines.map(normalizeForFuzzy).join("\n");
1035
+ if (!newLinesNorm.includes(part)) {
1036
+ throw new ApplyPatchError(
1037
+ `Refusing partial-line match in ${path} at line ${matchStartIndex + j + 1}: ` +
1038
+ `the file line also contains ${JSON.stringify(part)}, which the replacement would silently drop. ` +
1039
+ `Provide the complete line in the hunk.`,
1040
+ );
1041
+ }
1042
+ }
1043
+ }
1044
+ }
1045
+
1046
+ /**
1047
+ * Compute replacements needed to transform originalLines using the diff hunks.
1048
+ */
1049
+ function computeReplacements(
1050
+ originalLines: string[],
1051
+ path: string,
1052
+ hunks: DiffHunk[],
1053
+ allowFuzzy: boolean,
1054
+ ): { replacements: Replacement[]; warnings: string[] } {
1055
+ const replacements: Replacement[] = [];
1056
+ const warnings: string[] = [];
1057
+ let lineIndex = 0;
1058
+
1059
+ for (const hunk of hunks) {
1060
+ let contextIndex: number | undefined;
1061
+ if (hunk.oldStartLine !== undefined && hunk.oldStartLine < 1) {
1062
+ throw new ApplyPatchError(
1063
+ `Line hint ${hunk.oldStartLine} is out of range for ${path} (line numbers start at 1)`,
1064
+ );
1065
+ }
1066
+ if (hunk.newStartLine !== undefined && hunk.newStartLine < 1) {
1067
+ throw new ApplyPatchError(
1068
+ `Line hint ${hunk.newStartLine} is out of range for ${path} (line numbers start at 1)`,
1069
+ );
1070
+ }
1071
+ const lineHint = hunk.oldStartLine;
1072
+ const allowAggressiveFallbacks = hunk.changeContext !== undefined || lineHint !== undefined || hunk.isEndOfFile;
1073
+ const fallbackVariants = filterFallbackVariants(buildFallbackVariants(hunk), allowAggressiveFallbacks);
1074
+ if (lineHint !== undefined && hunk.changeContext === undefined && !hunk.hasContextLines) {
1075
+ lineIndex = Math.max(0, Math.min(lineHint - 1, originalLines.length - 1));
1076
+ }
1077
+
1078
+ // If hunk has a changeContext, find it and adjust lineIndex
1079
+ if (hunk.changeContext !== undefined) {
1080
+ // Use hierarchical context matching for nested @@ anchors and space-separated contexts
1081
+ const result = findHierarchicalContext(originalLines, hunk.changeContext, lineIndex, lineHint, allowFuzzy);
1082
+ const idx = result.index;
1083
+ contextIndex = idx;
1084
+
1085
+ if (idx === undefined || (result.matchCount !== undefined && result.matchCount > 1)) {
1086
+ const fallback = attemptSequenceFallback(
1087
+ originalLines,
1088
+ hunk,
1089
+ lineIndex,
1090
+ lineHint,
1091
+ allowFuzzy,
1092
+ allowAggressiveFallbacks,
1093
+ );
1094
+ if (fallback !== undefined) {
1095
+ lineIndex = fallback;
1096
+ } else if (result.matchCount !== undefined && result.matchCount > 1) {
1097
+ const displayContext = hunk.changeContext.includes("\n")
1098
+ ? hunk.changeContext.split("\n").pop()
1099
+ : hunk.changeContext;
1100
+ const previews = formatSequenceMatchPreviews(originalLines, result.matchIndices, result.matchCount);
1101
+ const strategyHint = result.strategy ? ` Matching strategy: ${result.strategy}.` : "";
1102
+ const previewText = previews ? `\n\n${previews}` : "";
1103
+ throw new ApplyPatchError(
1104
+ `Found ${result.matchCount} matches for context '${displayContext}' in ${path}.${strategyHint}` +
1105
+ `${previewText}\n\nAdd more surrounding context or additional @@ anchors to make it unique.`,
1106
+ );
1107
+ } else {
1108
+ const displayContext = hunk.changeContext.includes("\n")
1109
+ ? hunk.changeContext.split("\n").join(" > ")
1110
+ : hunk.changeContext;
1111
+ throw new ApplyPatchError(`Failed to find context '${displayContext}' in ${path}`);
1112
+ }
1113
+ } else {
1114
+ // If oldLines[0] matches the final context, start search at idx (not idx+1)
1115
+ // This handles the common case where @@ scope and first context line are identical
1116
+ const firstOldLine = hunk.oldLines[0];
1117
+ const finalContext = hunk.changeContext.includes("\n")
1118
+ ? hunk.changeContext.split("\n").pop()?.trim()
1119
+ : hunk.changeContext.trim();
1120
+ const isHierarchicalContext =
1121
+ hunk.changeContext.includes("\n") || hunk.changeContext.trim().split(/\s+/).length > 2;
1122
+ if (firstOldLine !== undefined && (firstOldLine.trim() === finalContext || isHierarchicalContext)) {
1123
+ lineIndex = idx;
1124
+ } else {
1125
+ lineIndex = idx + 1;
1126
+ }
1127
+ }
1128
+ }
1129
+
1130
+ if (hunk.oldLines.length === 0) {
1131
+ // Pure addition - prefer changeContext position, then line hint, then end of file
1132
+ let insertionIdx: number;
1133
+ if (hunk.changeContext !== undefined) {
1134
+ // changeContext was processed above; lineIndex is set to the context line or after it
1135
+ insertionIdx = lineIndex;
1136
+ } else {
1137
+ const lineHintForInsertion = hunk.oldStartLine ?? hunk.newStartLine;
1138
+ if (lineHintForInsertion !== undefined) {
1139
+ // Reject if line hint is out of range for insertion
1140
+ // Valid insertion points are 1 to (file length + 1) for 1-indexed hints
1141
+ if (lineHintForInsertion < 1) {
1142
+ throw new ApplyPatchError(
1143
+ `Line hint ${lineHintForInsertion} is out of range for insertion in ${path} ` +
1144
+ `(line numbers start at 1)`,
1145
+ );
1146
+ }
1147
+ if (lineHintForInsertion > originalLines.length + 1) {
1148
+ throw new ApplyPatchError(
1149
+ `Line hint ${lineHintForInsertion} is out of range for insertion in ${path} ` +
1150
+ `(file has ${originalLines.length} lines)`,
1151
+ );
1152
+ }
1153
+ insertionIdx = Math.max(0, lineHintForInsertion - 1);
1154
+ } else {
1155
+ insertionIdx =
1156
+ originalLines.length > 0 && originalLines[originalLines.length - 1] === ""
1157
+ ? originalLines.length - 1
1158
+ : originalLines.length;
1159
+ }
1160
+ }
1161
+
1162
+ replacements.push({ startIndex: insertionIdx, oldLen: 0, newLines: [...hunk.newLines] });
1163
+ continue;
1164
+ }
1165
+
1166
+ // Try to find the old lines in the file
1167
+ let pattern = [...hunk.oldLines];
1168
+ const matchHint = getHunkHintIndex(hunk, lineIndex);
1169
+ let searchResult = findSequenceWithHint(
1170
+ originalLines,
1171
+ pattern,
1172
+ lineIndex,
1173
+ matchHint,
1174
+ hunk.isEndOfFile,
1175
+ allowFuzzy,
1176
+ );
1177
+ let newSlice = [...hunk.newLines];
1178
+
1179
+ // Retry without trailing empty line if present
1180
+ if (searchResult.index === undefined && pattern.length > 0 && pattern[pattern.length - 1] === "") {
1181
+ pattern = pattern.slice(0, -1);
1182
+ if (newSlice.length > 0 && newSlice[newSlice.length - 1] === "") {
1183
+ newSlice = newSlice.slice(0, -1);
1184
+ }
1185
+ searchResult = findSequenceWithHint(
1186
+ originalLines,
1187
+ pattern,
1188
+ lineIndex,
1189
+ matchHint,
1190
+ hunk.isEndOfFile,
1191
+ allowFuzzy,
1192
+ );
1193
+ }
1194
+
1195
+ if (searchResult.index === undefined || (searchResult.matchCount ?? 0) > 1) {
1196
+ for (const variant of fallbackVariants) {
1197
+ if (variant.oldLines.length === 0) continue;
1198
+ const variantResult = findSequenceWithHint(
1199
+ originalLines,
1200
+ variant.oldLines,
1201
+ lineIndex,
1202
+ matchHint,
1203
+ hunk.isEndOfFile,
1204
+ allowFuzzy,
1205
+ );
1206
+ if (variantResult.index !== undefined && (variantResult.matchCount ?? 1) <= 1) {
1207
+ pattern = variant.oldLines;
1208
+ newSlice = variant.newLines;
1209
+ searchResult = variantResult;
1210
+ break;
1211
+ }
1212
+ }
1213
+ }
1214
+
1215
+ if (searchResult.index === undefined && contextIndex !== undefined) {
1216
+ for (const variant of fallbackVariants) {
1217
+ if (variant.oldLines.length !== 1 || variant.newLines.length !== 1) continue;
1218
+ const removedLine = variant.oldLines[0];
1219
+ const hasSharedDuplicate = hunk.newLines.some(line => line.trim() === removedLine.trim());
1220
+ const adjacentIndex = findContextRelativeMatch(
1221
+ originalLines,
1222
+ removedLine,
1223
+ contextIndex,
1224
+ hasSharedDuplicate,
1225
+ );
1226
+ if (adjacentIndex !== undefined) {
1227
+ pattern = variant.oldLines;
1228
+ newSlice = variant.newLines;
1229
+ searchResult = { index: adjacentIndex, confidence: 0.95 };
1230
+ break;
1231
+ }
1232
+ }
1233
+ }
1234
+
1235
+ if (searchResult.index !== undefined && contextIndex !== undefined && pattern.length === 1) {
1236
+ const trimmed = pattern[0].trim();
1237
+ let occurrenceCount = 0;
1238
+ for (const line of originalLines) {
1239
+ if (line.trim() === trimmed) occurrenceCount++;
1240
+ }
1241
+ if (occurrenceCount > 1) {
1242
+ const hasSharedDuplicate = hunk.newLines.some(line => line.trim() === trimmed);
1243
+ const contextMatch = findContextRelativeMatch(originalLines, pattern[0], contextIndex, hasSharedDuplicate);
1244
+ if (contextMatch !== undefined) {
1245
+ searchResult = { index: contextMatch, confidence: searchResult.confidence ?? 0.95 };
1246
+ }
1247
+ }
1248
+ }
1249
+
1250
+ if ((searchResult.matchCount ?? 0) > 1) {
1251
+ const hintIndex = matchHint ?? (lineHint ? lineHint - 1 : undefined);
1252
+ const hinted = chooseHintedMatch(searchResult.matchIndices, hintIndex, AMBIGUITY_HINT_WINDOW);
1253
+ if (hinted !== undefined) {
1254
+ searchResult = { ...searchResult, index: hinted, matchCount: 1 };
1255
+ }
1256
+ }
1257
+
1258
+ if (searchResult.index === undefined) {
1259
+ if (searchResult.matchCount !== undefined && searchResult.matchCount > 1) {
1260
+ const previews = formatSequenceMatchPreviews(
1261
+ originalLines,
1262
+ searchResult.matchIndices,
1263
+ searchResult.matchCount,
1264
+ );
1265
+ const strategyHint = searchResult.strategy ? ` Matching strategy: ${searchResult.strategy}.` : "";
1266
+ const previewText = previews ? `\n\n${previews}` : "";
1267
+ throw new ApplyPatchError(
1268
+ `Found ${searchResult.matchCount} matches for the text in ${path}.${strategyHint}` +
1269
+ `${previewText}\n\nAdd more surrounding context or additional @@ anchors to make it unique.`,
1270
+ );
1271
+ }
1272
+ const closest = findClosestSequenceMatch(originalLines, pattern, {
1273
+ start: lineIndex,
1274
+ eof: hunk.isEndOfFile,
1275
+ });
1276
+ if (closest.index !== undefined && closest.confidence > 0) {
1277
+ const similarity = Math.round(closest.confidence * 100);
1278
+ const preview = formatSequenceMatchPreview(originalLines, closest.index);
1279
+ throw new ApplyPatchError(
1280
+ `Failed to find expected lines in ${path}:\n${hunk.oldLines.join("\n")}\n\n` +
1281
+ `Closest match (${similarity}% similar) near line ${closest.index + 1}:\n${preview}`,
1282
+ );
1283
+ }
1284
+ throw new ApplyPatchError(`Failed to find expected lines in ${path}:\n${hunk.oldLines.join("\n")}`);
1285
+ }
1286
+
1287
+ const found = searchResult.index;
1288
+
1289
+ if (searchResult.strategy === "fuzzy-dominant") {
1290
+ const similarity = Math.round(searchResult.confidence * 100);
1291
+ warnings.push(`Dominant fuzzy match selected in ${path} near line ${found + 1} (${similarity}% similar).`);
1292
+ } else if (
1293
+ searchResult.strategy === "comment-prefix" ||
1294
+ searchResult.strategy === "prefix" ||
1295
+ searchResult.strategy === "substring" ||
1296
+ searchResult.strategy === "fuzzy" ||
1297
+ searchResult.strategy === "character"
1298
+ ) {
1299
+ const similarity = Math.round(searchResult.confidence * 100);
1300
+ warnings.push(
1301
+ `Inexact match in ${path} near line ${found + 1}: matched via ${searchResult.strategy} strategy ` +
1302
+ `(${similarity}% similar). Re-read the file if the result is not what you intended.`,
1303
+ );
1304
+ }
1305
+
1306
+ // Reject if match is ambiguous (prefix/substring matching found multiple matches)
1307
+ if (searchResult.matchCount !== undefined && searchResult.matchCount > 1) {
1308
+ const previews = formatSequenceMatchPreviews(
1309
+ originalLines,
1310
+ searchResult.matchIndices,
1311
+ searchResult.matchCount,
1312
+ );
1313
+ const strategyHint = searchResult.strategy ? ` Matching strategy: ${searchResult.strategy}.` : "";
1314
+ const previewText = previews ? `\n\n${previews}` : "";
1315
+ throw new ApplyPatchError(
1316
+ `Found ${searchResult.matchCount} matches for the text in ${path}.${strategyHint}` +
1317
+ `${previewText}\n\nAdd more surrounding context or additional @@ anchors to make it unique.`,
1318
+ );
1319
+ }
1320
+
1321
+ // For simple diffs (no context marker, no context lines), check for multiple occurrences
1322
+ // This ensures ambiguous replacements are rejected
1323
+ // Skip this check if isEndOfFile is set (EOF marker provides disambiguation)
1324
+ if (hunk.changeContext === undefined && !hunk.hasContextLines && !hunk.isEndOfFile && lineHint === undefined) {
1325
+ const secondMatch = seekSequence(originalLines, pattern, found + 1, false, { allowFuzzy });
1326
+ if (secondMatch.index !== undefined) {
1327
+ const preview1 = formatSequenceMatchPreview(originalLines, found);
1328
+ const preview2 = formatSequenceMatchPreview(originalLines, secondMatch.index);
1329
+ throw new ApplyPatchError(
1330
+ `Found 2 occurrences in ${path}:\n\n${preview1}\n\n${preview2}\n\n` +
1331
+ `Add more context lines to disambiguate.`,
1332
+ );
1333
+ }
1334
+ }
1335
+
1336
+ // Adjust indentation if needed (handles fuzzy matches where indentation differs)
1337
+ const actualMatchedLines = originalLines.slice(found, found + pattern.length);
1338
+
1339
+ // Skip pure-context hunks (no +/- lines — oldLines === newLines).
1340
+ // They serve only to advance lineIndex for subsequent hunks.
1341
+ let isNoOp = pattern.length === newSlice.length;
1342
+ if (isNoOp) {
1343
+ for (let i = 0; i < pattern.length; i++) {
1344
+ if (pattern[i] !== newSlice[i]) {
1345
+ isNoOp = false;
1346
+ break;
1347
+ }
1348
+ }
1349
+ }
1350
+
1351
+ if (isNoOp) {
1352
+ lineIndex = found + pattern.length;
1353
+ continue;
1354
+ }
1355
+
1356
+ if (searchResult.strategy === "prefix" || searchResult.strategy === "substring") {
1357
+ assertPartialMatchPreservesDiscardedText(path, pattern, actualMatchedLines, newSlice, found);
1358
+ }
1359
+
1360
+ const adjustedNewLines = adjustLinesIndentation(pattern, actualMatchedLines, newSlice);
1361
+ replacements.push({ startIndex: found, oldLen: pattern.length, newLines: adjustedNewLines });
1362
+ lineIndex = found + pattern.length;
1363
+ }
1364
+
1365
+ // Sort by start index
1366
+ replacements.sort((a, b) => a.startIndex - b.startIndex);
1367
+
1368
+ for (let i = 1; i < replacements.length; i++) {
1369
+ const prev = replacements[i - 1];
1370
+ const next = replacements[i];
1371
+ const prevEnd = prev.startIndex + prev.oldLen;
1372
+ if (next.startIndex < prevEnd) {
1373
+ const formatRange = (replacement: Replacement): string => {
1374
+ if (replacement.oldLen === 0) {
1375
+ return `${replacement.startIndex + 1} (insertion)`;
1376
+ }
1377
+ return `${replacement.startIndex + 1}-${replacement.startIndex + replacement.oldLen}`;
1378
+ };
1379
+ const prevRange = formatRange(prev);
1380
+ const nextRange = formatRange(next);
1381
+ throw new ApplyPatchError(
1382
+ `Overlapping hunks detected in ${path} at lines ${prevRange} and ${nextRange}. ` +
1383
+ `Split hunks or add more context to avoid overlap.`,
1384
+ );
1385
+ }
1386
+ }
1387
+
1388
+ return { replacements, warnings };
1389
+ }
1390
+
1391
+ /**
1392
+ * Apply replacements to lines, returning the modified content.
1393
+ */
1394
+ function applyReplacements(lines: string[], replacements: Replacement[]): string[] {
1395
+ const result = [...lines];
1396
+
1397
+ // Apply in reverse order to maintain indices
1398
+ for (let i = replacements.length - 1; i >= 0; i--) {
1399
+ const { startIndex, oldLen, newLines } = replacements[i];
1400
+ result.splice(startIndex, oldLen);
1401
+ result.splice(startIndex, 0, ...newLines);
1402
+ }
1403
+
1404
+ return result;
1405
+ }
1406
+
1407
+ /**
1408
+ * Apply diff hunks to file content.
1409
+ */
1410
+ function applyHunksToContent(
1411
+ originalContent: string,
1412
+ path: string,
1413
+ hunks: DiffHunk[],
1414
+ fuzzyThreshold: number,
1415
+ allowFuzzy: boolean,
1416
+ ): { content: string; warnings: string[] } {
1417
+ const hadFinalNewline = originalContent.endsWith("\n");
1418
+
1419
+ // Detect simple replace pattern: single hunk, no @@ context, no context lines, has old lines to match
1420
+ // Only use character-based matching when there are no hints to disambiguate
1421
+ if (hunks.length === 1) {
1422
+ const hunk = hunks[0];
1423
+ if (
1424
+ hunk.changeContext === undefined &&
1425
+ !hunk.hasContextLines &&
1426
+ hunk.oldLines.length > 0 &&
1427
+ hunk.oldStartLine === undefined && // No line hint to use for positioning
1428
+ !hunk.isEndOfFile // No EOF targeting (prefer end of file)
1429
+ ) {
1430
+ const { content, warnings } = applyCharacterMatch(originalContent, path, hunk, fuzzyThreshold, allowFuzzy);
1431
+ return { content: applyTrailingNewlinePolicy(content, hadFinalNewline), warnings };
1432
+ }
1433
+ }
1434
+
1435
+ let originalLines = originalContent.split("\n");
1436
+
1437
+ // Track if we have a trailing empty element from the final newline
1438
+ // Only strip ONE trailing empty (the newline marker), preserve actual blank lines
1439
+ let strippedTrailingEmpty = false;
1440
+ if (hadFinalNewline && originalLines.length > 0 && originalLines[originalLines.length - 1] === "") {
1441
+ // Check if the second-to-last is also empty (actual blank line) - if so, only strip one
1442
+ originalLines = originalLines.slice(0, -1);
1443
+ strippedTrailingEmpty = true;
1444
+ }
1445
+
1446
+ const { replacements, warnings } = computeReplacements(originalLines, path, hunks, allowFuzzy);
1447
+ const newLines = applyReplacements(originalLines, replacements);
1448
+
1449
+ // Restore the trailing empty element if we stripped it
1450
+ if (strippedTrailingEmpty) {
1451
+ newLines.push("");
1452
+ }
1453
+
1454
+ const content = newLines.join("\n");
1455
+ return { content: applyTrailingNewlinePolicy(content, hadFinalNewline), warnings };
1456
+ }
1457
+
1458
+ // ═══════════════════════════════════════════════════════════════════════════
1459
+ // Public API
1460
+ // ═══════════════════════════════════════════════════════════════════════════
1461
+
1462
+ /**
1463
+ * Apply a patch operation to the filesystem.
1464
+ */
1465
+ export async function applyPatch(input: PatchInput, options: ApplyPatchOptions): Promise<ApplyPatchResult> {
1466
+ return applyNormalizedPatch(input, options);
1467
+ }
1468
+
1469
+ /**
1470
+ * Apply a normalized patch operation to the filesystem.
1471
+ * @internal
1472
+ */
1473
+ async function applyNormalizedPatch(input: PatchInput, options: ApplyPatchOptions): Promise<ApplyPatchResult> {
1474
+ const {
1475
+ cwd,
1476
+ dryRun = false,
1477
+ fs = defaultFileSystem,
1478
+ fuzzyThreshold = DEFAULT_FUZZY_THRESHOLD,
1479
+ allowFuzzy = true,
1480
+ } = options;
1481
+
1482
+ const resolvePath = (p: string): string => resolveToCwd(p, cwd);
1483
+ const absolutePath = resolvePath(input.path);
1484
+ const op = input.op ?? "update";
1485
+
1486
+ if (input.rename) {
1487
+ const destPath = resolvePath(input.rename);
1488
+ if (destPath === absolutePath) {
1489
+ throw new ApplyPatchError("rename path is the same as source path");
1490
+ }
1491
+ }
1492
+
1493
+ // Handle CREATE operation
1494
+ if (op === "create") {
1495
+ if (!input.diff) {
1496
+ throw new ApplyPatchError("Create operation requires diff (file content)");
1497
+ }
1498
+ // Strip + prefixes if present (handles diffs formatted as additions)
1499
+ const normalizedContent = normalizeCreateContent(input.diff);
1500
+ const content = normalizedContent.endsWith("\n") ? normalizedContent : `${normalizedContent}\n`;
1501
+
1502
+ if (!dryRun) {
1503
+ const parentDir = path.dirname(absolutePath);
1504
+ if (parentDir && parentDir !== ".") {
1505
+ await fs.mkdir(parentDir);
1506
+ }
1507
+ await fs.write(absolutePath, content);
1508
+ }
1509
+
1510
+ return {
1511
+ change: {
1512
+ type: "create",
1513
+ path: absolutePath,
1514
+ newContent: content,
1515
+ },
1516
+ };
1517
+ }
1518
+
1519
+ // Handle DELETE operation
1520
+ if (op === "delete") {
1521
+ const oldContent = await readExistingPatchFile(fs, absolutePath, input.path);
1522
+ if (!dryRun) {
1523
+ await fs.delete(absolutePath);
1524
+ }
1525
+
1526
+ return {
1527
+ change: {
1528
+ type: "delete",
1529
+ path: absolutePath,
1530
+ oldContent,
1531
+ },
1532
+ };
1533
+ }
1534
+
1535
+ // Handle UPDATE operation
1536
+ if (!input.diff) {
1537
+ throw new ApplyPatchError("Update operation requires diff (hunks)");
1538
+ }
1539
+
1540
+ const originalContent = await readExistingPatchFile(fs, absolutePath, input.path);
1541
+ const { bom: bomFromText, text: strippedContent } = stripBom(originalContent);
1542
+ let bom = bomFromText;
1543
+ if (!bom && fs.readBinary) {
1544
+ const bytes = await fs.readBinary(absolutePath);
1545
+ if (bytes.length >= 3 && bytes[0] === 0xef && bytes[1] === 0xbb && bytes[2] === 0xbf) {
1546
+ bom = "\uFEFF";
1547
+ }
1548
+ }
1549
+ const lineEnding = detectLineEnding(strippedContent);
1550
+ const normalizedContent = normalizeToLF(strippedContent);
1551
+ const hunks = parseDiffHunks(input.diff);
1552
+
1553
+ if (hunks.length === 0) {
1554
+ throw new ApplyPatchError("Diff contains no hunks");
1555
+ }
1556
+
1557
+ const { content: newContent, warnings } = applyHunksToContent(
1558
+ normalizedContent,
1559
+ input.path,
1560
+ hunks,
1561
+ fuzzyThreshold,
1562
+ allowFuzzy,
1563
+ );
1564
+ const finalContent = bom + restoreLineEndings(newContent, lineEnding);
1565
+ const destPath = input.rename ? resolvePath(input.rename) : absolutePath;
1566
+ const isMove = Boolean(input.rename) && destPath !== absolutePath;
1567
+
1568
+ if (!dryRun) {
1569
+ if (isMove) {
1570
+ const parentDir = path.dirname(destPath);
1571
+ if (parentDir && parentDir !== ".") {
1572
+ await fs.mkdir(parentDir);
1573
+ }
1574
+ await fs.write(destPath, finalContent);
1575
+ await fs.delete(absolutePath);
1576
+ } else {
1577
+ await fs.write(absolutePath, finalContent);
1578
+ }
1579
+ }
1580
+
1581
+ return {
1582
+ change: {
1583
+ type: "update",
1584
+ path: absolutePath,
1585
+ newPath: isMove ? destPath : undefined,
1586
+ oldContent: originalContent,
1587
+ newContent: finalContent,
1588
+ },
1589
+ warnings: warnings.length > 0 ? warnings : undefined,
1590
+ };
1591
+ }
1592
+
1593
+ /**
1594
+ * Preview what changes a patch would make without applying it.
1595
+ */
1596
+ export async function previewPatch(input: PatchInput, options: ApplyPatchOptions): Promise<ApplyPatchResult> {
1597
+ return applyPatch(input, { ...options, dryRun: true });
1598
+ }
1599
+
1600
+ export async function computePatchDiff(
1601
+ input: PatchInput,
1602
+ cwd: string,
1603
+ options?: { fuzzyThreshold?: number; allowFuzzy?: boolean },
1604
+ ): Promise<
1605
+ | {
1606
+ diff: string;
1607
+ firstChangedLine: number | undefined;
1608
+ }
1609
+ | {
1610
+ error: string;
1611
+ }
1612
+ > {
1613
+ try {
1614
+ const result = await previewPatch(input, {
1615
+ cwd,
1616
+ fuzzyThreshold: options?.fuzzyThreshold,
1617
+ allowFuzzy: options?.allowFuzzy,
1618
+ });
1619
+ const oldContent = result.change.oldContent ?? "";
1620
+ const newContent = result.change.newContent ?? "";
1621
+ const normalizedOld = normalizeToLF(stripBom(oldContent).text);
1622
+ const normalizedNew = normalizeToLF(stripBom(newContent).text);
1623
+ if (!normalizedOld && !normalizedNew) {
1624
+ return { diff: "", firstChangedLine: undefined };
1625
+ }
1626
+ return generateUnifiedDiffString(normalizedOld, normalizedNew, undefined, {
1627
+ path: result.change.newPath ?? result.change.path,
1628
+ });
1629
+ } catch (err) {
1630
+ return { error: err instanceof Error ? err.message : String(err) };
1631
+ }
1632
+ }
1633
+
1634
+ export const patchEditEntrySchema = z
1635
+ .object({
1636
+ op: z.enum(["create", "delete", "update"]).optional().describe("operation (default update)"),
1637
+ rename: z.string().describe("new path for move").optional(),
1638
+ diff: z.string().describe("diff hunks or full content for create").optional(),
1639
+ })
1640
+ .strict();
1641
+
1642
+ export const patchEditSchema = z
1643
+ .object({
1644
+ path: z.string().describe("file path"),
1645
+ edits: z.array(patchEditEntrySchema).min(1).describe("patch operations"),
1646
+ })
1647
+ .strict();
1648
+
1649
+ export type PatchEditEntry = z.infer<typeof patchEditEntrySchema>;
1650
+ export type PatchParams = z.infer<typeof patchEditSchema>;
1651
+
1652
+ export interface ExecutePatchSingleOptions {
1653
+ session: ToolSession;
1654
+ path: string;
1655
+ params: PatchEditEntry;
1656
+ signal?: AbortSignal;
1657
+ batchRequest?: LspBatchRequest;
1658
+ allowFuzzy: boolean;
1659
+ fuzzyThreshold: number;
1660
+ writethrough: WritethroughCallback;
1661
+ beginDeferredDiagnosticsForPath: (path: string) => WritethroughDeferredHandle;
1662
+ }
1663
+
1664
+ class LspFileSystem implements FileSystem {
1665
+ #lastDiagnostics: FileDiagnosticsResult | undefined;
1666
+ #fileCache: Record<string, Bun.BunFile> = {};
1667
+
1668
+ constructor(
1669
+ private readonly writethrough: WritethroughCallback,
1670
+ private readonly signal?: AbortSignal,
1671
+ private readonly batchRequest?: LspBatchRequest,
1672
+ private readonly deferredForPath?: (path: string) => WritethroughDeferredHandle,
1673
+ ) {}
1674
+
1675
+ #getFile(path: string): Bun.BunFile {
1676
+ if (this.#fileCache[path]) {
1677
+ return this.#fileCache[path];
1678
+ }
1679
+ const file = Bun.file(path);
1680
+ this.#fileCache[path] = file;
1681
+ return file;
1682
+ }
1683
+
1684
+ async exists(path: string): Promise<boolean> {
1685
+ return this.#getFile(path).exists();
1686
+ }
1687
+
1688
+ async read(path: string): Promise<string> {
1689
+ return readEditFileText(path, path);
1690
+ }
1691
+
1692
+ async readBinary(path: string): Promise<Uint8Array> {
1693
+ const bytes = await fs.promises.readFile(path);
1694
+ return bytes;
1695
+ }
1696
+
1697
+ async write(path: string, content: string): Promise<void> {
1698
+ const file = this.#getFile(path);
1699
+ const finalContent = await serializeEditFileText(path, path, content);
1700
+ const deferredForPath = this.deferredForPath;
1701
+ const result = await this.writethrough(
1702
+ path,
1703
+ finalContent,
1704
+ this.signal,
1705
+ file,
1706
+ this.batchRequest,
1707
+ deferredForPath ? (dst: string) => deferredForPath(dst) : undefined,
1708
+ );
1709
+ if (result) {
1710
+ this.#lastDiagnostics = result;
1711
+ }
1712
+ }
1713
+
1714
+ async delete(path: string): Promise<void> {
1715
+ await this.#getFile(path).unlink();
1716
+ }
1717
+
1718
+ async mkdir(path: string): Promise<void> {
1719
+ await fs.promises.mkdir(path, { recursive: true });
1720
+ }
1721
+
1722
+ getDiagnostics(): FileDiagnosticsResult | undefined {
1723
+ return this.#lastDiagnostics;
1724
+ }
1725
+ }
1726
+
1727
+ function mergeDiagnosticsWithWarnings(
1728
+ diagnostics: FileDiagnosticsResult | undefined,
1729
+ warnings: string[],
1730
+ ): FileDiagnosticsResult | undefined {
1731
+ if (warnings.length === 0) return diagnostics;
1732
+ const warningMessages = warnings.map(warning => `patch: ${warning}`);
1733
+ if (!diagnostics) {
1734
+ return {
1735
+ server: "patch",
1736
+ messages: warningMessages,
1737
+ summary: `Patch warnings: ${warnings.length}`,
1738
+ errored: false,
1739
+ };
1740
+ }
1741
+ return {
1742
+ ...diagnostics,
1743
+ messages: [...warningMessages, ...diagnostics.messages],
1744
+ summary: `${diagnostics.summary}; Patch warnings: ${warnings.length}`,
1745
+ };
1746
+ }
1747
+
1748
+ export async function executePatchSingle(
1749
+ options: ExecutePatchSingleOptions,
1750
+ ): Promise<AgentToolResult<EditToolDetails, typeof patchEditEntrySchema>> {
1751
+ const {
1752
+ session,
1753
+ path,
1754
+ params,
1755
+ signal,
1756
+ batchRequest,
1757
+ allowFuzzy,
1758
+ fuzzyThreshold,
1759
+ writethrough,
1760
+ beginDeferredDiagnosticsForPath,
1761
+ } = options;
1762
+ const { op: rawOp, rename, diff } = params;
1763
+
1764
+ const op: Operation = rawOp === "create" || rawOp === "delete" ? rawOp : "update";
1765
+
1766
+ enforcePlanModeWrite(session, path, { op, move: rename });
1767
+ const resolvedPath = resolvePlanPath(session, path);
1768
+ const resolvedRename = rename ? resolvePlanPath(session, rename) : undefined;
1769
+
1770
+ await assertEditableFile(resolvedPath, path);
1771
+
1772
+ // Capture pre-edit content so we can verify the write actually hit disk.
1773
+ // `LspFileSystem.writeFile` delegates to a writethrough callback that, in
1774
+ // some host integrations, has been observed to report success without
1775
+ // persisting bytes — leaving the tool to claim "Updated <path>" while the
1776
+ // file on disk is byte-identical to before. After the write we re-read
1777
+ // the file and assert the bytes match the expected newContent; relying
1778
+ // on stat (mtime/size) is unreliable because filesystems with coarse
1779
+ // timestamp resolution can record an unchanged mtime even when the
1780
+ // content was rewritten, and same-length rewrites leave size unchanged.
1781
+ let preEditContent: Uint8Array | undefined;
1782
+ if (op === "update") {
1783
+ try {
1784
+ preEditContent = await fs.promises.readFile(resolvedPath);
1785
+ } catch (err) {
1786
+ if (!isEnoent(err)) throw err;
1787
+ }
1788
+ }
1789
+
1790
+ const input: PatchInput = { path: resolvedPath, op, rename: resolvedRename, diff };
1791
+ const patchFileSystem = new LspFileSystem(writethrough, signal, batchRequest, beginDeferredDiagnosticsForPath);
1792
+ const result = await applyPatch(input, {
1793
+ cwd: session.cwd,
1794
+ fs: patchFileSystem,
1795
+ fuzzyThreshold,
1796
+ allowFuzzy,
1797
+ });
1798
+
1799
+ // Post-write verification: only meaningful for in-place updates where the
1800
+ // patch actually changes content and the file is not being renamed away.
1801
+ if (
1802
+ result.change.type === "update" &&
1803
+ !result.change.newPath &&
1804
+ preEditContent !== undefined &&
1805
+ result.change.oldContent !== undefined &&
1806
+ result.change.newContent !== undefined &&
1807
+ result.change.oldContent !== result.change.newContent
1808
+ ) {
1809
+ let postEditContent: Uint8Array | undefined;
1810
+ try {
1811
+ postEditContent = await fs.promises.readFile(resolvedPath);
1812
+ } catch (err) {
1813
+ if (!isEnoent(err)) throw err;
1814
+ }
1815
+ const unchanged =
1816
+ postEditContent !== undefined &&
1817
+ postEditContent.length === preEditContent.length &&
1818
+ postEditContent.every((b, i) => b === preEditContent[i]);
1819
+ if (unchanged) {
1820
+ throw new ToolError(`edit appeared successful but file content did not change on disk: ${path}`, {
1821
+ path: resolvedPath,
1822
+ });
1823
+ }
1824
+ }
1825
+
1826
+ if (resolvedRename) {
1827
+ invalidateFsScanAfterRename(resolvedPath, resolvedRename);
1828
+ } else if (result.change.type === "delete") {
1829
+ invalidateFsScanAfterDelete(resolvedPath);
1830
+ } else {
1831
+ invalidateFsScanAfterWrite(resolvedPath);
1832
+ }
1833
+ const effectiveRename = result.change.newPath ? rename : undefined;
1834
+
1835
+ let diffResult: { diff: string; firstChangedLine: number | undefined } = {
1836
+ diff: "",
1837
+ firstChangedLine: undefined,
1838
+ };
1839
+ if (result.change.type === "update" && result.change.oldContent && result.change.newContent) {
1840
+ const normalizedOld = normalizeToLF(stripBom(result.change.oldContent).text);
1841
+ const normalizedNew = normalizeToLF(stripBom(result.change.newContent).text);
1842
+ diffResult = generateUnifiedDiffString(normalizedOld, normalizedNew, undefined, {
1843
+ path: result.change.newPath ?? result.change.path,
1844
+ });
1845
+ }
1846
+
1847
+ let resultText: string;
1848
+ switch (result.change.type) {
1849
+ case "create":
1850
+ resultText = `Created ${path}`;
1851
+ break;
1852
+ case "delete":
1853
+ resultText = `Deleted ${path}`;
1854
+ break;
1855
+ case "update":
1856
+ resultText = effectiveRename ? `Updated and moved ${path} to ${effectiveRename}` : `Updated ${path}`;
1857
+ break;
1858
+ }
1859
+
1860
+ let diagnostics = patchFileSystem.getDiagnostics();
1861
+ if (op === "delete" && batchRequest?.flush) {
1862
+ const flushedDiagnostics = await flushLspWritethroughBatch(batchRequest.id, session.cwd, signal);
1863
+ diagnostics ??= flushedDiagnostics;
1864
+ }
1865
+ const mergedDiagnostics = mergeDiagnosticsWithWarnings(diagnostics, result.warnings ?? []);
1866
+ const meta = outputMeta()
1867
+ .diagnostics(mergedDiagnostics?.summary ?? "", mergedDiagnostics?.messages ?? [])
1868
+ .get();
1869
+
1870
+ const oldText = result.change.type !== "create" ? result.change.oldContent : undefined;
1871
+ const newText = result.change.type !== "delete" ? result.change.newContent : undefined;
1872
+
1873
+ return {
1874
+ content: [{ type: "text", text: resultText }],
1875
+ details: {
1876
+ diff: diffResult.diff,
1877
+ // When the patch moves the file, anchor the diff to the destination
1878
+ // path. ACP `ToolCallContent.diff.path` comes from this field, and
1879
+ // clients use it to open or focus the file post-change; pointing at
1880
+ // the (now-deleted) source navigates to nothing.
1881
+ path: result.change.newPath ?? resolvedPath,
1882
+ firstChangedLine: diffResult.firstChangedLine,
1883
+ diagnostics: mergedDiagnostics,
1884
+ op,
1885
+ move: effectiveRename,
1886
+ meta,
1887
+ oldText,
1888
+ newText,
1889
+ },
1890
+ };
1891
+ }