@nghyane/arcane 0.1.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 (738) hide show
  1. package/CHANGELOG.md +3 -0
  2. package/README.md +12 -0
  3. package/examples/README.md +21 -0
  4. package/examples/custom-tools/README.md +109 -0
  5. package/examples/custom-tools/hello/index.ts +20 -0
  6. package/examples/custom-tools/todo/index.ts +206 -0
  7. package/examples/extensions/README.md +143 -0
  8. package/examples/extensions/api-demo.ts +89 -0
  9. package/examples/extensions/chalk-logger.ts +25 -0
  10. package/examples/extensions/hello.ts +32 -0
  11. package/examples/extensions/pirate.ts +43 -0
  12. package/examples/extensions/plan-mode.ts +550 -0
  13. package/examples/extensions/reload-runtime.ts +37 -0
  14. package/examples/extensions/todo.ts +296 -0
  15. package/examples/extensions/tools.ts +144 -0
  16. package/examples/extensions/with-deps/index.ts +35 -0
  17. package/examples/extensions/with-deps/package-lock.json +31 -0
  18. package/examples/extensions/with-deps/package.json +16 -0
  19. package/examples/hooks/README.md +56 -0
  20. package/examples/hooks/auto-commit-on-exit.ts +48 -0
  21. package/examples/hooks/confirm-destructive.ts +58 -0
  22. package/examples/hooks/custom-compaction.ts +116 -0
  23. package/examples/hooks/dirty-repo-guard.ts +51 -0
  24. package/examples/hooks/file-trigger.ts +40 -0
  25. package/examples/hooks/git-checkpoint.ts +52 -0
  26. package/examples/hooks/handoff.ts +150 -0
  27. package/examples/hooks/permission-gate.ts +33 -0
  28. package/examples/hooks/protected-paths.ts +29 -0
  29. package/examples/hooks/qna.ts +119 -0
  30. package/examples/hooks/status-line.ts +39 -0
  31. package/examples/sdk/01-minimal.ts +21 -0
  32. package/examples/sdk/02-custom-model.ts +49 -0
  33. package/examples/sdk/03-custom-prompt.ts +43 -0
  34. package/examples/sdk/04-skills.ts +43 -0
  35. package/examples/sdk/06-extensions.ts +80 -0
  36. package/examples/sdk/06-hooks.ts +61 -0
  37. package/examples/sdk/07-context-files.ts +35 -0
  38. package/examples/sdk/08-prompt-templates.ts +36 -0
  39. package/examples/sdk/08-slash-commands.ts +41 -0
  40. package/examples/sdk/09-api-keys-and-oauth.ts +54 -0
  41. package/examples/sdk/11-sessions.ts +47 -0
  42. package/examples/sdk/README.md +150 -0
  43. package/package.json +464 -0
  44. package/scripts/format-prompts.ts +184 -0
  45. package/scripts/generate-docs-index.ts +40 -0
  46. package/scripts/generate-template.ts +32 -0
  47. package/src/bun-imports.d.ts +22 -0
  48. package/src/capability/context-file.ts +39 -0
  49. package/src/capability/extension-module.ts +33 -0
  50. package/src/capability/extension.ts +47 -0
  51. package/src/capability/fs.ts +89 -0
  52. package/src/capability/hook.ts +39 -0
  53. package/src/capability/index.ts +432 -0
  54. package/src/capability/instruction.ts +36 -0
  55. package/src/capability/mcp.ts +60 -0
  56. package/src/capability/prompt.ts +34 -0
  57. package/src/capability/rule.ts +223 -0
  58. package/src/capability/settings.ts +34 -0
  59. package/src/capability/skill.ts +48 -0
  60. package/src/capability/slash-command.ts +39 -0
  61. package/src/capability/ssh.ts +41 -0
  62. package/src/capability/system-prompt.ts +34 -0
  63. package/src/capability/tool.ts +37 -0
  64. package/src/capability/types.ts +156 -0
  65. package/src/cli/args.ts +259 -0
  66. package/src/cli/config-cli.ts +357 -0
  67. package/src/cli/file-processor.ts +124 -0
  68. package/src/cli/grep-cli.ts +152 -0
  69. package/src/cli/jupyter-cli.ts +106 -0
  70. package/src/cli/list-models.ts +103 -0
  71. package/src/cli/plugin-cli.ts +661 -0
  72. package/src/cli/session-picker.ts +42 -0
  73. package/src/cli/setup-cli.ts +376 -0
  74. package/src/cli/shell-cli.ts +174 -0
  75. package/src/cli/ssh-cli.ts +179 -0
  76. package/src/cli/stats-cli.ts +197 -0
  77. package/src/cli/update-cli.ts +286 -0
  78. package/src/cli/web-search-cli.ts +143 -0
  79. package/src/cli.ts +65 -0
  80. package/src/commands/commit.ts +36 -0
  81. package/src/commands/config.ts +51 -0
  82. package/src/commands/grep.ts +41 -0
  83. package/src/commands/jupyter.ts +32 -0
  84. package/src/commands/launch.ts +139 -0
  85. package/src/commands/plugin.ts +70 -0
  86. package/src/commands/setup.ts +42 -0
  87. package/src/commands/shell.ts +29 -0
  88. package/src/commands/ssh.ts +60 -0
  89. package/src/commands/stats.ts +29 -0
  90. package/src/commands/update.ts +21 -0
  91. package/src/commands/web-search.ts +42 -0
  92. package/src/commit/agentic/agent.ts +311 -0
  93. package/src/commit/agentic/fallback.ts +96 -0
  94. package/src/commit/agentic/index.ts +359 -0
  95. package/src/commit/agentic/prompts/analyze-file.md +22 -0
  96. package/src/commit/agentic/prompts/session-user.md +25 -0
  97. package/src/commit/agentic/prompts/split-confirm.md +1 -0
  98. package/src/commit/agentic/prompts/system.md +38 -0
  99. package/src/commit/agentic/state.ts +69 -0
  100. package/src/commit/agentic/tools/analyze-file.ts +118 -0
  101. package/src/commit/agentic/tools/git-file-diff.ts +194 -0
  102. package/src/commit/agentic/tools/git-hunk.ts +50 -0
  103. package/src/commit/agentic/tools/git-overview.ts +84 -0
  104. package/src/commit/agentic/tools/index.ts +56 -0
  105. package/src/commit/agentic/tools/propose-changelog.ts +128 -0
  106. package/src/commit/agentic/tools/propose-commit.ts +154 -0
  107. package/src/commit/agentic/tools/recent-commits.ts +81 -0
  108. package/src/commit/agentic/tools/split-commit.ts +280 -0
  109. package/src/commit/agentic/topo-sort.ts +44 -0
  110. package/src/commit/agentic/trivial.ts +51 -0
  111. package/src/commit/agentic/validation.ts +200 -0
  112. package/src/commit/analysis/conventional.ts +165 -0
  113. package/src/commit/analysis/index.ts +4 -0
  114. package/src/commit/analysis/scope.ts +242 -0
  115. package/src/commit/analysis/summary.ts +112 -0
  116. package/src/commit/analysis/validation.ts +66 -0
  117. package/src/commit/changelog/detect.ts +37 -0
  118. package/src/commit/changelog/generate.ts +110 -0
  119. package/src/commit/changelog/index.ts +234 -0
  120. package/src/commit/changelog/parse.ts +44 -0
  121. package/src/commit/cli.ts +93 -0
  122. package/src/commit/git/diff.ts +148 -0
  123. package/src/commit/git/errors.ts +9 -0
  124. package/src/commit/git/index.ts +211 -0
  125. package/src/commit/git/operations.ts +54 -0
  126. package/src/commit/index.ts +5 -0
  127. package/src/commit/map-reduce/index.ts +64 -0
  128. package/src/commit/map-reduce/map-phase.ts +178 -0
  129. package/src/commit/map-reduce/reduce-phase.ts +145 -0
  130. package/src/commit/map-reduce/utils.ts +9 -0
  131. package/src/commit/message.ts +11 -0
  132. package/src/commit/model-selection.ts +69 -0
  133. package/src/commit/pipeline.ts +243 -0
  134. package/src/commit/prompts/analysis-system.md +148 -0
  135. package/src/commit/prompts/analysis-user.md +38 -0
  136. package/src/commit/prompts/changelog-system.md +50 -0
  137. package/src/commit/prompts/changelog-user.md +18 -0
  138. package/src/commit/prompts/file-observer-system.md +24 -0
  139. package/src/commit/prompts/file-observer-user.md +8 -0
  140. package/src/commit/prompts/reduce-system.md +50 -0
  141. package/src/commit/prompts/reduce-user.md +17 -0
  142. package/src/commit/prompts/summary-retry.md +3 -0
  143. package/src/commit/prompts/summary-system.md +38 -0
  144. package/src/commit/prompts/summary-user.md +13 -0
  145. package/src/commit/prompts/types-description.md +2 -0
  146. package/src/commit/types.ts +109 -0
  147. package/src/commit/utils/exclusions.ts +42 -0
  148. package/src/config/file-lock.ts +121 -0
  149. package/src/config/keybindings.ts +280 -0
  150. package/src/config/model-registry.ts +1140 -0
  151. package/src/config/model-resolver.ts +812 -0
  152. package/src/config/prompt-templates.ts +526 -0
  153. package/src/config/resolve-config-value.ts +92 -0
  154. package/src/config/settings-schema.ts +1236 -0
  155. package/src/config/settings.ts +706 -0
  156. package/src/config.ts +414 -0
  157. package/src/cursor.ts +239 -0
  158. package/src/debug/index.ts +431 -0
  159. package/src/debug/log-formatting.ts +60 -0
  160. package/src/debug/log-viewer.ts +903 -0
  161. package/src/debug/profiler.ts +158 -0
  162. package/src/debug/report-bundle.ts +366 -0
  163. package/src/debug/system-info.ts +112 -0
  164. package/src/discovery/agents-md.ts +68 -0
  165. package/src/discovery/agents.ts +199 -0
  166. package/src/discovery/builtin.ts +815 -0
  167. package/src/discovery/claude-plugins.ts +205 -0
  168. package/src/discovery/claude.ts +506 -0
  169. package/src/discovery/cline.ts +83 -0
  170. package/src/discovery/codex.ts +532 -0
  171. package/src/discovery/cursor.ts +218 -0
  172. package/src/discovery/gemini.ts +395 -0
  173. package/src/discovery/github.ts +117 -0
  174. package/src/discovery/helpers.ts +698 -0
  175. package/src/discovery/index.ts +89 -0
  176. package/src/discovery/mcp-json.ts +156 -0
  177. package/src/discovery/opencode.ts +394 -0
  178. package/src/discovery/ssh.ts +160 -0
  179. package/src/discovery/vscode.ts +103 -0
  180. package/src/discovery/windsurf.ts +145 -0
  181. package/src/exa/company.ts +57 -0
  182. package/src/exa/index.ts +62 -0
  183. package/src/exa/linkedin.ts +57 -0
  184. package/src/exa/mcp-client.ts +289 -0
  185. package/src/exa/render.ts +244 -0
  186. package/src/exa/researcher.ts +89 -0
  187. package/src/exa/search.ts +330 -0
  188. package/src/exa/types.ts +166 -0
  189. package/src/exa/websets.ts +247 -0
  190. package/src/exec/bash-executor.ts +184 -0
  191. package/src/exec/exec.ts +53 -0
  192. package/src/export/custom-share.ts +65 -0
  193. package/src/export/html/index.ts +162 -0
  194. package/src/export/html/template.css +889 -0
  195. package/src/export/html/template.generated.ts +2 -0
  196. package/src/export/html/template.html +45 -0
  197. package/src/export/html/template.js +1329 -0
  198. package/src/export/html/template.macro.ts +24 -0
  199. package/src/export/html/vendor/highlight.min.js +1213 -0
  200. package/src/export/html/vendor/marked.min.js +6 -0
  201. package/src/export/ttsr.ts +434 -0
  202. package/src/extensibility/custom-commands/bundled/review/index.ts +433 -0
  203. package/src/extensibility/custom-commands/index.ts +15 -0
  204. package/src/extensibility/custom-commands/loader.ts +231 -0
  205. package/src/extensibility/custom-commands/types.ts +111 -0
  206. package/src/extensibility/custom-tools/index.ts +22 -0
  207. package/src/extensibility/custom-tools/loader.ts +235 -0
  208. package/src/extensibility/custom-tools/types.ts +226 -0
  209. package/src/extensibility/custom-tools/wrapper.ts +45 -0
  210. package/src/extensibility/extensions/index.ts +136 -0
  211. package/src/extensibility/extensions/loader.ts +520 -0
  212. package/src/extensibility/extensions/runner.ts +774 -0
  213. package/src/extensibility/extensions/types.ts +1293 -0
  214. package/src/extensibility/extensions/wrapper.ts +188 -0
  215. package/src/extensibility/hooks/index.ts +16 -0
  216. package/src/extensibility/hooks/loader.ts +273 -0
  217. package/src/extensibility/hooks/runner.ts +441 -0
  218. package/src/extensibility/hooks/tool-wrapper.ts +106 -0
  219. package/src/extensibility/hooks/types.ts +817 -0
  220. package/src/extensibility/plugins/doctor.ts +65 -0
  221. package/src/extensibility/plugins/git-url.ts +281 -0
  222. package/src/extensibility/plugins/index.ts +33 -0
  223. package/src/extensibility/plugins/installer.ts +192 -0
  224. package/src/extensibility/plugins/loader.ts +338 -0
  225. package/src/extensibility/plugins/manager.ts +716 -0
  226. package/src/extensibility/plugins/parser.ts +105 -0
  227. package/src/extensibility/plugins/types.ts +190 -0
  228. package/src/extensibility/skills.ts +385 -0
  229. package/src/extensibility/slash-commands.ts +287 -0
  230. package/src/extensibility/tool-proxy.ts +25 -0
  231. package/src/index.ts +275 -0
  232. package/src/internal-urls/agent-protocol.ts +136 -0
  233. package/src/internal-urls/artifact-protocol.ts +97 -0
  234. package/src/internal-urls/docs-index.generated.ts +54 -0
  235. package/src/internal-urls/docs-protocol.ts +84 -0
  236. package/src/internal-urls/index.ts +31 -0
  237. package/src/internal-urls/json-query.ts +126 -0
  238. package/src/internal-urls/memory-protocol.ts +133 -0
  239. package/src/internal-urls/router.ts +70 -0
  240. package/src/internal-urls/rule-protocol.ts +55 -0
  241. package/src/internal-urls/skill-protocol.ts +111 -0
  242. package/src/internal-urls/types.ts +52 -0
  243. package/src/ipy/executor.ts +556 -0
  244. package/src/ipy/gateway-coordinator.ts +426 -0
  245. package/src/ipy/kernel.ts +892 -0
  246. package/src/ipy/modules.ts +109 -0
  247. package/src/ipy/prelude.py +831 -0
  248. package/src/ipy/prelude.ts +3 -0
  249. package/src/ipy/runtime.ts +222 -0
  250. package/src/lsp/client.ts +867 -0
  251. package/src/lsp/clients/biome-client.ts +202 -0
  252. package/src/lsp/clients/index.ts +50 -0
  253. package/src/lsp/clients/lsp-linter-client.ts +93 -0
  254. package/src/lsp/clients/swiftlint-client.ts +120 -0
  255. package/src/lsp/config.ts +397 -0
  256. package/src/lsp/defaults.json +464 -0
  257. package/src/lsp/edits.ts +109 -0
  258. package/src/lsp/index.ts +1268 -0
  259. package/src/lsp/lspmux.ts +250 -0
  260. package/src/lsp/render.ts +689 -0
  261. package/src/lsp/types.ts +414 -0
  262. package/src/lsp/utils.ts +549 -0
  263. package/src/main.ts +773 -0
  264. package/src/mcp/client.ts +239 -0
  265. package/src/mcp/config-writer.ts +215 -0
  266. package/src/mcp/config.ts +363 -0
  267. package/src/mcp/index.ts +55 -0
  268. package/src/mcp/json-rpc.ts +84 -0
  269. package/src/mcp/loader.ts +124 -0
  270. package/src/mcp/manager.ts +490 -0
  271. package/src/mcp/oauth-discovery.ts +274 -0
  272. package/src/mcp/oauth-flow.ts +229 -0
  273. package/src/mcp/render.ts +123 -0
  274. package/src/mcp/tool-bridge.ts +372 -0
  275. package/src/mcp/tool-cache.ts +121 -0
  276. package/src/mcp/transports/http.ts +332 -0
  277. package/src/mcp/transports/index.ts +6 -0
  278. package/src/mcp/transports/stdio.ts +281 -0
  279. package/src/mcp/types.ts +248 -0
  280. package/src/memories/index.ts +1099 -0
  281. package/src/memories/storage.ts +563 -0
  282. package/src/modes/components/agent-dashboard.ts +1130 -0
  283. package/src/modes/components/assistant-message.ts +144 -0
  284. package/src/modes/components/bash-execution.ts +218 -0
  285. package/src/modes/components/bordered-loader.ts +41 -0
  286. package/src/modes/components/branch-summary-message.ts +45 -0
  287. package/src/modes/components/codemode-group.ts +369 -0
  288. package/src/modes/components/compaction-summary-message.ts +51 -0
  289. package/src/modes/components/countdown-timer.ts +46 -0
  290. package/src/modes/components/custom-editor.ts +181 -0
  291. package/src/modes/components/custom-message.ts +91 -0
  292. package/src/modes/components/diff.ts +186 -0
  293. package/src/modes/components/dynamic-border.ts +25 -0
  294. package/src/modes/components/extensions/extension-dashboard.ts +325 -0
  295. package/src/modes/components/extensions/extension-list.ts +484 -0
  296. package/src/modes/components/extensions/index.ts +9 -0
  297. package/src/modes/components/extensions/inspector-panel.ts +321 -0
  298. package/src/modes/components/extensions/state-manager.ts +586 -0
  299. package/src/modes/components/extensions/types.ts +191 -0
  300. package/src/modes/components/footer.ts +315 -0
  301. package/src/modes/components/history-search.ts +157 -0
  302. package/src/modes/components/hook-editor.ts +101 -0
  303. package/src/modes/components/hook-input.ts +72 -0
  304. package/src/modes/components/hook-message.ts +100 -0
  305. package/src/modes/components/hook-selector.ts +155 -0
  306. package/src/modes/components/index.ts +41 -0
  307. package/src/modes/components/keybinding-hints.ts +65 -0
  308. package/src/modes/components/login-dialog.ts +164 -0
  309. package/src/modes/components/mcp-add-wizard.ts +1295 -0
  310. package/src/modes/components/model-selector.ts +625 -0
  311. package/src/modes/components/oauth-selector.ts +210 -0
  312. package/src/modes/components/plugin-settings.ts +477 -0
  313. package/src/modes/components/python-execution.ts +196 -0
  314. package/src/modes/components/queue-mode-selector.ts +56 -0
  315. package/src/modes/components/read-tool-group.ts +119 -0
  316. package/src/modes/components/session-selector.ts +242 -0
  317. package/src/modes/components/settings-defs.ts +340 -0
  318. package/src/modes/components/settings-selector.ts +529 -0
  319. package/src/modes/components/show-images-selector.ts +45 -0
  320. package/src/modes/components/skill-message.ts +90 -0
  321. package/src/modes/components/status-line/index.ts +4 -0
  322. package/src/modes/components/status-line/presets.ts +94 -0
  323. package/src/modes/components/status-line/segments.ts +352 -0
  324. package/src/modes/components/status-line/separators.ts +55 -0
  325. package/src/modes/components/status-line/types.ts +75 -0
  326. package/src/modes/components/status-line-segment-editor.ts +354 -0
  327. package/src/modes/components/status-line.ts +421 -0
  328. package/src/modes/components/theme-selector.ts +63 -0
  329. package/src/modes/components/thinking-selector.ts +64 -0
  330. package/src/modes/components/todo-display.ts +115 -0
  331. package/src/modes/components/todo-reminder.ts +40 -0
  332. package/src/modes/components/tool-execution.ts +703 -0
  333. package/src/modes/components/tree-selector.ts +904 -0
  334. package/src/modes/components/ttsr-notification.ts +80 -0
  335. package/src/modes/components/user-message-selector.ts +146 -0
  336. package/src/modes/components/user-message.ts +22 -0
  337. package/src/modes/components/visual-truncate.ts +63 -0
  338. package/src/modes/components/welcome.ts +247 -0
  339. package/src/modes/controllers/command-controller.ts +1120 -0
  340. package/src/modes/controllers/event-controller.ts +479 -0
  341. package/src/modes/controllers/extension-ui-controller.ts +778 -0
  342. package/src/modes/controllers/input-controller.ts +671 -0
  343. package/src/modes/controllers/mcp-command-controller.ts +1315 -0
  344. package/src/modes/controllers/selector-controller.ts +712 -0
  345. package/src/modes/controllers/ssh-command-controller.ts +452 -0
  346. package/src/modes/index.ts +15 -0
  347. package/src/modes/interactive-mode.ts +1027 -0
  348. package/src/modes/print-mode.ts +191 -0
  349. package/src/modes/rpc/rpc-client.ts +583 -0
  350. package/src/modes/rpc/rpc-mode.ts +700 -0
  351. package/src/modes/rpc/rpc-types.ts +236 -0
  352. package/src/modes/theme/dark.json +95 -0
  353. package/src/modes/theme/defaults/alabaster.json +93 -0
  354. package/src/modes/theme/defaults/amethyst.json +96 -0
  355. package/src/modes/theme/defaults/anthracite.json +93 -0
  356. package/src/modes/theme/defaults/basalt.json +91 -0
  357. package/src/modes/theme/defaults/birch.json +95 -0
  358. package/src/modes/theme/defaults/dark-abyss.json +91 -0
  359. package/src/modes/theme/defaults/dark-arctic.json +104 -0
  360. package/src/modes/theme/defaults/dark-aurora.json +95 -0
  361. package/src/modes/theme/defaults/dark-catppuccin.json +107 -0
  362. package/src/modes/theme/defaults/dark-cavern.json +91 -0
  363. package/src/modes/theme/defaults/dark-copper.json +95 -0
  364. package/src/modes/theme/defaults/dark-cosmos.json +90 -0
  365. package/src/modes/theme/defaults/dark-cyberpunk.json +102 -0
  366. package/src/modes/theme/defaults/dark-dracula.json +98 -0
  367. package/src/modes/theme/defaults/dark-eclipse.json +91 -0
  368. package/src/modes/theme/defaults/dark-ember.json +95 -0
  369. package/src/modes/theme/defaults/dark-equinox.json +90 -0
  370. package/src/modes/theme/defaults/dark-forest.json +96 -0
  371. package/src/modes/theme/defaults/dark-github.json +105 -0
  372. package/src/modes/theme/defaults/dark-gruvbox.json +112 -0
  373. package/src/modes/theme/defaults/dark-lavender.json +95 -0
  374. package/src/modes/theme/defaults/dark-lunar.json +89 -0
  375. package/src/modes/theme/defaults/dark-midnight.json +95 -0
  376. package/src/modes/theme/defaults/dark-monochrome.json +94 -0
  377. package/src/modes/theme/defaults/dark-monokai.json +98 -0
  378. package/src/modes/theme/defaults/dark-nebula.json +90 -0
  379. package/src/modes/theme/defaults/dark-nord.json +97 -0
  380. package/src/modes/theme/defaults/dark-ocean.json +101 -0
  381. package/src/modes/theme/defaults/dark-one.json +100 -0
  382. package/src/modes/theme/defaults/dark-rainforest.json +91 -0
  383. package/src/modes/theme/defaults/dark-reef.json +91 -0
  384. package/src/modes/theme/defaults/dark-retro.json +92 -0
  385. package/src/modes/theme/defaults/dark-rose-pine.json +96 -0
  386. package/src/modes/theme/defaults/dark-sakura.json +95 -0
  387. package/src/modes/theme/defaults/dark-slate.json +95 -0
  388. package/src/modes/theme/defaults/dark-solarized.json +97 -0
  389. package/src/modes/theme/defaults/dark-solstice.json +90 -0
  390. package/src/modes/theme/defaults/dark-starfall.json +91 -0
  391. package/src/modes/theme/defaults/dark-sunset.json +99 -0
  392. package/src/modes/theme/defaults/dark-swamp.json +90 -0
  393. package/src/modes/theme/defaults/dark-synthwave.json +103 -0
  394. package/src/modes/theme/defaults/dark-taiga.json +91 -0
  395. package/src/modes/theme/defaults/dark-terminal.json +95 -0
  396. package/src/modes/theme/defaults/dark-tokyo-night.json +101 -0
  397. package/src/modes/theme/defaults/dark-tundra.json +91 -0
  398. package/src/modes/theme/defaults/dark-twilight.json +91 -0
  399. package/src/modes/theme/defaults/dark-volcanic.json +91 -0
  400. package/src/modes/theme/defaults/graphite.json +92 -0
  401. package/src/modes/theme/defaults/index.ts +195 -0
  402. package/src/modes/theme/defaults/light-arctic.json +107 -0
  403. package/src/modes/theme/defaults/light-aurora-day.json +91 -0
  404. package/src/modes/theme/defaults/light-canyon.json +91 -0
  405. package/src/modes/theme/defaults/light-catppuccin.json +106 -0
  406. package/src/modes/theme/defaults/light-cirrus.json +90 -0
  407. package/src/modes/theme/defaults/light-coral.json +95 -0
  408. package/src/modes/theme/defaults/light-cyberpunk.json +96 -0
  409. package/src/modes/theme/defaults/light-dawn.json +90 -0
  410. package/src/modes/theme/defaults/light-dunes.json +91 -0
  411. package/src/modes/theme/defaults/light-eucalyptus.json +95 -0
  412. package/src/modes/theme/defaults/light-forest.json +100 -0
  413. package/src/modes/theme/defaults/light-frost.json +95 -0
  414. package/src/modes/theme/defaults/light-github.json +115 -0
  415. package/src/modes/theme/defaults/light-glacier.json +91 -0
  416. package/src/modes/theme/defaults/light-gruvbox.json +108 -0
  417. package/src/modes/theme/defaults/light-haze.json +90 -0
  418. package/src/modes/theme/defaults/light-honeycomb.json +95 -0
  419. package/src/modes/theme/defaults/light-lagoon.json +91 -0
  420. package/src/modes/theme/defaults/light-lavender.json +95 -0
  421. package/src/modes/theme/defaults/light-meadow.json +91 -0
  422. package/src/modes/theme/defaults/light-mint.json +95 -0
  423. package/src/modes/theme/defaults/light-monochrome.json +101 -0
  424. package/src/modes/theme/defaults/light-ocean.json +99 -0
  425. package/src/modes/theme/defaults/light-one.json +99 -0
  426. package/src/modes/theme/defaults/light-opal.json +91 -0
  427. package/src/modes/theme/defaults/light-orchard.json +91 -0
  428. package/src/modes/theme/defaults/light-paper.json +95 -0
  429. package/src/modes/theme/defaults/light-prism.json +90 -0
  430. package/src/modes/theme/defaults/light-retro.json +98 -0
  431. package/src/modes/theme/defaults/light-sand.json +95 -0
  432. package/src/modes/theme/defaults/light-savanna.json +91 -0
  433. package/src/modes/theme/defaults/light-solarized.json +102 -0
  434. package/src/modes/theme/defaults/light-soleil.json +90 -0
  435. package/src/modes/theme/defaults/light-sunset.json +99 -0
  436. package/src/modes/theme/defaults/light-synthwave.json +98 -0
  437. package/src/modes/theme/defaults/light-tokyo-night.json +111 -0
  438. package/src/modes/theme/defaults/light-wetland.json +91 -0
  439. package/src/modes/theme/defaults/light-zenith.json +89 -0
  440. package/src/modes/theme/defaults/limestone.json +94 -0
  441. package/src/modes/theme/defaults/mahogany.json +97 -0
  442. package/src/modes/theme/defaults/marble.json +93 -0
  443. package/src/modes/theme/defaults/obsidian.json +91 -0
  444. package/src/modes/theme/defaults/onyx.json +91 -0
  445. package/src/modes/theme/defaults/pearl.json +93 -0
  446. package/src/modes/theme/defaults/porcelain.json +91 -0
  447. package/src/modes/theme/defaults/quartz.json +96 -0
  448. package/src/modes/theme/defaults/sandstone.json +95 -0
  449. package/src/modes/theme/defaults/titanium.json +90 -0
  450. package/src/modes/theme/light.json +93 -0
  451. package/src/modes/theme/mermaid-cache.ts +111 -0
  452. package/src/modes/theme/theme-schema.json +429 -0
  453. package/src/modes/theme/theme.ts +2333 -0
  454. package/src/modes/types.ts +216 -0
  455. package/src/modes/utils/ui-helpers.ts +529 -0
  456. package/src/patch/applicator.ts +1482 -0
  457. package/src/patch/diff.ts +425 -0
  458. package/src/patch/fuzzy.ts +784 -0
  459. package/src/patch/hashline.ts +972 -0
  460. package/src/patch/index.ts +964 -0
  461. package/src/patch/normalize.ts +397 -0
  462. package/src/patch/normative.ts +72 -0
  463. package/src/patch/parser.ts +532 -0
  464. package/src/patch/shared.ts +400 -0
  465. package/src/patch/types.ts +292 -0
  466. package/src/priority.json +35 -0
  467. package/src/prompts/agents/explore.md +48 -0
  468. package/src/prompts/agents/frontmatter.md +9 -0
  469. package/src/prompts/agents/init.md +36 -0
  470. package/src/prompts/agents/librarian.md +53 -0
  471. package/src/prompts/agents/oracle.md +51 -0
  472. package/src/prompts/agents/reviewer.md +70 -0
  473. package/src/prompts/agents/task.md +14 -0
  474. package/src/prompts/compaction/branch-summary-context.md +5 -0
  475. package/src/prompts/compaction/branch-summary-preamble.md +2 -0
  476. package/src/prompts/compaction/branch-summary.md +30 -0
  477. package/src/prompts/compaction/compaction-short-summary.md +9 -0
  478. package/src/prompts/compaction/compaction-summary-context.md +5 -0
  479. package/src/prompts/compaction/compaction-summary.md +38 -0
  480. package/src/prompts/compaction/compaction-turn-prefix.md +17 -0
  481. package/src/prompts/compaction/compaction-update-summary.md +45 -0
  482. package/src/prompts/memories/consolidation.md +30 -0
  483. package/src/prompts/memories/read_path.md +11 -0
  484. package/src/prompts/memories/stage_one_input.md +6 -0
  485. package/src/prompts/memories/stage_one_system.md +21 -0
  486. package/src/prompts/review-request.md +64 -0
  487. package/src/prompts/system/agent-creation-architect.md +65 -0
  488. package/src/prompts/system/agent-creation-user.md +6 -0
  489. package/src/prompts/system/custom-system-prompt.md +68 -0
  490. package/src/prompts/system/file-operations.md +10 -0
  491. package/src/prompts/system/subagent-submit-reminder.md +11 -0
  492. package/src/prompts/system/subagent-system-prompt.md +31 -0
  493. package/src/prompts/system/subagent-user-prompt.md +8 -0
  494. package/src/prompts/system/summarization-system.md +3 -0
  495. package/src/prompts/system/system-prompt.md +300 -0
  496. package/src/prompts/system/title-system.md +2 -0
  497. package/src/prompts/system/ttsr-interrupt.md +7 -0
  498. package/src/prompts/system/web-search.md +28 -0
  499. package/src/prompts/tools/ask.md +44 -0
  500. package/src/prompts/tools/bash.md +24 -0
  501. package/src/prompts/tools/browser.md +33 -0
  502. package/src/prompts/tools/calculator.md +12 -0
  503. package/src/prompts/tools/explore.md +29 -0
  504. package/src/prompts/tools/fetch.md +16 -0
  505. package/src/prompts/tools/find.md +18 -0
  506. package/src/prompts/tools/gemini-image.md +23 -0
  507. package/src/prompts/tools/grep.md +28 -0
  508. package/src/prompts/tools/hashline.md +232 -0
  509. package/src/prompts/tools/librarian.md +24 -0
  510. package/src/prompts/tools/lsp.md +28 -0
  511. package/src/prompts/tools/oracle.md +26 -0
  512. package/src/prompts/tools/patch.md +74 -0
  513. package/src/prompts/tools/python.md +66 -0
  514. package/src/prompts/tools/read.md +36 -0
  515. package/src/prompts/tools/replace.md +38 -0
  516. package/src/prompts/tools/reviewer.md +41 -0
  517. package/src/prompts/tools/ssh.md +51 -0
  518. package/src/prompts/tools/task-summary.md +28 -0
  519. package/src/prompts/tools/task.md +275 -0
  520. package/src/prompts/tools/todo-write.md +65 -0
  521. package/src/prompts/tools/undo-edit.md +7 -0
  522. package/src/prompts/tools/web-search.md +19 -0
  523. package/src/prompts/tools/write.md +18 -0
  524. package/src/sdk.ts +1287 -0
  525. package/src/secrets/index.ts +116 -0
  526. package/src/secrets/obfuscator.ts +269 -0
  527. package/src/secrets/regex.ts +21 -0
  528. package/src/session/agent-session.ts +4669 -0
  529. package/src/session/agent-storage.ts +621 -0
  530. package/src/session/artifacts.ts +132 -0
  531. package/src/session/auth-storage.ts +1433 -0
  532. package/src/session/blob-store.ts +103 -0
  533. package/src/session/compaction/branch-summarization.ts +315 -0
  534. package/src/session/compaction/compaction.ts +864 -0
  535. package/src/session/compaction/index.ts +7 -0
  536. package/src/session/compaction/pruning.ts +91 -0
  537. package/src/session/compaction/utils.ts +171 -0
  538. package/src/session/history-storage.ts +170 -0
  539. package/src/session/messages.ts +317 -0
  540. package/src/session/session-manager.ts +2276 -0
  541. package/src/session/session-storage.ts +342 -0
  542. package/src/session/streaming-output.ts +565 -0
  543. package/src/slash-commands/builtin-registry.ts +439 -0
  544. package/src/ssh/config-writer.ts +183 -0
  545. package/src/ssh/connection-manager.ts +444 -0
  546. package/src/ssh/ssh-executor.ts +127 -0
  547. package/src/ssh/sshfs-mount.ts +135 -0
  548. package/src/stt/downloader.ts +71 -0
  549. package/src/stt/index.ts +3 -0
  550. package/src/stt/recorder.ts +351 -0
  551. package/src/stt/setup.ts +52 -0
  552. package/src/stt/stt-controller.ts +160 -0
  553. package/src/stt/transcribe.py +70 -0
  554. package/src/stt/transcriber.ts +91 -0
  555. package/src/system-prompt.ts +685 -0
  556. package/src/task/agents.ts +155 -0
  557. package/src/task/batch.ts +102 -0
  558. package/src/task/commands.ts +134 -0
  559. package/src/task/discovery.ts +126 -0
  560. package/src/task/executor.ts +908 -0
  561. package/src/task/index.ts +223 -0
  562. package/src/task/output-manager.ts +107 -0
  563. package/src/task/parallel.ts +84 -0
  564. package/src/task/render.ts +326 -0
  565. package/src/task/subprocess-tool-registry.ts +88 -0
  566. package/src/task/template.ts +32 -0
  567. package/src/task/types.ts +144 -0
  568. package/src/tools/ask.ts +523 -0
  569. package/src/tools/bash-interactive.ts +419 -0
  570. package/src/tools/bash-interceptor.ts +105 -0
  571. package/src/tools/bash-normalize.ts +107 -0
  572. package/src/tools/bash-skill-urls.ts +177 -0
  573. package/src/tools/bash.ts +347 -0
  574. package/src/tools/browser.ts +1374 -0
  575. package/src/tools/calculator.ts +537 -0
  576. package/src/tools/context.ts +39 -0
  577. package/src/tools/explore.ts +23 -0
  578. package/src/tools/fetch.ts +1091 -0
  579. package/src/tools/find.ts +540 -0
  580. package/src/tools/fs-cache-invalidation.ts +28 -0
  581. package/src/tools/gemini-image.ts +907 -0
  582. package/src/tools/grep.ts +489 -0
  583. package/src/tools/index.ts +337 -0
  584. package/src/tools/json-tree.ts +231 -0
  585. package/src/tools/jtd-to-json-schema.ts +247 -0
  586. package/src/tools/jtd-to-typescript.ts +198 -0
  587. package/src/tools/librarian.ts +33 -0
  588. package/src/tools/list-limit.ts +40 -0
  589. package/src/tools/notebook.ts +287 -0
  590. package/src/tools/oracle.ts +40 -0
  591. package/src/tools/output-meta.ts +459 -0
  592. package/src/tools/output-utils.ts +63 -0
  593. package/src/tools/path-utils.ts +116 -0
  594. package/src/tools/puppeteer/00_stealth_tampering.txt +63 -0
  595. package/src/tools/puppeteer/01_stealth_activity.txt +20 -0
  596. package/src/tools/puppeteer/02_stealth_hairline.txt +11 -0
  597. package/src/tools/puppeteer/03_stealth_botd.txt +384 -0
  598. package/src/tools/puppeteer/04_stealth_iframe.txt +81 -0
  599. package/src/tools/puppeteer/05_stealth_webgl.txt +75 -0
  600. package/src/tools/puppeteer/06_stealth_screen.txt +72 -0
  601. package/src/tools/puppeteer/07_stealth_fonts.txt +97 -0
  602. package/src/tools/puppeteer/08_stealth_audio.txt +51 -0
  603. package/src/tools/puppeteer/09_stealth_locale.txt +46 -0
  604. package/src/tools/puppeteer/10_stealth_plugins.txt +206 -0
  605. package/src/tools/puppeteer/11_stealth_hardware.txt +8 -0
  606. package/src/tools/puppeteer/12_stealth_codecs.txt +40 -0
  607. package/src/tools/puppeteer/13_stealth_worker.txt +74 -0
  608. package/src/tools/python.ts +1118 -0
  609. package/src/tools/read.ts +1193 -0
  610. package/src/tools/render-utils.ts +680 -0
  611. package/src/tools/renderers.ts +60 -0
  612. package/src/tools/reviewer-tool.ts +41 -0
  613. package/src/tools/ssh.ts +326 -0
  614. package/src/tools/subagent-tool.ts +169 -0
  615. package/src/tools/submit-result.ts +152 -0
  616. package/src/tools/todo-write.ts +255 -0
  617. package/src/tools/tool-errors.ts +92 -0
  618. package/src/tools/tool-result.ts +86 -0
  619. package/src/tools/undo-edit.ts +145 -0
  620. package/src/tools/undo-history.ts +22 -0
  621. package/src/tools/write.ts +274 -0
  622. package/src/tui/code-cell.ts +108 -0
  623. package/src/tui/file-list.ts +47 -0
  624. package/src/tui/index.ts +11 -0
  625. package/src/tui/output-block.ts +144 -0
  626. package/src/tui/status-line.ts +39 -0
  627. package/src/tui/tree-list.ts +53 -0
  628. package/src/tui/types.ts +16 -0
  629. package/src/tui/utils.ts +116 -0
  630. package/src/utils/changelog.ts +98 -0
  631. package/src/utils/event-bus.ts +33 -0
  632. package/src/utils/external-editor.ts +59 -0
  633. package/src/utils/file-display-mode.ts +36 -0
  634. package/src/utils/file-mentions.ts +384 -0
  635. package/src/utils/frontmatter.ts +101 -0
  636. package/src/utils/fuzzy.ts +108 -0
  637. package/src/utils/ignore-files.ts +119 -0
  638. package/src/utils/image-convert.ts +27 -0
  639. package/src/utils/image-resize.ts +236 -0
  640. package/src/utils/mime.ts +30 -0
  641. package/src/utils/open.ts +20 -0
  642. package/src/utils/shell-snapshot.ts +199 -0
  643. package/src/utils/timings.ts +26 -0
  644. package/src/utils/title-generator.ts +167 -0
  645. package/src/utils/tools-manager.ts +362 -0
  646. package/src/web/scrapers/artifacthub.ts +215 -0
  647. package/src/web/scrapers/arxiv.ts +88 -0
  648. package/src/web/scrapers/aur.ts +175 -0
  649. package/src/web/scrapers/biorxiv.ts +141 -0
  650. package/src/web/scrapers/bluesky.ts +284 -0
  651. package/src/web/scrapers/brew.ts +177 -0
  652. package/src/web/scrapers/cheatsh.ts +78 -0
  653. package/src/web/scrapers/chocolatey.ts +158 -0
  654. package/src/web/scrapers/choosealicense.ts +110 -0
  655. package/src/web/scrapers/cisa-kev.ts +100 -0
  656. package/src/web/scrapers/clojars.ts +180 -0
  657. package/src/web/scrapers/coingecko.ts +184 -0
  658. package/src/web/scrapers/crates-io.ts +128 -0
  659. package/src/web/scrapers/crossref.ts +149 -0
  660. package/src/web/scrapers/devto.ts +177 -0
  661. package/src/web/scrapers/discogs.ts +307 -0
  662. package/src/web/scrapers/discourse.ts +221 -0
  663. package/src/web/scrapers/dockerhub.ts +160 -0
  664. package/src/web/scrapers/fdroid.ts +158 -0
  665. package/src/web/scrapers/firefox-addons.ts +214 -0
  666. package/src/web/scrapers/flathub.ts +239 -0
  667. package/src/web/scrapers/github-gist.ts +68 -0
  668. package/src/web/scrapers/github.ts +490 -0
  669. package/src/web/scrapers/gitlab.ts +456 -0
  670. package/src/web/scrapers/go-pkg.ts +275 -0
  671. package/src/web/scrapers/hackage.ts +94 -0
  672. package/src/web/scrapers/hackernews.ts +208 -0
  673. package/src/web/scrapers/hex.ts +121 -0
  674. package/src/web/scrapers/huggingface.ts +385 -0
  675. package/src/web/scrapers/iacr.ts +86 -0
  676. package/src/web/scrapers/index.ts +249 -0
  677. package/src/web/scrapers/jetbrains-marketplace.ts +169 -0
  678. package/src/web/scrapers/lemmy.ts +220 -0
  679. package/src/web/scrapers/lobsters.ts +186 -0
  680. package/src/web/scrapers/mastodon.ts +310 -0
  681. package/src/web/scrapers/maven.ts +152 -0
  682. package/src/web/scrapers/mdn.ts +172 -0
  683. package/src/web/scrapers/metacpan.ts +253 -0
  684. package/src/web/scrapers/musicbrainz.ts +272 -0
  685. package/src/web/scrapers/npm.ts +114 -0
  686. package/src/web/scrapers/nuget.ts +205 -0
  687. package/src/web/scrapers/nvd.ts +243 -0
  688. package/src/web/scrapers/ollama.ts +265 -0
  689. package/src/web/scrapers/open-vsx.ts +119 -0
  690. package/src/web/scrapers/opencorporates.ts +275 -0
  691. package/src/web/scrapers/openlibrary.ts +319 -0
  692. package/src/web/scrapers/orcid.ts +298 -0
  693. package/src/web/scrapers/osv.ts +192 -0
  694. package/src/web/scrapers/packagist.ts +174 -0
  695. package/src/web/scrapers/pub-dev.ts +185 -0
  696. package/src/web/scrapers/pubmed.ts +177 -0
  697. package/src/web/scrapers/pypi.ts +129 -0
  698. package/src/web/scrapers/rawg.ts +124 -0
  699. package/src/web/scrapers/readthedocs.ts +125 -0
  700. package/src/web/scrapers/reddit.ts +104 -0
  701. package/src/web/scrapers/repology.ts +262 -0
  702. package/src/web/scrapers/rfc.ts +209 -0
  703. package/src/web/scrapers/rubygems.ts +117 -0
  704. package/src/web/scrapers/searchcode.ts +217 -0
  705. package/src/web/scrapers/sec-edgar.ts +274 -0
  706. package/src/web/scrapers/semantic-scholar.ts +190 -0
  707. package/src/web/scrapers/snapcraft.ts +200 -0
  708. package/src/web/scrapers/sourcegraph.ts +373 -0
  709. package/src/web/scrapers/spdx.ts +121 -0
  710. package/src/web/scrapers/spotify.ts +217 -0
  711. package/src/web/scrapers/stackoverflow.ts +124 -0
  712. package/src/web/scrapers/terraform.ts +304 -0
  713. package/src/web/scrapers/tldr.ts +51 -0
  714. package/src/web/scrapers/twitter.ts +97 -0
  715. package/src/web/scrapers/types.ts +200 -0
  716. package/src/web/scrapers/utils.ts +142 -0
  717. package/src/web/scrapers/vimeo.ts +152 -0
  718. package/src/web/scrapers/vscode-marketplace.ts +195 -0
  719. package/src/web/scrapers/w3c.ts +163 -0
  720. package/src/web/scrapers/wikidata.ts +357 -0
  721. package/src/web/scrapers/wikipedia.ts +95 -0
  722. package/src/web/scrapers/youtube.ts +312 -0
  723. package/src/web/search/auth.ts +178 -0
  724. package/src/web/search/index.ts +598 -0
  725. package/src/web/search/provider.ts +77 -0
  726. package/src/web/search/providers/anthropic.ts +284 -0
  727. package/src/web/search/providers/base.ts +22 -0
  728. package/src/web/search/providers/brave.ts +165 -0
  729. package/src/web/search/providers/codex.ts +377 -0
  730. package/src/web/search/providers/exa.ts +158 -0
  731. package/src/web/search/providers/gemini.ts +437 -0
  732. package/src/web/search/providers/jina.ts +99 -0
  733. package/src/web/search/providers/kimi.ts +196 -0
  734. package/src/web/search/providers/perplexity.ts +546 -0
  735. package/src/web/search/providers/synthetic.ts +136 -0
  736. package/src/web/search/providers/zai.ts +352 -0
  737. package/src/web/search/render.ts +299 -0
  738. package/src/web/search/types.ts +437 -0
@@ -0,0 +1,1482 @@
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 { resolveToCwd } from "../tools/path-utils";
11
+ import { DEFAULT_FUZZY_THRESHOLD, findClosestSequenceMatch, findContextLine, findMatch, seekSequence } from "./fuzzy";
12
+ import {
13
+ adjustIndentation,
14
+ convertLeadingTabsToSpaces,
15
+ countLeadingWhitespace,
16
+ detectLineEnding,
17
+ getLeadingWhitespace,
18
+ normalizeToLF,
19
+ restoreLineEndings,
20
+ stripBom,
21
+ } from "./normalize";
22
+ import { normalizeCreateContent, parseHunks } from "./parser";
23
+ import type {
24
+ ApplyPatchOptions,
25
+ ApplyPatchResult,
26
+ ContextLineResult,
27
+ DiffHunk,
28
+ FileSystem,
29
+ NormalizedPatchInput,
30
+ PatchInput,
31
+ } from "./types";
32
+ import { ApplyPatchError, normalizePatchInput } from "./types";
33
+
34
+ // ═══════════════════════════════════════════════════════════════════════════
35
+ // Default File System
36
+ // ═══════════════════════════════════════════════════════════════════════════
37
+
38
+ /** Default filesystem implementation using Bun APIs */
39
+ export const defaultFileSystem: FileSystem = {
40
+ async exists(path: string): Promise<boolean> {
41
+ return fs.existsSync(path);
42
+ },
43
+ async read(path: string): Promise<string> {
44
+ return Bun.file(path).text();
45
+ },
46
+ async readBinary(path: string): Promise<Uint8Array> {
47
+ const buffer = await Bun.file(path).arrayBuffer();
48
+ return new Uint8Array(buffer);
49
+ },
50
+ async write(path: string, content: string): Promise<void> {
51
+ await Bun.write(path, content);
52
+ },
53
+ async delete(path: string): Promise<void> {
54
+ await fs.promises.unlink(path);
55
+ },
56
+ async mkdir(path: string): Promise<void> {
57
+ await fs.promises.mkdir(path, { recursive: true });
58
+ },
59
+ };
60
+
61
+ // ═══════════════════════════════════════════════════════════════════════════
62
+ // Internal Types
63
+ // ═══════════════════════════════════════════════════════════════════════════
64
+
65
+ interface Replacement {
66
+ startIndex: number;
67
+ oldLen: number;
68
+ newLines: string[];
69
+ }
70
+
71
+ type HunkVariantKind = "trim-common" | "dedupe-shared" | "collapse-repeated" | "single-line";
72
+
73
+ interface HunkVariant {
74
+ oldLines: string[];
75
+ newLines: string[];
76
+ kind: HunkVariantKind;
77
+ }
78
+
79
+ // ═══════════════════════════════════════════════════════════════════════════
80
+ // Replacement Computation
81
+ // ═══════════════════════════════════════════════════════════════════════════
82
+
83
+ /** Adjust indentation of newLines to match the delta between patternLines and actualLines */
84
+ function adjustLinesIndentation(patternLines: string[], actualLines: string[], newLines: string[]): string[] {
85
+ if (patternLines.length === 0 || actualLines.length === 0 || newLines.length === 0) {
86
+ return newLines;
87
+ }
88
+
89
+ // If pattern already matches actual exactly (including indentation), preserve agent's intended changes
90
+ if (patternLines.length === actualLines.length) {
91
+ let exactMatch = true;
92
+ for (let i = 0; i < patternLines.length; i++) {
93
+ if (patternLines[i] !== actualLines[i]) {
94
+ exactMatch = false;
95
+ break;
96
+ }
97
+ }
98
+ if (exactMatch) {
99
+ return newLines;
100
+ }
101
+ }
102
+
103
+ // If the patch is purely an indentation change (same trimmed content), apply exactly as specified
104
+ if (patternLines.length === newLines.length) {
105
+ let indentationOnly = true;
106
+ for (let i = 0; i < patternLines.length; i++) {
107
+ if (patternLines[i].trim() !== newLines[i].trim()) {
108
+ indentationOnly = false;
109
+ break;
110
+ }
111
+ }
112
+ if (indentationOnly) {
113
+ return newLines;
114
+ }
115
+ }
116
+
117
+ // Detect indent character from actual content
118
+ let indentChar = " ";
119
+ for (const line of actualLines) {
120
+ const ws = getLeadingWhitespace(line);
121
+ if (ws.length > 0) {
122
+ indentChar = ws[0];
123
+ break;
124
+ }
125
+ }
126
+
127
+ let patternTabOnly = true;
128
+ let actualSpaceOnly = true;
129
+ let patternSpaceOnly = true;
130
+ let actualTabOnly = true;
131
+ let patternMixed = false;
132
+ let actualMixed = false;
133
+
134
+ for (const line of patternLines) {
135
+ if (line.trim().length === 0) continue;
136
+ const ws = getLeadingWhitespace(line);
137
+ if (ws.includes(" ")) patternTabOnly = false;
138
+ if (ws.includes("\t")) patternSpaceOnly = false;
139
+ if (ws.includes(" ") && ws.includes("\t")) patternMixed = true;
140
+ }
141
+
142
+ for (const line of actualLines) {
143
+ if (line.trim().length === 0) continue;
144
+ const ws = getLeadingWhitespace(line);
145
+ if (ws.includes("\t")) actualSpaceOnly = false;
146
+ if (ws.includes(" ")) actualTabOnly = false;
147
+ if (ws.includes(" ") && ws.includes("\t")) actualMixed = true;
148
+ }
149
+
150
+ if (!patternMixed && !actualMixed && patternTabOnly && actualSpaceOnly) {
151
+ let ratio: number | undefined;
152
+ const lineCount = Math.min(patternLines.length, actualLines.length);
153
+ let consistent = true;
154
+ for (let i = 0; i < lineCount; i++) {
155
+ const patternLine = patternLines[i];
156
+ const actualLine = actualLines[i];
157
+ if (patternLine.trim().length === 0 || actualLine.trim().length === 0) continue;
158
+ const patternIndent = countLeadingWhitespace(patternLine);
159
+ const actualIndent = countLeadingWhitespace(actualLine);
160
+ if (patternIndent === 0) continue;
161
+ if (actualIndent % patternIndent !== 0) {
162
+ consistent = false;
163
+ break;
164
+ }
165
+ const nextRatio = actualIndent / patternIndent;
166
+ if (!ratio) {
167
+ ratio = nextRatio;
168
+ } else if (ratio !== nextRatio) {
169
+ consistent = false;
170
+ break;
171
+ }
172
+ }
173
+
174
+ if (consistent && ratio) {
175
+ const converted = convertLeadingTabsToSpaces(newLines.join("\n"), ratio).split("\n");
176
+ return converted;
177
+ }
178
+ }
179
+
180
+ // Reverse: pattern uses spaces, actual uses tabs — infer spaces = tabs * width + offset
181
+ // Collect (tabs, spaces) pairs from matched lines to solve for the model's tab rendering.
182
+ // With one data point: spaces = tabs * width (offset=0).
183
+ // With two+: solve ax + b via pairs with distinct tab counts.
184
+ if (!patternMixed && !actualMixed && patternSpaceOnly && actualTabOnly) {
185
+ const samples = new Map<number, number>(); // tabs -> spaces
186
+ const lineCount = Math.min(patternLines.length, actualLines.length);
187
+ let consistent = true;
188
+ for (let i = 0; i < lineCount; i++) {
189
+ const patternLine = patternLines[i];
190
+ const actualLine = actualLines[i];
191
+ if (patternLine.trim().length === 0 || actualLine.trim().length === 0) continue;
192
+ const spaces = countLeadingWhitespace(patternLine);
193
+ const tabs = countLeadingWhitespace(actualLine);
194
+ if (tabs === 0) continue;
195
+ const existing = samples.get(tabs);
196
+ if (existing !== undefined && existing !== spaces) {
197
+ consistent = false;
198
+ break;
199
+ }
200
+ samples.set(tabs, spaces);
201
+ }
202
+
203
+ if (consistent && samples.size > 0) {
204
+ let tabWidth: number | undefined;
205
+ let offset = 0;
206
+
207
+ if (samples.size === 1) {
208
+ // One level: assume offset=0, width = spaces / tabs
209
+ const [[tabs, spaces]] = samples;
210
+ if (spaces % tabs === 0) {
211
+ tabWidth = spaces / tabs;
212
+ }
213
+ } else {
214
+ // Two+ levels: solve via any two distinct pairs
215
+ // spaces = tabs * width + offset => width = (s2 - s1) / (t2 - t1)
216
+ const entries = [...samples.entries()];
217
+ const [t1, s1] = entries[0];
218
+ const [t2, s2] = entries[1];
219
+ if (t1 !== t2) {
220
+ const w = (s2 - s1) / (t2 - t1);
221
+ if (w > 0 && Number.isInteger(w)) {
222
+ const b = s1 - t1 * w;
223
+ // Validate all samples against this model
224
+ let valid = true;
225
+ for (const [t, s] of samples) {
226
+ if (t * w + b !== s) {
227
+ valid = false;
228
+ break;
229
+ }
230
+ }
231
+ if (valid) {
232
+ tabWidth = w;
233
+ offset = b;
234
+ }
235
+ }
236
+ }
237
+ }
238
+
239
+ if (tabWidth !== undefined && tabWidth > 0) {
240
+ const converted = newLines.map(line => {
241
+ if (line.trim().length === 0) return line;
242
+ const ws = countLeadingWhitespace(line);
243
+ if (ws === 0) return line;
244
+ // Reverse: tabs = (spaces - offset) / width
245
+ const adjusted = ws - offset;
246
+ if (adjusted >= 0 && adjusted % tabWidth! === 0) {
247
+ return "\t".repeat(adjusted / tabWidth!) + line.slice(ws);
248
+ }
249
+ // Partial tab — keep remainder as spaces
250
+ const tabCount = Math.floor(adjusted / tabWidth!);
251
+ const remainder = adjusted - tabCount * tabWidth!;
252
+ if (tabCount >= 0) {
253
+ return "\t".repeat(tabCount) + " ".repeat(remainder) + line.slice(ws);
254
+ }
255
+ return line;
256
+ });
257
+ return converted;
258
+ }
259
+ }
260
+ }
261
+
262
+ // Build a map from trimmed content to actual lines (by content, not position)
263
+ // This handles fuzzy matches where pattern and actual may not be positionally aligned
264
+ const contentToActualLines = new Map<string, string[]>();
265
+ for (const line of actualLines) {
266
+ const trimmed = line.trim();
267
+ if (trimmed.length === 0) continue;
268
+ const arr = contentToActualLines.get(trimmed);
269
+ if (arr) {
270
+ arr.push(line);
271
+ } else {
272
+ contentToActualLines.set(trimmed, [line]);
273
+ }
274
+ }
275
+
276
+ let patternMin = Infinity;
277
+ for (const line of patternLines) {
278
+ if (line.trim().length === 0) continue;
279
+ patternMin = Math.min(patternMin, countLeadingWhitespace(line));
280
+ }
281
+ if (patternMin === Infinity) {
282
+ patternMin = 0;
283
+ }
284
+
285
+ let delta: number | undefined;
286
+ const deltas: number[] = [];
287
+ for (let i = 0; i < Math.min(patternLines.length, actualLines.length); i++) {
288
+ const patternLine = patternLines[i];
289
+ const actualLine = actualLines[i];
290
+ if (patternLine.trim().length === 0 || actualLine.trim().length === 0) continue;
291
+ const pIndent = countLeadingWhitespace(patternLine);
292
+ const aIndent = countLeadingWhitespace(actualLine);
293
+ deltas.push(aIndent - pIndent);
294
+ }
295
+
296
+ if (deltas.length > 0 && deltas.every(value => value === deltas[0])) {
297
+ delta = deltas[0];
298
+ }
299
+
300
+ // Track which actual lines we've used to handle duplicate content correctly
301
+ const usedActualLines = new Map<string, number>(); // trimmed content -> count used
302
+
303
+ return newLines.map(newLine => {
304
+ if (newLine.trim().length === 0) {
305
+ return newLine;
306
+ }
307
+
308
+ const trimmed = newLine.trim();
309
+ const matchingActualLines = contentToActualLines.get(trimmed);
310
+
311
+ // Check if this is a context line (same trimmed content exists in actual)
312
+ if (matchingActualLines && matchingActualLines.length > 0) {
313
+ if (matchingActualLines.length === 1) {
314
+ return matchingActualLines[0];
315
+ }
316
+ if (matchingActualLines.includes(newLine)) {
317
+ return newLine;
318
+ }
319
+ const usedCount = usedActualLines.get(trimmed) ?? 0;
320
+ if (usedCount < matchingActualLines.length) {
321
+ usedActualLines.set(trimmed, usedCount + 1);
322
+ // Use actual file content directly for context lines
323
+ return matchingActualLines[usedCount];
324
+ }
325
+ }
326
+
327
+ // This is a new/added line - apply consistent delta if safe
328
+ if (delta && delta !== 0) {
329
+ const newIndent = countLeadingWhitespace(newLine);
330
+ if (newIndent === patternMin) {
331
+ if (delta > 0) {
332
+ return indentChar.repeat(delta) + newLine;
333
+ }
334
+ const toRemove = Math.min(-delta, newIndent);
335
+ return newLine.slice(toRemove);
336
+ }
337
+ }
338
+ return newLine;
339
+ });
340
+ }
341
+
342
+ function trimCommonContext(oldLines: string[], newLines: string[]): HunkVariant | undefined {
343
+ let start = 0;
344
+ let endOld = oldLines.length;
345
+ let endNew = newLines.length;
346
+
347
+ while (start < endOld && start < endNew && oldLines[start] === newLines[start]) {
348
+ start++;
349
+ }
350
+
351
+ while (endOld > start && endNew > start && oldLines[endOld - 1] === newLines[endNew - 1]) {
352
+ endOld--;
353
+ endNew--;
354
+ }
355
+
356
+ if (start === 0 && endOld === oldLines.length && endNew === newLines.length) {
357
+ return undefined;
358
+ }
359
+
360
+ const trimmedOld = oldLines.slice(start, endOld);
361
+ const trimmedNew = newLines.slice(start, endNew);
362
+ if (trimmedOld.length === 0 && trimmedNew.length === 0) {
363
+ return undefined;
364
+ }
365
+ return { oldLines: trimmedOld, newLines: trimmedNew, kind: "trim-common" };
366
+ }
367
+
368
+ function collapseConsecutiveSharedLines(oldLines: string[], newLines: string[]): HunkVariant | undefined {
369
+ const shared = new Set(oldLines.filter(line => newLines.includes(line)));
370
+ const collapse = (lines: string[]): string[] => {
371
+ const out: string[] = [];
372
+ let i = 0;
373
+ while (i < lines.length) {
374
+ const line = lines[i];
375
+ out.push(line);
376
+ let j = i + 1;
377
+ while (j < lines.length && lines[j] === line && shared.has(line)) {
378
+ j++;
379
+ }
380
+ i = j;
381
+ }
382
+ return out;
383
+ };
384
+
385
+ const collapsedOld = collapse(oldLines);
386
+ const collapsedNew = collapse(newLines);
387
+ if (collapsedOld.length === oldLines.length && collapsedNew.length === newLines.length) {
388
+ return undefined;
389
+ }
390
+ return { oldLines: collapsedOld, newLines: collapsedNew, kind: "dedupe-shared" };
391
+ }
392
+
393
+ function collapseRepeatedBlocks(oldLines: string[], newLines: string[]): HunkVariant | undefined {
394
+ const shared = new Set(oldLines.filter(line => newLines.includes(line)));
395
+ const collapse = (lines: string[]): string[] => {
396
+ const output = [...lines];
397
+ let changed = false;
398
+ let i = 0;
399
+ while (i < output.length) {
400
+ let collapsed = false;
401
+ for (let size = Math.floor((output.length - i) / 2); size >= 2; size--) {
402
+ const first = output.slice(i, i + size);
403
+ const second = output.slice(i + size, i + size * 2);
404
+ if (first.length !== second.length || first.length === 0) continue;
405
+ if (!first.every(line => shared.has(line))) continue;
406
+ let same = true;
407
+ for (let idx = 0; idx < size; idx++) {
408
+ if (first[idx] !== second[idx]) {
409
+ same = false;
410
+ break;
411
+ }
412
+ }
413
+ if (same) {
414
+ output.splice(i + size, size);
415
+ changed = true;
416
+ collapsed = true;
417
+ break;
418
+ }
419
+ }
420
+ if (!collapsed) {
421
+ i++;
422
+ }
423
+ }
424
+ return changed ? output : lines;
425
+ };
426
+
427
+ const collapsedOld = collapse(oldLines);
428
+ const collapsedNew = collapse(newLines);
429
+ if (collapsedOld.length === oldLines.length && collapsedNew.length === newLines.length) {
430
+ return undefined;
431
+ }
432
+ return { oldLines: collapsedOld, newLines: collapsedNew, kind: "collapse-repeated" };
433
+ }
434
+
435
+ function reduceToSingleLineChange(oldLines: string[], newLines: string[]): HunkVariant | undefined {
436
+ if (oldLines.length !== newLines.length || oldLines.length === 0) return undefined;
437
+ let changedIndex: number | undefined;
438
+ for (let i = 0; i < oldLines.length; i++) {
439
+ if (oldLines[i] !== newLines[i]) {
440
+ if (changedIndex !== undefined) return undefined;
441
+ changedIndex = i;
442
+ }
443
+ }
444
+ if (changedIndex === undefined) return undefined;
445
+ return { oldLines: [oldLines[changedIndex]], newLines: [newLines[changedIndex]], kind: "single-line" };
446
+ }
447
+
448
+ function buildFallbackVariants(hunk: DiffHunk): HunkVariant[] {
449
+ const variants: HunkVariant[] = [];
450
+ const base: HunkVariant = { oldLines: hunk.oldLines, newLines: hunk.newLines, kind: "trim-common" };
451
+
452
+ const trimmed = trimCommonContext(base.oldLines, base.newLines);
453
+ if (trimmed) variants.push(trimmed);
454
+
455
+ const deduped = collapseConsecutiveSharedLines(
456
+ trimmed?.oldLines ?? base.oldLines,
457
+ trimmed?.newLines ?? base.newLines,
458
+ );
459
+ if (deduped) variants.push(deduped);
460
+
461
+ const collapsed = collapseRepeatedBlocks(
462
+ deduped?.oldLines ?? trimmed?.oldLines ?? base.oldLines,
463
+ deduped?.newLines ?? trimmed?.newLines ?? base.newLines,
464
+ );
465
+ if (collapsed) variants.push(collapsed);
466
+
467
+ const singleLine = reduceToSingleLineChange(trimmed?.oldLines ?? base.oldLines, trimmed?.newLines ?? base.newLines);
468
+ if (singleLine) variants.push(singleLine);
469
+
470
+ const seen = new Set<string>();
471
+ return variants.filter(variant => {
472
+ if (variant.oldLines.length === 0 && variant.newLines.length === 0) return false;
473
+ const key = `${variant.oldLines.join("\n")}||${variant.newLines.join("\n")}`;
474
+ if (seen.has(key)) return false;
475
+ seen.add(key);
476
+ return true;
477
+ });
478
+ }
479
+
480
+ function filterFallbackVariants(variants: HunkVariant[], allowAggressive: boolean): HunkVariant[] {
481
+ if (allowAggressive) return variants;
482
+ return variants.filter(variant => variant.kind !== "collapse-repeated" && variant.kind !== "single-line");
483
+ }
484
+
485
+ function findContextRelativeMatch(
486
+ lines: string[],
487
+ patternLine: string,
488
+ contextIndex: number,
489
+ preferSecondForwardMatch: boolean,
490
+ ): number | undefined {
491
+ const trimmed = patternLine.trim();
492
+ const forwardMatches: number[] = [];
493
+ for (let i = contextIndex + 1; i < lines.length; i++) {
494
+ if (lines[i].trim() === trimmed) {
495
+ forwardMatches.push(i);
496
+ }
497
+ }
498
+ if (forwardMatches.length > 0) {
499
+ if (preferSecondForwardMatch && forwardMatches.length > 1) {
500
+ return forwardMatches[1];
501
+ }
502
+ return forwardMatches[0];
503
+ }
504
+ for (let i = contextIndex - 1; i >= 0; i--) {
505
+ if (lines[i].trim() === trimmed) {
506
+ return i;
507
+ }
508
+ }
509
+ return undefined;
510
+ }
511
+
512
+ const AMBIGUITY_HINT_WINDOW = 200;
513
+ const MATCH_PREVIEW_CONTEXT = 2;
514
+ const MATCH_PREVIEW_MAX_LEN = 80;
515
+
516
+ function formatSequenceMatchPreview(lines: string[], startIdx: number): string {
517
+ const start = Math.max(0, startIdx - MATCH_PREVIEW_CONTEXT);
518
+ const end = Math.min(lines.length, startIdx + MATCH_PREVIEW_CONTEXT + 1);
519
+ const previewLines = lines.slice(start, end);
520
+ return previewLines
521
+ .map((line, i) => {
522
+ const num = start + i + 1;
523
+ const truncated = line.length > MATCH_PREVIEW_MAX_LEN ? `${line.slice(0, MATCH_PREVIEW_MAX_LEN - 1)}…` : line;
524
+ return ` ${num} | ${truncated}`;
525
+ })
526
+ .join("\n");
527
+ }
528
+
529
+ function formatSequenceMatchPreviews(
530
+ lines: string[],
531
+ matchIndices: number[] | undefined,
532
+ matchCount: number | undefined,
533
+ ): string | undefined {
534
+ if (!matchIndices || matchIndices.length === 0) return undefined;
535
+ const previews = matchIndices.map(index => formatSequenceMatchPreview(lines, index));
536
+ const moreMsg =
537
+ matchCount && matchCount > matchIndices.length ? ` (showing first ${matchIndices.length} of ${matchCount})` : "";
538
+ return `${previews.join("\n\n")}${moreMsg}`;
539
+ }
540
+
541
+ function chooseHintedMatch(
542
+ matchIndices: number[] | undefined,
543
+ hintIndex: number | undefined,
544
+ window: number,
545
+ ): number | undefined {
546
+ if (!matchIndices || matchIndices.length === 0 || hintIndex === undefined) return undefined;
547
+ const candidates = matchIndices.filter(index => Math.abs(index - hintIndex) <= window);
548
+ if (candidates.length === 1) return candidates[0];
549
+ return undefined;
550
+ }
551
+
552
+ /** Get hint index from hunk's line number */
553
+ function getHunkHintIndex(hunk: DiffHunk, currentIndex: number): number | undefined {
554
+ if (hunk.oldStartLine === undefined) return undefined;
555
+ const hintIndex = Math.max(0, hunk.oldStartLine - 1);
556
+ return hintIndex >= currentIndex ? hintIndex : undefined;
557
+ }
558
+
559
+ /**
560
+ * Find hierarchical context in file lines.
561
+ *
562
+ * Handles three formats:
563
+ * 1. Simple context: "function foo" - find this line
564
+ * 2. Hierarchical (newline): "class Foo\nmethod" - find class, then method after it
565
+ * 3. Hierarchical (space): "class Foo method" - try as literal first, then split and search
566
+ *
567
+ * @returns The result from finding the final (innermost) context, or undefined if not found
568
+ */
569
+ function findHierarchicalContext(
570
+ lines: string[],
571
+ context: string,
572
+ startFrom: number,
573
+ lineHint: number | undefined,
574
+ allowFuzzy: boolean,
575
+ ): ContextLineResult {
576
+ // Check for newline-separated hierarchical contexts (from nested @@ anchors)
577
+ if (context.includes("\n")) {
578
+ const parts = context
579
+ .split("\n")
580
+ .map(p => p.trim())
581
+ .filter(p => p.length > 0);
582
+ let currentStart = startFrom;
583
+
584
+ for (let i = 0; i < parts.length; i++) {
585
+ const part = parts[i];
586
+ const isLast = i === parts.length - 1;
587
+
588
+ const result = findContextLine(lines, part, currentStart, { allowFuzzy });
589
+
590
+ if (result.matchCount !== undefined && result.matchCount > 1) {
591
+ if (isLast && lineHint !== undefined) {
592
+ const hintStart = Math.max(0, lineHint - 1);
593
+ if (hintStart >= currentStart) {
594
+ const hintedResult = findContextLine(lines, part, hintStart, { allowFuzzy });
595
+ if (hintedResult.index !== undefined) {
596
+ return { ...hintedResult, matchCount: 1, matchIndices: [hintedResult.index] };
597
+ }
598
+ }
599
+ }
600
+ return {
601
+ index: undefined,
602
+ confidence: result.confidence,
603
+ matchCount: result.matchCount,
604
+ matchIndices: result.matchIndices,
605
+ strategy: result.strategy,
606
+ };
607
+ }
608
+
609
+ if (result.index === undefined) {
610
+ if (isLast && lineHint !== undefined) {
611
+ const hintStart = Math.max(0, lineHint - 1);
612
+ if (hintStart >= currentStart) {
613
+ const hintedResult = findContextLine(lines, part, hintStart, { allowFuzzy });
614
+ if (hintedResult.index !== undefined) {
615
+ return { ...hintedResult, matchCount: 1, matchIndices: [hintedResult.index] };
616
+ }
617
+ }
618
+ }
619
+ return { index: undefined, confidence: result.confidence };
620
+ }
621
+
622
+ if (isLast) {
623
+ return result;
624
+ }
625
+ currentStart = result.index + 1;
626
+ }
627
+ return { index: undefined, confidence: 0 };
628
+ }
629
+
630
+ // Try literal context first
631
+ const spaceParts = context.split(/\s+/).filter(p => p.length > 0);
632
+ const hasSignatureChars = /[(){}[\]]/.test(context);
633
+ if (!hasSignatureChars && spaceParts.length > 2) {
634
+ const outer = spaceParts.slice(0, -1).join(" ");
635
+ const inner = spaceParts[spaceParts.length - 1];
636
+ const outerResult = findContextLine(lines, outer, startFrom, { allowFuzzy });
637
+ if (outerResult.matchCount !== undefined && outerResult.matchCount > 1) {
638
+ return {
639
+ index: undefined,
640
+ confidence: outerResult.confidence,
641
+ matchCount: outerResult.matchCount,
642
+ matchIndices: outerResult.matchIndices,
643
+ strategy: outerResult.strategy,
644
+ };
645
+ }
646
+ if (outerResult.index !== undefined) {
647
+ const innerResult = findContextLine(lines, inner, outerResult.index + 1, { allowFuzzy });
648
+ if (innerResult.index !== undefined) {
649
+ return innerResult.matchCount && innerResult.matchCount > 1
650
+ ? { ...innerResult, matchCount: 1, matchIndices: [innerResult.index] }
651
+ : innerResult;
652
+ }
653
+ if (innerResult.matchCount !== undefined && innerResult.matchCount > 1) {
654
+ return {
655
+ ...innerResult,
656
+ matchCount: 1,
657
+ matchIndices: innerResult.index !== undefined ? [innerResult.index] : innerResult.matchIndices,
658
+ };
659
+ }
660
+ }
661
+ }
662
+
663
+ const result = findContextLine(lines, context, startFrom, { allowFuzzy });
664
+
665
+ // If line hint exists and result is ambiguous or missing, try from hint
666
+ if ((result.index === undefined || (result.matchCount ?? 0) > 1) && lineHint !== undefined) {
667
+ const hintStart = Math.max(0, lineHint - 1);
668
+ const hintedResult = findContextLine(lines, context, hintStart, { allowFuzzy });
669
+ if (hintedResult.index !== undefined) {
670
+ return { ...hintedResult, matchCount: 1, matchIndices: [hintedResult.index] };
671
+ }
672
+ }
673
+
674
+ // If found uniquely, return it
675
+ if (result.index !== undefined && (result.matchCount ?? 0) <= 1) {
676
+ return result;
677
+ }
678
+ if (result.matchCount !== undefined && result.matchCount > 1) {
679
+ return result;
680
+ }
681
+
682
+ // Try from beginning if not found from current position
683
+ if (result.index === undefined && startFrom !== 0) {
684
+ const fromStartResult = findContextLine(lines, context, 0, { allowFuzzy });
685
+ if (fromStartResult.index !== undefined && (fromStartResult.matchCount ?? 0) <= 1) {
686
+ return fromStartResult;
687
+ }
688
+ if (fromStartResult.matchCount !== undefined && fromStartResult.matchCount > 1) {
689
+ return fromStartResult;
690
+ }
691
+ }
692
+
693
+ // Fallback: try space-separated hierarchical matching
694
+ // e.g., "class PatchTool constructor" -> find "class PatchTool", then "constructor" after it
695
+ if (!hasSignatureChars && spaceParts.length > 1) {
696
+ const outer = spaceParts.slice(0, -1).join(" ");
697
+ const inner = spaceParts[spaceParts.length - 1];
698
+ const outerResult = findContextLine(lines, outer, startFrom, { allowFuzzy });
699
+
700
+ if (outerResult.matchCount !== undefined && outerResult.matchCount > 1) {
701
+ return {
702
+ index: undefined,
703
+ confidence: outerResult.confidence,
704
+ matchCount: outerResult.matchCount,
705
+ matchIndices: outerResult.matchIndices,
706
+ strategy: outerResult.strategy,
707
+ };
708
+ }
709
+
710
+ if (outerResult.index === undefined) {
711
+ return { index: undefined, confidence: outerResult.confidence };
712
+ }
713
+
714
+ const innerResult = findContextLine(lines, inner, outerResult.index + 1, { allowFuzzy });
715
+ if (innerResult.index !== undefined) {
716
+ return innerResult.matchCount && innerResult.matchCount > 1
717
+ ? { ...innerResult, matchCount: 1, matchIndices: [innerResult.index] }
718
+ : innerResult;
719
+ }
720
+ if (innerResult.matchCount !== undefined && innerResult.matchCount > 1) {
721
+ return {
722
+ ...innerResult,
723
+ matchCount: 1,
724
+ matchIndices: innerResult.index !== undefined ? [innerResult.index] : innerResult.matchIndices,
725
+ };
726
+ }
727
+ }
728
+
729
+ return result;
730
+ }
731
+
732
+ /** Find sequence with optional hint position, returning full search result */
733
+ function findSequenceWithHint(
734
+ lines: string[],
735
+ pattern: string[],
736
+ currentIndex: number,
737
+ hintIndex: number | undefined,
738
+ eof: boolean,
739
+ allowFuzzy: boolean,
740
+ ): import("./types").SequenceSearchResult {
741
+ // Prefer content-based search starting from currentIndex
742
+ const primaryResult = seekSequence(lines, pattern, currentIndex, eof, { allowFuzzy });
743
+ if (
744
+ primaryResult.matchCount &&
745
+ primaryResult.matchCount > 1 &&
746
+ hintIndex !== undefined &&
747
+ hintIndex !== currentIndex
748
+ ) {
749
+ const hintedResult = seekSequence(lines, pattern, hintIndex, eof, { allowFuzzy });
750
+ if (hintedResult.index !== undefined && (hintedResult.matchCount ?? 1) <= 1) {
751
+ return hintedResult;
752
+ }
753
+ if (hintedResult.matchCount && hintedResult.matchCount > 1) {
754
+ return hintedResult;
755
+ }
756
+ }
757
+ if (primaryResult.index !== undefined || (primaryResult.matchCount && primaryResult.matchCount > 1)) {
758
+ return primaryResult;
759
+ }
760
+
761
+ // Use line hint as a secondary bias only if needed
762
+ if (hintIndex !== undefined && hintIndex !== currentIndex) {
763
+ const hintedResult = seekSequence(lines, pattern, hintIndex, eof, { allowFuzzy });
764
+ if (hintedResult.index !== undefined || (hintedResult.matchCount && hintedResult.matchCount > 1)) {
765
+ return hintedResult;
766
+ }
767
+ }
768
+
769
+ // Last resort: search from beginning (handles out-of-order hunks)
770
+ if (currentIndex !== 0) {
771
+ const fromStartResult = seekSequence(lines, pattern, 0, eof, { allowFuzzy });
772
+ if (fromStartResult.index !== undefined || (fromStartResult.matchCount && fromStartResult.matchCount > 1)) {
773
+ return fromStartResult;
774
+ }
775
+ }
776
+
777
+ return primaryResult;
778
+ }
779
+
780
+ function attemptSequenceFallback(
781
+ lines: string[],
782
+ hunk: DiffHunk,
783
+ currentIndex: number,
784
+ lineHint: number | undefined,
785
+ allowFuzzy: boolean,
786
+ allowAggressiveFallbacks: boolean,
787
+ ): number | undefined {
788
+ if (hunk.oldLines.length === 0) return undefined;
789
+ const matchHint = getHunkHintIndex(hunk, currentIndex);
790
+ const fallbackResult = findSequenceWithHint(
791
+ lines,
792
+ hunk.oldLines,
793
+ currentIndex,
794
+ matchHint ?? lineHint,
795
+ false,
796
+ allowFuzzy,
797
+ );
798
+ if (fallbackResult.index !== undefined && (fallbackResult.matchCount ?? 1) <= 1) {
799
+ const nextIndex = fallbackResult.index + 1;
800
+ if (nextIndex <= lines.length - hunk.oldLines.length) {
801
+ const secondMatch = seekSequence(lines, hunk.oldLines, nextIndex, false, { allowFuzzy });
802
+ if (secondMatch.index !== undefined) {
803
+ return undefined;
804
+ }
805
+ }
806
+ return fallbackResult.index;
807
+ }
808
+
809
+ for (const variant of filterFallbackVariants(buildFallbackVariants(hunk), allowAggressiveFallbacks)) {
810
+ if (variant.oldLines.length === 0) continue;
811
+ const variantResult = findSequenceWithHint(
812
+ lines,
813
+ variant.oldLines,
814
+ currentIndex,
815
+ matchHint ?? lineHint,
816
+ false,
817
+ allowFuzzy,
818
+ );
819
+ if (variantResult.index !== undefined && (variantResult.matchCount ?? 1) <= 1) {
820
+ return variantResult.index;
821
+ }
822
+ }
823
+ return undefined;
824
+ }
825
+
826
+ /**
827
+ * Apply a hunk using character-based fuzzy matching.
828
+ * Used when the hunk contains only -/+ lines without context.
829
+ */
830
+ function applyCharacterMatch(
831
+ originalContent: string,
832
+ path: string,
833
+ hunk: DiffHunk,
834
+ fuzzyThreshold: number,
835
+ allowFuzzy: boolean,
836
+ ): { content: string; warnings: string[] } {
837
+ const oldText = hunk.oldLines.join("\n");
838
+ const newText = hunk.newLines.join("\n");
839
+
840
+ const normalizedContent = normalizeToLF(originalContent);
841
+ const normalizedOldText = normalizeToLF(oldText);
842
+
843
+ let matchOutcome = findMatch(normalizedContent, normalizedOldText, {
844
+ allowFuzzy,
845
+ threshold: fuzzyThreshold,
846
+ });
847
+ if (!matchOutcome.match && allowFuzzy) {
848
+ const relaxedThreshold = Math.min(fuzzyThreshold, 0.92);
849
+ if (relaxedThreshold < fuzzyThreshold) {
850
+ const relaxedOutcome = findMatch(normalizedContent, normalizedOldText, {
851
+ allowFuzzy,
852
+ threshold: relaxedThreshold,
853
+ });
854
+ if (relaxedOutcome.match) {
855
+ matchOutcome = relaxedOutcome;
856
+ }
857
+ }
858
+ }
859
+
860
+ // Check for multiple exact occurrences
861
+ if (matchOutcome.occurrences && matchOutcome.occurrences > 1) {
862
+ const previews = matchOutcome.occurrencePreviews?.join("\n\n") ?? "";
863
+ const moreMsg = matchOutcome.occurrences > 5 ? ` (showing first 5 of ${matchOutcome.occurrences})` : "";
864
+ throw new ApplyPatchError(
865
+ `Found ${matchOutcome.occurrences} occurrences in ${path}${moreMsg}:\n\n${previews}\n\n` +
866
+ `Add more context lines to disambiguate.`,
867
+ );
868
+ }
869
+
870
+ if (matchOutcome.fuzzyMatches && matchOutcome.fuzzyMatches > 1) {
871
+ throw new ApplyPatchError(
872
+ `Found ${matchOutcome.fuzzyMatches} high-confidence matches in ${path}. ` +
873
+ `The text must be unique. Please provide more context to make it unique.`,
874
+ );
875
+ }
876
+
877
+ if (!matchOutcome.match) {
878
+ const closest = matchOutcome.closest;
879
+ if (closest) {
880
+ const similarity = Math.round(closest.confidence * 100);
881
+ throw new ApplyPatchError(
882
+ `Could not find a close enough match in ${path}. ` +
883
+ `Closest match (${similarity}% similar) at line ${closest.startLine}.`,
884
+ );
885
+ }
886
+ throw new ApplyPatchError(`Failed to find expected lines in ${path}:\n${oldText}`);
887
+ }
888
+
889
+ // Adjust indentation to match what was actually found
890
+ const adjustedNewText = adjustIndentation(normalizedOldText, matchOutcome.match.actualText, newText);
891
+
892
+ const warnings: string[] = [];
893
+ if (matchOutcome.dominantFuzzy && matchOutcome.match) {
894
+ const similarity = Math.round(matchOutcome.match.confidence * 100);
895
+ warnings.push(
896
+ `Dominant fuzzy match selected in ${path} near line ${matchOutcome.match.startLine} (${similarity}% similar).`,
897
+ );
898
+ }
899
+
900
+ // Apply the replacement
901
+ const before = normalizedContent.substring(0, matchOutcome.match.startIndex);
902
+ const after = normalizedContent.substring(matchOutcome.match.startIndex + matchOutcome.match.actualText.length);
903
+ return { content: before + adjustedNewText + after, warnings };
904
+ }
905
+
906
+ function applyTrailingNewlinePolicy(content: string, hadFinalNewline: boolean): string {
907
+ if (hadFinalNewline) {
908
+ return content.endsWith("\n") ? content : `${content}\n`;
909
+ }
910
+ return content.replace(/\n+$/u, "");
911
+ }
912
+
913
+ /**
914
+ * Compute replacements needed to transform originalLines using the diff hunks.
915
+ */
916
+ function computeReplacements(
917
+ originalLines: string[],
918
+ path: string,
919
+ hunks: DiffHunk[],
920
+ allowFuzzy: boolean,
921
+ ): { replacements: Replacement[]; warnings: string[] } {
922
+ const replacements: Replacement[] = [];
923
+ const warnings: string[] = [];
924
+ let lineIndex = 0;
925
+
926
+ for (const hunk of hunks) {
927
+ let contextIndex: number | undefined;
928
+ if (hunk.oldStartLine !== undefined && hunk.oldStartLine < 1) {
929
+ throw new ApplyPatchError(
930
+ `Line hint ${hunk.oldStartLine} is out of range for ${path} (line numbers start at 1)`,
931
+ );
932
+ }
933
+ if (hunk.newStartLine !== undefined && hunk.newStartLine < 1) {
934
+ throw new ApplyPatchError(
935
+ `Line hint ${hunk.newStartLine} is out of range for ${path} (line numbers start at 1)`,
936
+ );
937
+ }
938
+ const lineHint = hunk.oldStartLine;
939
+ const allowAggressiveFallbacks = hunk.changeContext !== undefined || lineHint !== undefined || hunk.isEndOfFile;
940
+ if (lineHint !== undefined && hunk.changeContext === undefined && !hunk.hasContextLines) {
941
+ lineIndex = Math.max(0, Math.min(lineHint - 1, originalLines.length - 1));
942
+ }
943
+
944
+ // If hunk has a changeContext, find it and adjust lineIndex
945
+ if (hunk.changeContext !== undefined) {
946
+ // Use hierarchical context matching for nested @@ anchors and space-separated contexts
947
+ const result = findHierarchicalContext(originalLines, hunk.changeContext, lineIndex, lineHint, allowFuzzy);
948
+ const idx = result.index;
949
+ contextIndex = idx;
950
+
951
+ if (idx === undefined || (result.matchCount !== undefined && result.matchCount > 1)) {
952
+ const fallback = attemptSequenceFallback(
953
+ originalLines,
954
+ hunk,
955
+ lineIndex,
956
+ lineHint,
957
+ allowFuzzy,
958
+ allowAggressiveFallbacks,
959
+ );
960
+ if (fallback !== undefined) {
961
+ lineIndex = fallback;
962
+ } else if (result.matchCount !== undefined && result.matchCount > 1) {
963
+ const displayContext = hunk.changeContext.includes("\n")
964
+ ? hunk.changeContext.split("\n").pop()
965
+ : hunk.changeContext;
966
+ const previews = formatSequenceMatchPreviews(originalLines, result.matchIndices, result.matchCount);
967
+ const strategyHint = result.strategy ? ` Matching strategy: ${result.strategy}.` : "";
968
+ const previewText = previews ? `\n\n${previews}` : "";
969
+ throw new ApplyPatchError(
970
+ `Found ${result.matchCount} matches for context '${displayContext}' in ${path}.${strategyHint}` +
971
+ `${previewText}\n\nAdd more surrounding context or additional @@ anchors to make it unique.`,
972
+ );
973
+ } else {
974
+ const displayContext = hunk.changeContext.includes("\n")
975
+ ? hunk.changeContext.split("\n").join(" > ")
976
+ : hunk.changeContext;
977
+ throw new ApplyPatchError(`Failed to find context '${displayContext}' in ${path}`);
978
+ }
979
+ } else {
980
+ // If oldLines[0] matches the final context, start search at idx (not idx+1)
981
+ // This handles the common case where @@ scope and first context line are identical
982
+ const firstOldLine = hunk.oldLines[0];
983
+ const finalContext = hunk.changeContext.includes("\n")
984
+ ? hunk.changeContext.split("\n").pop()?.trim()
985
+ : hunk.changeContext.trim();
986
+ const isHierarchicalContext =
987
+ hunk.changeContext.includes("\n") || hunk.changeContext.trim().split(/\s+/).length > 2;
988
+ if (firstOldLine !== undefined && (firstOldLine.trim() === finalContext || isHierarchicalContext)) {
989
+ lineIndex = idx;
990
+ } else {
991
+ lineIndex = idx + 1;
992
+ }
993
+ }
994
+ }
995
+
996
+ if (hunk.oldLines.length === 0) {
997
+ // Pure addition - prefer changeContext position, then line hint, then end of file
998
+ let insertionIdx: number;
999
+ if (hunk.changeContext !== undefined) {
1000
+ // changeContext was processed above; lineIndex is set to the context line or after it
1001
+ insertionIdx = lineIndex;
1002
+ } else {
1003
+ const lineHintForInsertion = hunk.oldStartLine ?? hunk.newStartLine;
1004
+ if (lineHintForInsertion !== undefined) {
1005
+ // Reject if line hint is out of range for insertion
1006
+ // Valid insertion points are 1 to (file length + 1) for 1-indexed hints
1007
+ if (lineHintForInsertion < 1) {
1008
+ throw new ApplyPatchError(
1009
+ `Line hint ${lineHintForInsertion} is out of range for insertion in ${path} ` +
1010
+ `(line numbers start at 1)`,
1011
+ );
1012
+ }
1013
+ if (lineHintForInsertion > originalLines.length + 1) {
1014
+ throw new ApplyPatchError(
1015
+ `Line hint ${lineHintForInsertion} is out of range for insertion in ${path} ` +
1016
+ `(file has ${originalLines.length} lines)`,
1017
+ );
1018
+ }
1019
+ insertionIdx = Math.max(0, lineHintForInsertion - 1);
1020
+ } else {
1021
+ insertionIdx =
1022
+ originalLines.length > 0 && originalLines[originalLines.length - 1] === ""
1023
+ ? originalLines.length - 1
1024
+ : originalLines.length;
1025
+ }
1026
+ }
1027
+
1028
+ replacements.push({ startIndex: insertionIdx, oldLen: 0, newLines: [...hunk.newLines] });
1029
+ continue;
1030
+ }
1031
+
1032
+ // Try to find the old lines in the file
1033
+ let pattern = [...hunk.oldLines];
1034
+ const matchHint = getHunkHintIndex(hunk, lineIndex);
1035
+ let searchResult = findSequenceWithHint(
1036
+ originalLines,
1037
+ pattern,
1038
+ lineIndex,
1039
+ matchHint,
1040
+ hunk.isEndOfFile,
1041
+ allowFuzzy,
1042
+ );
1043
+ let newSlice = [...hunk.newLines];
1044
+
1045
+ // Retry without trailing empty line if present
1046
+ if (searchResult.index === undefined && pattern.length > 0 && pattern[pattern.length - 1] === "") {
1047
+ pattern = pattern.slice(0, -1);
1048
+ if (newSlice.length > 0 && newSlice[newSlice.length - 1] === "") {
1049
+ newSlice = newSlice.slice(0, -1);
1050
+ }
1051
+ searchResult = findSequenceWithHint(
1052
+ originalLines,
1053
+ pattern,
1054
+ lineIndex,
1055
+ matchHint,
1056
+ hunk.isEndOfFile,
1057
+ allowFuzzy,
1058
+ );
1059
+ }
1060
+
1061
+ if (searchResult.index === undefined || (searchResult.matchCount ?? 0) > 1) {
1062
+ for (const variant of filterFallbackVariants(buildFallbackVariants(hunk), allowAggressiveFallbacks)) {
1063
+ if (variant.oldLines.length === 0) continue;
1064
+ const variantResult = findSequenceWithHint(
1065
+ originalLines,
1066
+ variant.oldLines,
1067
+ lineIndex,
1068
+ matchHint,
1069
+ hunk.isEndOfFile,
1070
+ allowFuzzy,
1071
+ );
1072
+ if (variantResult.index !== undefined && (variantResult.matchCount ?? 1) <= 1) {
1073
+ pattern = variant.oldLines;
1074
+ newSlice = variant.newLines;
1075
+ searchResult = variantResult;
1076
+ break;
1077
+ }
1078
+ }
1079
+ }
1080
+
1081
+ if (searchResult.index === undefined && contextIndex !== undefined) {
1082
+ for (const variant of filterFallbackVariants(buildFallbackVariants(hunk), allowAggressiveFallbacks)) {
1083
+ if (variant.oldLines.length !== 1 || variant.newLines.length !== 1) continue;
1084
+ const removedLine = variant.oldLines[0];
1085
+ const hasSharedDuplicate = hunk.newLines.some(line => line.trim() === removedLine.trim());
1086
+ const adjacentIndex = findContextRelativeMatch(
1087
+ originalLines,
1088
+ removedLine,
1089
+ contextIndex,
1090
+ hasSharedDuplicate,
1091
+ );
1092
+ if (adjacentIndex !== undefined) {
1093
+ pattern = variant.oldLines;
1094
+ newSlice = variant.newLines;
1095
+ searchResult = { index: adjacentIndex, confidence: 0.95 };
1096
+ break;
1097
+ }
1098
+ }
1099
+ }
1100
+
1101
+ if (searchResult.index !== undefined && contextIndex !== undefined && pattern.length === 1) {
1102
+ const trimmed = pattern[0].trim();
1103
+ let occurrenceCount = 0;
1104
+ for (const line of originalLines) {
1105
+ if (line.trim() === trimmed) occurrenceCount++;
1106
+ }
1107
+ if (occurrenceCount > 1) {
1108
+ const hasSharedDuplicate = hunk.newLines.some(line => line.trim() === trimmed);
1109
+ const contextMatch = findContextRelativeMatch(originalLines, pattern[0], contextIndex, hasSharedDuplicate);
1110
+ if (contextMatch !== undefined) {
1111
+ searchResult = { index: contextMatch, confidence: searchResult.confidence ?? 0.95 };
1112
+ }
1113
+ }
1114
+ }
1115
+
1116
+ if ((searchResult.matchCount ?? 0) > 1) {
1117
+ const hintIndex = matchHint ?? (lineHint ? lineHint - 1 : undefined);
1118
+ const hinted = chooseHintedMatch(searchResult.matchIndices, hintIndex, AMBIGUITY_HINT_WINDOW);
1119
+ if (hinted !== undefined) {
1120
+ searchResult = { ...searchResult, index: hinted, matchCount: 1 };
1121
+ }
1122
+ }
1123
+
1124
+ if (searchResult.index === undefined) {
1125
+ if (searchResult.matchCount !== undefined && searchResult.matchCount > 1) {
1126
+ const previews = formatSequenceMatchPreviews(
1127
+ originalLines,
1128
+ searchResult.matchIndices,
1129
+ searchResult.matchCount,
1130
+ );
1131
+ const strategyHint = searchResult.strategy ? ` Matching strategy: ${searchResult.strategy}.` : "";
1132
+ const previewText = previews ? `\n\n${previews}` : "";
1133
+ throw new ApplyPatchError(
1134
+ `Found ${searchResult.matchCount} matches for the text in ${path}.${strategyHint}` +
1135
+ `${previewText}\n\nAdd more surrounding context or additional @@ anchors to make it unique.`,
1136
+ );
1137
+ }
1138
+ const closest = findClosestSequenceMatch(originalLines, pattern, {
1139
+ start: lineIndex,
1140
+ eof: hunk.isEndOfFile,
1141
+ });
1142
+ if (closest.index !== undefined && closest.confidence > 0) {
1143
+ const similarity = Math.round(closest.confidence * 100);
1144
+ const preview = formatSequenceMatchPreview(originalLines, closest.index);
1145
+ throw new ApplyPatchError(
1146
+ `Failed to find expected lines in ${path}:\n${hunk.oldLines.join("\n")}\n\n` +
1147
+ `Closest match (${similarity}% similar) near line ${closest.index + 1}:\n${preview}`,
1148
+ );
1149
+ }
1150
+ throw new ApplyPatchError(`Failed to find expected lines in ${path}:\n${hunk.oldLines.join("\n")}`);
1151
+ }
1152
+
1153
+ const found = searchResult.index;
1154
+
1155
+ if (searchResult.strategy === "fuzzy-dominant") {
1156
+ const similarity = Math.round(searchResult.confidence * 100);
1157
+ warnings.push(`Dominant fuzzy match selected in ${path} near line ${found + 1} (${similarity}% similar).`);
1158
+ }
1159
+
1160
+ // Reject if match is ambiguous (prefix/substring matching found multiple matches)
1161
+ if (searchResult.matchCount !== undefined && searchResult.matchCount > 1) {
1162
+ const previews = formatSequenceMatchPreviews(
1163
+ originalLines,
1164
+ searchResult.matchIndices,
1165
+ searchResult.matchCount,
1166
+ );
1167
+ const strategyHint = searchResult.strategy ? ` Matching strategy: ${searchResult.strategy}.` : "";
1168
+ const previewText = previews ? `\n\n${previews}` : "";
1169
+ throw new ApplyPatchError(
1170
+ `Found ${searchResult.matchCount} matches for the text in ${path}.${strategyHint}` +
1171
+ `${previewText}\n\nAdd more surrounding context or additional @@ anchors to make it unique.`,
1172
+ );
1173
+ }
1174
+
1175
+ // For simple diffs (no context marker, no context lines), check for multiple occurrences
1176
+ // This ensures ambiguous replacements are rejected
1177
+ // Skip this check if isEndOfFile is set (EOF marker provides disambiguation)
1178
+ if (hunk.changeContext === undefined && !hunk.hasContextLines && !hunk.isEndOfFile && lineHint === undefined) {
1179
+ const secondMatch = seekSequence(originalLines, pattern, found + 1, false, { allowFuzzy });
1180
+ if (secondMatch.index !== undefined) {
1181
+ // Extract 3-line previews for each match
1182
+ const formatPreview = (startIdx: number) => {
1183
+ const contextLines = 2;
1184
+ const maxLineLength = 80;
1185
+ const start = Math.max(0, startIdx - contextLines);
1186
+ const end = Math.min(originalLines.length, startIdx + contextLines + 1);
1187
+ const lines = originalLines.slice(start, end);
1188
+ return lines
1189
+ .map((line, i) => {
1190
+ const num = start + i + 1;
1191
+ const truncated = line.length > maxLineLength ? `${line.slice(0, maxLineLength - 1)}…` : line;
1192
+ return ` ${num} | ${truncated}`;
1193
+ })
1194
+ .join("\n");
1195
+ };
1196
+ const preview1 = formatPreview(found);
1197
+ const preview2 = formatPreview(secondMatch.index);
1198
+ throw new ApplyPatchError(
1199
+ `Found 2 occurrences in ${path}:\n\n${preview1}\n\n${preview2}\n\n` +
1200
+ `Add more context lines to disambiguate.`,
1201
+ );
1202
+ }
1203
+ }
1204
+
1205
+ // Adjust indentation if needed (handles fuzzy matches where indentation differs)
1206
+ const actualMatchedLines = originalLines.slice(found, found + pattern.length);
1207
+
1208
+ // Skip pure-context hunks (no +/- lines — oldLines === newLines).
1209
+ // They serve only to advance lineIndex for subsequent hunks.
1210
+ let isNoOp = pattern.length === newSlice.length;
1211
+ if (isNoOp) {
1212
+ for (let i = 0; i < pattern.length; i++) {
1213
+ if (pattern[i] !== newSlice[i]) {
1214
+ isNoOp = false;
1215
+ break;
1216
+ }
1217
+ }
1218
+ }
1219
+
1220
+ if (isNoOp) {
1221
+ lineIndex = found + pattern.length;
1222
+ continue;
1223
+ }
1224
+
1225
+ const adjustedNewLines = adjustLinesIndentation(pattern, actualMatchedLines, newSlice);
1226
+ replacements.push({ startIndex: found, oldLen: pattern.length, newLines: adjustedNewLines });
1227
+ lineIndex = found + pattern.length;
1228
+ }
1229
+
1230
+ // Sort by start index
1231
+ replacements.sort((a, b) => a.startIndex - b.startIndex);
1232
+
1233
+ for (let i = 1; i < replacements.length; i++) {
1234
+ const prev = replacements[i - 1];
1235
+ const next = replacements[i];
1236
+ const prevEnd = prev.startIndex + prev.oldLen;
1237
+ if (next.startIndex < prevEnd) {
1238
+ const formatRange = (replacement: Replacement): string => {
1239
+ if (replacement.oldLen === 0) {
1240
+ return `${replacement.startIndex + 1} (insertion)`;
1241
+ }
1242
+ return `${replacement.startIndex + 1}-${replacement.startIndex + replacement.oldLen}`;
1243
+ };
1244
+ const prevRange = formatRange(prev);
1245
+ const nextRange = formatRange(next);
1246
+ throw new ApplyPatchError(
1247
+ `Overlapping hunks detected in ${path} at lines ${prevRange} and ${nextRange}. ` +
1248
+ `Split hunks or add more context to avoid overlap.`,
1249
+ );
1250
+ }
1251
+ }
1252
+
1253
+ return { replacements, warnings };
1254
+ }
1255
+
1256
+ /**
1257
+ * Apply replacements to lines, returning the modified content.
1258
+ */
1259
+ function applyReplacements(lines: string[], replacements: Replacement[]): string[] {
1260
+ const result = [...lines];
1261
+
1262
+ // Apply in reverse order to maintain indices
1263
+ for (let i = replacements.length - 1; i >= 0; i--) {
1264
+ const { startIndex, oldLen, newLines } = replacements[i];
1265
+ result.splice(startIndex, oldLen);
1266
+ result.splice(startIndex, 0, ...newLines);
1267
+ }
1268
+
1269
+ return result;
1270
+ }
1271
+
1272
+ /**
1273
+ * Apply diff hunks to file content.
1274
+ */
1275
+ function applyHunksToContent(
1276
+ originalContent: string,
1277
+ path: string,
1278
+ hunks: DiffHunk[],
1279
+ fuzzyThreshold: number,
1280
+ allowFuzzy: boolean,
1281
+ ): { content: string; warnings: string[] } {
1282
+ const hadFinalNewline = originalContent.endsWith("\n");
1283
+
1284
+ // Detect simple replace pattern: single hunk, no @@ context, no context lines, has old lines to match
1285
+ // Only use character-based matching when there are no hints to disambiguate
1286
+ if (hunks.length === 1) {
1287
+ const hunk = hunks[0];
1288
+ if (
1289
+ hunk.changeContext === undefined &&
1290
+ !hunk.hasContextLines &&
1291
+ hunk.oldLines.length > 0 &&
1292
+ hunk.oldStartLine === undefined && // No line hint to use for positioning
1293
+ !hunk.isEndOfFile // No EOF targeting (prefer end of file)
1294
+ ) {
1295
+ const { content, warnings } = applyCharacterMatch(originalContent, path, hunk, fuzzyThreshold, allowFuzzy);
1296
+ return { content: applyTrailingNewlinePolicy(content, hadFinalNewline), warnings };
1297
+ }
1298
+ }
1299
+
1300
+ let originalLines = originalContent.split("\n");
1301
+
1302
+ // Track if we have a trailing empty element from the final newline
1303
+ // Only strip ONE trailing empty (the newline marker), preserve actual blank lines
1304
+ let strippedTrailingEmpty = false;
1305
+ if (hadFinalNewline && originalLines.length > 0 && originalLines[originalLines.length - 1] === "") {
1306
+ // Check if the second-to-last is also empty (actual blank line) - if so, only strip one
1307
+ originalLines = originalLines.slice(0, -1);
1308
+ strippedTrailingEmpty = true;
1309
+ }
1310
+
1311
+ const { replacements, warnings } = computeReplacements(originalLines, path, hunks, allowFuzzy);
1312
+ const newLines = applyReplacements(originalLines, replacements);
1313
+
1314
+ // Restore the trailing empty element if we stripped it
1315
+ if (strippedTrailingEmpty) {
1316
+ newLines.push("");
1317
+ }
1318
+
1319
+ const content = newLines.join("\n");
1320
+
1321
+ // Preserve original trailing newline behavior
1322
+ if (hadFinalNewline && !content.endsWith("\n")) {
1323
+ return { content: `${content}\n`, warnings };
1324
+ }
1325
+ if (!hadFinalNewline && content.endsWith("\n")) {
1326
+ return { content: content.slice(0, -1), warnings };
1327
+ }
1328
+ return { content, warnings };
1329
+ }
1330
+
1331
+ // ═══════════════════════════════════════════════════════════════════════════
1332
+ // Public API
1333
+ // ═══════════════════════════════════════════════════════════════════════════
1334
+
1335
+ /**
1336
+ * Apply a patch operation to the filesystem.
1337
+ */
1338
+ export async function applyPatch(input: PatchInput, options: ApplyPatchOptions): Promise<ApplyPatchResult> {
1339
+ const normalized = normalizePatchInput(input);
1340
+ return applyNormalizedPatch(normalized, options);
1341
+ }
1342
+
1343
+ /**
1344
+ * Apply a normalized patch operation to the filesystem.
1345
+ * @internal
1346
+ */
1347
+ async function applyNormalizedPatch(
1348
+ input: NormalizedPatchInput,
1349
+ options: ApplyPatchOptions,
1350
+ ): Promise<ApplyPatchResult> {
1351
+ const {
1352
+ cwd,
1353
+ dryRun = false,
1354
+ fs = defaultFileSystem,
1355
+ fuzzyThreshold = DEFAULT_FUZZY_THRESHOLD,
1356
+ allowFuzzy = true,
1357
+ } = options;
1358
+
1359
+ const resolvePath = (p: string): string => resolveToCwd(p, cwd);
1360
+ const absolutePath = resolvePath(input.path);
1361
+
1362
+ if (input.rename) {
1363
+ const destPath = resolvePath(input.rename);
1364
+ if (destPath === absolutePath) {
1365
+ throw new ApplyPatchError("rename path is the same as source path");
1366
+ }
1367
+ }
1368
+
1369
+ // Handle CREATE operation
1370
+ if (input.op === "create") {
1371
+ if (!input.diff) {
1372
+ throw new ApplyPatchError("Create operation requires diff (file content)");
1373
+ }
1374
+ // Strip + prefixes if present (handles diffs formatted as additions)
1375
+ const normalizedContent = normalizeCreateContent(input.diff);
1376
+ const content = normalizedContent.endsWith("\n") ? normalizedContent : `${normalizedContent}\n`;
1377
+
1378
+ if (!dryRun) {
1379
+ const parentDir = path.dirname(absolutePath);
1380
+ if (parentDir && parentDir !== ".") {
1381
+ await fs.mkdir(parentDir);
1382
+ }
1383
+ await fs.write(absolutePath, content);
1384
+ }
1385
+
1386
+ return {
1387
+ change: {
1388
+ type: "create",
1389
+ path: absolutePath,
1390
+ newContent: content,
1391
+ },
1392
+ };
1393
+ }
1394
+
1395
+ // Handle DELETE operation
1396
+ if (input.op === "delete") {
1397
+ if (!(await fs.exists(absolutePath))) {
1398
+ throw new ApplyPatchError(`File not found: ${input.path}`);
1399
+ }
1400
+
1401
+ const oldContent = await fs.read(absolutePath);
1402
+ if (!dryRun) {
1403
+ await fs.delete(absolutePath);
1404
+ }
1405
+
1406
+ return {
1407
+ change: {
1408
+ type: "delete",
1409
+ path: absolutePath,
1410
+ oldContent,
1411
+ },
1412
+ };
1413
+ }
1414
+
1415
+ // Handle UPDATE operation
1416
+ if (!input.diff) {
1417
+ throw new ApplyPatchError("Update operation requires diff (hunks)");
1418
+ }
1419
+
1420
+ if (!(await fs.exists(absolutePath))) {
1421
+ throw new ApplyPatchError(`File not found: ${input.path}`);
1422
+ }
1423
+
1424
+ const originalContent = await fs.read(absolutePath);
1425
+ const { bom: bomFromText, text: strippedContent } = stripBom(originalContent);
1426
+ let bom = bomFromText;
1427
+ if (!bom && fs.readBinary) {
1428
+ const bytes = await fs.readBinary(absolutePath);
1429
+ if (bytes.length >= 3 && bytes[0] === 0xef && bytes[1] === 0xbb && bytes[2] === 0xbf) {
1430
+ bom = "\uFEFF";
1431
+ }
1432
+ }
1433
+ const lineEnding = detectLineEnding(strippedContent);
1434
+ const normalizedContent = normalizeToLF(strippedContent);
1435
+ const hunks = parseHunks(input.diff);
1436
+
1437
+ if (hunks.length === 0) {
1438
+ throw new ApplyPatchError("Diff contains no hunks");
1439
+ }
1440
+
1441
+ const { content: newContent, warnings } = applyHunksToContent(
1442
+ normalizedContent,
1443
+ input.path,
1444
+ hunks,
1445
+ fuzzyThreshold,
1446
+ allowFuzzy,
1447
+ );
1448
+ const finalContent = bom + restoreLineEndings(newContent, lineEnding);
1449
+ const destPath = input.rename ? resolvePath(input.rename) : absolutePath;
1450
+ const isMove = Boolean(input.rename) && destPath !== absolutePath;
1451
+
1452
+ if (!dryRun) {
1453
+ if (isMove) {
1454
+ const parentDir = path.dirname(destPath);
1455
+ if (parentDir && parentDir !== ".") {
1456
+ await fs.mkdir(parentDir);
1457
+ }
1458
+ await fs.write(destPath, finalContent);
1459
+ await fs.delete(absolutePath);
1460
+ } else {
1461
+ await fs.write(absolutePath, finalContent);
1462
+ }
1463
+ }
1464
+
1465
+ return {
1466
+ change: {
1467
+ type: "update",
1468
+ path: absolutePath,
1469
+ newPath: isMove ? destPath : undefined,
1470
+ oldContent: originalContent,
1471
+ newContent: finalContent,
1472
+ },
1473
+ warnings: warnings.length > 0 ? warnings : undefined,
1474
+ };
1475
+ }
1476
+
1477
+ /**
1478
+ * Preview what changes a patch would make without applying it.
1479
+ */
1480
+ export async function previewPatch(input: PatchInput, options: ApplyPatchOptions): Promise<ApplyPatchResult> {
1481
+ return applyPatch(input, { ...options, dryRun: true });
1482
+ }