rimuru-ai 1.19.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 (470) hide show
  1. package/.rimuru/AGENTS.md +30 -0
  2. package/.rimuru/agents/backend.md +27 -0
  3. package/.rimuru/agents/database.md +31 -0
  4. package/.rimuru/agents/devops.md +30 -0
  5. package/.rimuru/agents/document-prep.md +49 -0
  6. package/.rimuru/agents/erp-architect.md +41 -0
  7. package/.rimuru/agents/ethical-hacking.md +49 -0
  8. package/.rimuru/agents/frontend.md +31 -0
  9. package/.rimuru/agents/fullstack.md +24 -0
  10. package/.rimuru/agents/system-engineer.md +31 -0
  11. package/.rimuru/agents/veldora-agent-tool-dev.md +30 -0
  12. package/.rimuru/agents/veldora-backend-dev.md +32 -0
  13. package/.rimuru/agents/veldora-cicd.md +32 -0
  14. package/.rimuru/agents/veldora-database.md +32 -0
  15. package/.rimuru/agents/veldora-doc.md +87 -0
  16. package/.rimuru/agents/veldora-frontend-dev.md +32 -0
  17. package/.rimuru/agents/veldora-great-sage.md +33 -0
  18. package/.rimuru/agents/veldora-mcp-creator.md +30 -0
  19. package/.rimuru/agents/veldora-pro.md +224 -0
  20. package/.rimuru/agents/veldora-prompt-enhancer.md +27 -0
  21. package/.rimuru/agents/veldora-skill-creator.md +28 -0
  22. package/.rimuru/agents/veldora.md +225 -0
  23. package/.rimuru/agents/veldorapro-agent-tool-dev.md +29 -0
  24. package/.rimuru/agents/veldorapro-backend-dev.md +29 -0
  25. package/.rimuru/agents/veldorapro-cicd.md +29 -0
  26. package/.rimuru/agents/veldorapro-database.md +29 -0
  27. package/.rimuru/agents/veldorapro-frontend-dev.md +29 -0
  28. package/.rimuru/agents/veldorapro-great-sage.md +33 -0
  29. package/.rimuru/agents/veldorapro-mcp-creator.md +27 -0
  30. package/.rimuru/agents/veldorapro-prompt-enhancer.md +25 -0
  31. package/.rimuru/agents/veldorapro-skill-creator.md +27 -0
  32. package/.rimuru/command/ai-deps.md +24 -0
  33. package/.rimuru/command/changelog.md +49 -0
  34. package/.rimuru/command/commit.md +37 -0
  35. package/.rimuru/command/issues.md +23 -0
  36. package/.rimuru/command/learn.md +42 -0
  37. package/.rimuru/command/rmslop.md +15 -0
  38. package/.rimuru/command/spellcheck.md +5 -0
  39. package/.rimuru/command/translate.md +14 -0
  40. package/.rimuru/glossary/README.md +63 -0
  41. package/.rimuru/glossary/ar.md +28 -0
  42. package/.rimuru/glossary/br.md +34 -0
  43. package/.rimuru/glossary/bs.md +33 -0
  44. package/.rimuru/glossary/da.md +27 -0
  45. package/.rimuru/glossary/de.md +27 -0
  46. package/.rimuru/glossary/es.md +27 -0
  47. package/.rimuru/glossary/fr.md +27 -0
  48. package/.rimuru/glossary/ja.md +33 -0
  49. package/.rimuru/glossary/ko.md +27 -0
  50. package/.rimuru/glossary/no.md +38 -0
  51. package/.rimuru/glossary/pl.md +27 -0
  52. package/.rimuru/glossary/ru.md +27 -0
  53. package/.rimuru/glossary/th.md +34 -0
  54. package/.rimuru/glossary/tr.md +38 -0
  55. package/.rimuru/glossary/zh-cn.md +42 -0
  56. package/.rimuru/glossary/zh-tw.md +42 -0
  57. package/.rimuru/improver/changelog.md +250 -0
  58. package/.rimuru/improver/knowledge.md +172 -0
  59. package/.rimuru/improver/plugins.md +21 -0
  60. package/.rimuru/improver/skills.md +60 -0
  61. package/.rimuru/improver/token-audit.md +21 -0
  62. package/.rimuru/opencode.jsonc +140 -0
  63. package/.rimuru/plugins/smoke-theme.json +223 -0
  64. package/.rimuru/plugins/tui-smoke.tsx +1019 -0
  65. package/.rimuru/skills/effect/SKILL.md +38 -0
  66. package/.rimuru/themes/mytheme.json +223 -0
  67. package/.rimuru/tool/github-pr-search.ts +64 -0
  68. package/.rimuru/tool/github-triage.ts +60 -0
  69. package/README.md +31 -0
  70. package/package.json +167 -0
  71. package/src/account/account.ts +463 -0
  72. package/src/account/repo.ts +173 -0
  73. package/src/account/schema.ts +99 -0
  74. package/src/account/url.ts +8 -0
  75. package/src/acp/agent.ts +95 -0
  76. package/src/acp/config-option.ts +203 -0
  77. package/src/acp/content.ts +250 -0
  78. package/src/acp/directory.ts +210 -0
  79. package/src/acp/error.ts +90 -0
  80. package/src/acp/event.ts +342 -0
  81. package/src/acp/permission.ts +124 -0
  82. package/src/acp/profile.ts +42 -0
  83. package/src/acp/service.ts +1048 -0
  84. package/src/acp/session.ts +231 -0
  85. package/src/acp/tool.ts +367 -0
  86. package/src/acp/usage.ts +232 -0
  87. package/src/agent/agent.ts +459 -0
  88. package/src/agent/generate.txt +75 -0
  89. package/src/agent/prompt/compaction.txt +9 -0
  90. package/src/agent/prompt/explore.txt +18 -0
  91. package/src/agent/prompt/summary.txt +11 -0
  92. package/src/agent/prompt/title.txt +44 -0
  93. package/src/agent/subagent-permissions.ts +27 -0
  94. package/src/audio.d.ts +14 -0
  95. package/src/auth/index.ts +99 -0
  96. package/src/background/job.ts +39 -0
  97. package/src/bus/global.ts +22 -0
  98. package/src/cli/bootstrap.ts +11 -0
  99. package/src/cli/cmd/account.ts +264 -0
  100. package/src/cli/cmd/acp.ts +73 -0
  101. package/src/cli/cmd/agent.ts +259 -0
  102. package/src/cli/cmd/attach.ts +97 -0
  103. package/src/cli/cmd/cmd.ts +7 -0
  104. package/src/cli/cmd/db.ts +62 -0
  105. package/src/cli/cmd/debug/agent.handler.ts +193 -0
  106. package/src/cli/cmd/debug/agent.ts +27 -0
  107. package/src/cli/cmd/debug/config.ts +14 -0
  108. package/src/cli/cmd/debug/file.ts +73 -0
  109. package/src/cli/cmd/debug/index.ts +87 -0
  110. package/src/cli/cmd/debug/lsp.ts +50 -0
  111. package/src/cli/cmd/debug/ripgrep.ts +79 -0
  112. package/src/cli/cmd/debug/scrap.ts +15 -0
  113. package/src/cli/cmd/debug/skill.ts +15 -0
  114. package/src/cli/cmd/debug/snapshot.ts +50 -0
  115. package/src/cli/cmd/debug/startup.ts +11 -0
  116. package/src/cli/cmd/debug/v2.ts +49 -0
  117. package/src/cli/cmd/export.ts +292 -0
  118. package/src/cli/cmd/generate.ts +54 -0
  119. package/src/cli/cmd/github.handler.ts +1593 -0
  120. package/src/cli/cmd/github.shared.ts +30 -0
  121. package/src/cli/cmd/github.ts +42 -0
  122. package/src/cli/cmd/import.ts +224 -0
  123. package/src/cli/cmd/mcp.ts +850 -0
  124. package/src/cli/cmd/models.ts +66 -0
  125. package/src/cli/cmd/plug.ts +230 -0
  126. package/src/cli/cmd/pr.ts +115 -0
  127. package/src/cli/cmd/prompt-display.ts +1 -0
  128. package/src/cli/cmd/providers.ts +534 -0
  129. package/src/cli/cmd/run/demo.ts +1274 -0
  130. package/src/cli/cmd/run/entry.body.ts +205 -0
  131. package/src/cli/cmd/run/footer.command.tsx +1064 -0
  132. package/src/cli/cmd/run/footer.menu.tsx +351 -0
  133. package/src/cli/cmd/run/footer.permission.tsx +472 -0
  134. package/src/cli/cmd/run/footer.prompt.tsx +1306 -0
  135. package/src/cli/cmd/run/footer.question.tsx +573 -0
  136. package/src/cli/cmd/run/footer.subagent.tsx +173 -0
  137. package/src/cli/cmd/run/footer.ts +1129 -0
  138. package/src/cli/cmd/run/footer.view.tsx +943 -0
  139. package/src/cli/cmd/run/footer.width.ts +27 -0
  140. package/src/cli/cmd/run/permission.shared.ts +256 -0
  141. package/src/cli/cmd/run/prompt.editor.ts +157 -0
  142. package/src/cli/cmd/run/prompt.shared.ts +153 -0
  143. package/src/cli/cmd/run/question.shared.ts +340 -0
  144. package/src/cli/cmd/run/runtime.boot.ts +202 -0
  145. package/src/cli/cmd/run/runtime.lifecycle.ts +406 -0
  146. package/src/cli/cmd/run/runtime.queue.ts +349 -0
  147. package/src/cli/cmd/run/runtime.shared.ts +17 -0
  148. package/src/cli/cmd/run/runtime.stdin.ts +37 -0
  149. package/src/cli/cmd/run/runtime.ts +814 -0
  150. package/src/cli/cmd/run/scrollback.shared.ts +92 -0
  151. package/src/cli/cmd/run/scrollback.surface.ts +431 -0
  152. package/src/cli/cmd/run/scrollback.writer.tsx +352 -0
  153. package/src/cli/cmd/run/session-data.ts +1113 -0
  154. package/src/cli/cmd/run/session-replay.ts +374 -0
  155. package/src/cli/cmd/run/session.shared.ts +196 -0
  156. package/src/cli/cmd/run/splash.ts +280 -0
  157. package/src/cli/cmd/run/stream.transport.ts +1462 -0
  158. package/src/cli/cmd/run/stream.ts +175 -0
  159. package/src/cli/cmd/run/subagent-data.ts +876 -0
  160. package/src/cli/cmd/run/theme.ts +690 -0
  161. package/src/cli/cmd/run/tool.ts +1489 -0
  162. package/src/cli/cmd/run/trace.ts +94 -0
  163. package/src/cli/cmd/run/turn-summary.ts +47 -0
  164. package/src/cli/cmd/run/types.ts +350 -0
  165. package/src/cli/cmd/run/variant.shared.ts +215 -0
  166. package/src/cli/cmd/run.ts +894 -0
  167. package/src/cli/cmd/serve.ts +24 -0
  168. package/src/cli/cmd/session.ts +147 -0
  169. package/src/cli/cmd/stats.ts +393 -0
  170. package/src/cli/cmd/tui.ts +224 -0
  171. package/src/cli/cmd/uninstall.ts +353 -0
  172. package/src/cli/cmd/upgrade.ts +74 -0
  173. package/src/cli/cmd/web.ts +84 -0
  174. package/src/cli/effect/prompt.ts +37 -0
  175. package/src/cli/effect-cmd.ts +96 -0
  176. package/src/cli/error.ts +130 -0
  177. package/src/cli/heap.ts +45 -0
  178. package/src/cli/logo.ts +1 -0
  179. package/src/cli/network.ts +64 -0
  180. package/src/cli/tui/layer.ts +7 -0
  181. package/src/cli/tui/validate-session.ts +29 -0
  182. package/src/cli/tui/worker.ts +71 -0
  183. package/src/cli/ui.ts +98 -0
  184. package/src/cli/upgrade.ts +53 -0
  185. package/src/command/index.ts +184 -0
  186. package/src/command/template/initialize.txt +66 -0
  187. package/src/command/template/review.txt +101 -0
  188. package/src/config/agent.ts +59 -0
  189. package/src/config/command.ts +39 -0
  190. package/src/config/config.ts +686 -0
  191. package/src/config/entry-name.ts +19 -0
  192. package/src/config/managed.ts +69 -0
  193. package/src/config/markdown.ts +36 -0
  194. package/src/config/parse.ts +79 -0
  195. package/src/config/paths.ts +45 -0
  196. package/src/config/plugin.ts +79 -0
  197. package/src/config/tui-cwd.ts +5 -0
  198. package/src/config/tui-host-attention.ts +21 -0
  199. package/src/config/tui-migrate.ts +132 -0
  200. package/src/config/tui.ts +274 -0
  201. package/src/config/variable.ts +91 -0
  202. package/src/control-plane/adapters/index.ts +41 -0
  203. package/src/control-plane/adapters/worktree.ts +96 -0
  204. package/src/control-plane/dev/README.md +19 -0
  205. package/src/control-plane/dev/debug-workspace-plugin.ts +73 -0
  206. package/src/control-plane/types.ts +59 -0
  207. package/src/control-plane/util.ts +39 -0
  208. package/src/control-plane/workspace-adapter-runtime.ts +51 -0
  209. package/src/control-plane/workspace-context.ts +26 -0
  210. package/src/control-plane/workspace.ts +989 -0
  211. package/src/effect/app-runtime.ts +132 -0
  212. package/src/effect/bootstrap-runtime.ts +23 -0
  213. package/src/effect/bridge.ts +84 -0
  214. package/src/effect/config-service.ts +67 -0
  215. package/src/effect/instance-ref.ts +11 -0
  216. package/src/effect/instance-registry.ts +12 -0
  217. package/src/effect/instance-state.ts +69 -0
  218. package/src/effect/promise.ts +17 -0
  219. package/src/effect/run-service.ts +47 -0
  220. package/src/effect/runner.ts +217 -0
  221. package/src/effect/runtime-flags.ts +79 -0
  222. package/src/env/index.ts +43 -0
  223. package/src/event-v2-bridge.ts +79 -0
  224. package/src/format/formatter.ts +404 -0
  225. package/src/format/index.ts +205 -0
  226. package/src/git/index.ts +350 -0
  227. package/src/id/id.ts +80 -0
  228. package/src/ide/index.ts +61 -0
  229. package/src/image/image.ts +174 -0
  230. package/src/index.ts +142 -0
  231. package/src/installation/index.ts +350 -0
  232. package/src/lsp/client.ts +650 -0
  233. package/src/lsp/diagnostic.ts +29 -0
  234. package/src/lsp/language.ts +121 -0
  235. package/src/lsp/launch.ts +21 -0
  236. package/src/lsp/lsp.ts +511 -0
  237. package/src/lsp/server.ts +1983 -0
  238. package/src/markdown.d.ts +4 -0
  239. package/src/mcp/auth.ts +174 -0
  240. package/src/mcp/catalog.ts +153 -0
  241. package/src/mcp/index.ts +946 -0
  242. package/src/mcp/oauth-callback.ts +233 -0
  243. package/src/mcp/oauth-provider.ts +206 -0
  244. package/src/node.ts +4 -0
  245. package/src/patch/index.ts +686 -0
  246. package/src/permission/arity.ts +163 -0
  247. package/src/permission/evaluate.ts +1 -0
  248. package/src/permission/index.ts +230 -0
  249. package/src/plugin/azure.ts +26 -0
  250. package/src/plugin/cloudflare.ts +76 -0
  251. package/src/plugin/digitalocean.ts +383 -0
  252. package/src/plugin/github-copilot/copilot.ts +413 -0
  253. package/src/plugin/github-copilot/models.ts +246 -0
  254. package/src/plugin/index.ts +316 -0
  255. package/src/plugin/install.ts +439 -0
  256. package/src/plugin/loader.ts +237 -0
  257. package/src/plugin/meta.ts +188 -0
  258. package/src/plugin/openai/README.md +31 -0
  259. package/src/plugin/openai/codex.ts +641 -0
  260. package/src/plugin/openai/ws-pool.ts +270 -0
  261. package/src/plugin/openai/ws.ts +381 -0
  262. package/src/plugin/pty-environment.ts +24 -0
  263. package/src/plugin/shared.ts +323 -0
  264. package/src/plugin/snowflake-cortex.ts +529 -0
  265. package/src/plugin/tui/internal.ts +10 -0
  266. package/src/plugin/tui/runtime.ts +1130 -0
  267. package/src/plugin/xai.ts +716 -0
  268. package/src/project/bootstrap-service.ts +9 -0
  269. package/src/project/bootstrap.ts +76 -0
  270. package/src/project/instance-context.ts +24 -0
  271. package/src/project/instance-layer.ts +11 -0
  272. package/src/project/instance-runtime.ts +16 -0
  273. package/src/project/instance-store.ts +209 -0
  274. package/src/project/project.ts +519 -0
  275. package/src/project/vcs.ts +431 -0
  276. package/src/provider/auth.ts +233 -0
  277. package/src/provider/error.ts +188 -0
  278. package/src/provider/model-status.ts +8 -0
  279. package/src/provider/provider.ts +1979 -0
  280. package/src/provider/transform.ts +1426 -0
  281. package/src/question/index.ts +229 -0
  282. package/src/question/schema.ts +10 -0
  283. package/src/server/auth.ts +48 -0
  284. package/src/server/event.ts +13 -0
  285. package/src/server/global-lifecycle.ts +28 -0
  286. package/src/server/init-projectors.ts +3 -0
  287. package/src/server/mdns.ts +47 -0
  288. package/src/server/projectors.ts +1 -0
  289. package/src/server/proxy-util.ts +48 -0
  290. package/src/server/routes/instance/httpapi/AGENTS.md +39 -0
  291. package/src/server/routes/instance/httpapi/api.ts +78 -0
  292. package/src/server/routes/instance/httpapi/errors.ts +193 -0
  293. package/src/server/routes/instance/httpapi/groups/config.ts +65 -0
  294. package/src/server/routes/instance/httpapi/groups/control-plane.ts +35 -0
  295. package/src/server/routes/instance/httpapi/groups/control.ts +76 -0
  296. package/src/server/routes/instance/httpapi/groups/event.ts +29 -0
  297. package/src/server/routes/instance/httpapi/groups/experimental.ts +260 -0
  298. package/src/server/routes/instance/httpapi/groups/file.ts +185 -0
  299. package/src/server/routes/instance/httpapi/groups/global.ts +138 -0
  300. package/src/server/routes/instance/httpapi/groups/instance.ts +206 -0
  301. package/src/server/routes/instance/httpapi/groups/mcp.ts +156 -0
  302. package/src/server/routes/instance/httpapi/groups/metadata.ts +18 -0
  303. package/src/server/routes/instance/httpapi/groups/permission.ts +61 -0
  304. package/src/server/routes/instance/httpapi/groups/project-copy.ts +32 -0
  305. package/src/server/routes/instance/httpapi/groups/project.ts +93 -0
  306. package/src/server/routes/instance/httpapi/groups/provider.ts +101 -0
  307. package/src/server/routes/instance/httpapi/groups/pty.ts +172 -0
  308. package/src/server/routes/instance/httpapi/groups/query.ts +12 -0
  309. package/src/server/routes/instance/httpapi/groups/question.ts +74 -0
  310. package/src/server/routes/instance/httpapi/groups/session.ts +462 -0
  311. package/src/server/routes/instance/httpapi/groups/sync.ts +113 -0
  312. package/src/server/routes/instance/httpapi/groups/tui.ts +208 -0
  313. package/src/server/routes/instance/httpapi/groups/workspace.ts +141 -0
  314. package/src/server/routes/instance/httpapi/handlers/config.ts +34 -0
  315. package/src/server/routes/instance/httpapi/handlers/control-plane.ts +37 -0
  316. package/src/server/routes/instance/httpapi/handlers/control.ts +43 -0
  317. package/src/server/routes/instance/httpapi/handlers/event.ts +99 -0
  318. package/src/server/routes/instance/httpapi/handlers/experimental.ts +187 -0
  319. package/src/server/routes/instance/httpapi/handlers/file.ts +139 -0
  320. package/src/server/routes/instance/httpapi/handlers/global.ts +156 -0
  321. package/src/server/routes/instance/httpapi/handlers/instance.ts +110 -0
  322. package/src/server/routes/instance/httpapi/handlers/mcp.ts +111 -0
  323. package/src/server/routes/instance/httpapi/handlers/permission.ts +41 -0
  324. package/src/server/routes/instance/httpapi/handlers/project-copy.ts +83 -0
  325. package/src/server/routes/instance/httpapi/handlers/project.ts +63 -0
  326. package/src/server/routes/instance/httpapi/handlers/provider.ts +113 -0
  327. package/src/server/routes/instance/httpapi/handlers/pty.ts +273 -0
  328. package/src/server/routes/instance/httpapi/handlers/question.ts +54 -0
  329. package/src/server/routes/instance/httpapi/handlers/session-errors.ts +21 -0
  330. package/src/server/routes/instance/httpapi/handlers/session.ts +440 -0
  331. package/src/server/routes/instance/httpapi/handlers/sync.ts +89 -0
  332. package/src/server/routes/instance/httpapi/handlers/tui.ts +131 -0
  333. package/src/server/routes/instance/httpapi/handlers/workspace.ts +102 -0
  334. package/src/server/routes/instance/httpapi/lifecycle.ts +54 -0
  335. package/src/server/routes/instance/httpapi/middleware/authorization.ts +150 -0
  336. package/src/server/routes/instance/httpapi/middleware/compression.ts +64 -0
  337. package/src/server/routes/instance/httpapi/middleware/cors-vary.ts +29 -0
  338. package/src/server/routes/instance/httpapi/middleware/error.ts +43 -0
  339. package/src/server/routes/instance/httpapi/middleware/fence.ts +25 -0
  340. package/src/server/routes/instance/httpapi/middleware/instance-context.ts +43 -0
  341. package/src/server/routes/instance/httpapi/middleware/proxy.ts +108 -0
  342. package/src/server/routes/instance/httpapi/middleware/schema-error.ts +41 -0
  343. package/src/server/routes/instance/httpapi/middleware/workspace-routing.ts +250 -0
  344. package/src/server/routes/instance/httpapi/public.ts +535 -0
  345. package/src/server/routes/instance/httpapi/server.ts +298 -0
  346. package/src/server/routes/instance/httpapi/websocket-tracker.ts +57 -0
  347. package/src/server/server.ts +225 -0
  348. package/src/server/shared/fence.ts +60 -0
  349. package/src/server/shared/pty-ticket.ts +15 -0
  350. package/src/server/shared/public-ui.ts +12 -0
  351. package/src/server/shared/tui-control.ts +28 -0
  352. package/src/server/shared/ui.ts +122 -0
  353. package/src/server/shared/workspace-routing.ts +38 -0
  354. package/src/server/tui-event.ts +53 -0
  355. package/src/session/compaction.ts +620 -0
  356. package/src/session/instruction.ts +241 -0
  357. package/src/session/llm/AGENTS.md +90 -0
  358. package/src/session/llm/ai-sdk.ts +288 -0
  359. package/src/session/llm/native-request.ts +196 -0
  360. package/src/session/llm/native-runtime.ts +196 -0
  361. package/src/session/llm/request.ts +218 -0
  362. package/src/session/llm.ts +415 -0
  363. package/src/session/message-error.ts +14 -0
  364. package/src/session/message-v2.ts +744 -0
  365. package/src/session/message.ts +148 -0
  366. package/src/session/overflow.ts +34 -0
  367. package/src/session/processor.ts +1084 -0
  368. package/src/session/prompt/anthropic.txt +105 -0
  369. package/src/session/prompt/beast.txt +147 -0
  370. package/src/session/prompt/build-switch.txt +5 -0
  371. package/src/session/prompt/codex.txt +79 -0
  372. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  373. package/src/session/prompt/default.txt +95 -0
  374. package/src/session/prompt/gemini.txt +155 -0
  375. package/src/session/prompt/gpt.txt +107 -0
  376. package/src/session/prompt/kimi.txt +95 -0
  377. package/src/session/prompt/max-steps.txt +16 -0
  378. package/src/session/prompt/plan-mode.txt +70 -0
  379. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  380. package/src/session/prompt/plan.txt +26 -0
  381. package/src/session/prompt/trinity.txt +97 -0
  382. package/src/session/prompt.ts +1722 -0
  383. package/src/session/reminders.ts +92 -0
  384. package/src/session/retry.ts +201 -0
  385. package/src/session/revert.ts +160 -0
  386. package/src/session/run-state.ts +156 -0
  387. package/src/session/schema.ts +26 -0
  388. package/src/session/session.ts +1119 -0
  389. package/src/session/status.ts +97 -0
  390. package/src/session/summary.ts +165 -0
  391. package/src/session/system.ts +117 -0
  392. package/src/session/todo.ts +90 -0
  393. package/src/session/tools.ts +207 -0
  394. package/src/share/session.ts +61 -0
  395. package/src/share/share-next.ts +385 -0
  396. package/src/skill/discovery.ts +109 -0
  397. package/src/skill/index.ts +366 -0
  398. package/src/snapshot/index.ts +808 -0
  399. package/src/sql.d.ts +4 -0
  400. package/src/storage/schema.ts +5 -0
  401. package/src/storage/storage.ts +329 -0
  402. package/src/sync/README.md +179 -0
  403. package/src/sync/schema.ts +11 -0
  404. package/src/temporary.ts +31 -0
  405. package/src/tool/apply_patch.ts +313 -0
  406. package/src/tool/apply_patch.txt +33 -0
  407. package/src/tool/edit.ts +737 -0
  408. package/src/tool/edit.txt +10 -0
  409. package/src/tool/external-directory.ts +49 -0
  410. package/src/tool/glob.ts +76 -0
  411. package/src/tool/glob.txt +6 -0
  412. package/src/tool/grep.ts +112 -0
  413. package/src/tool/grep.txt +8 -0
  414. package/src/tool/invalid.ts +21 -0
  415. package/src/tool/json-schema.ts +164 -0
  416. package/src/tool/lsp.ts +113 -0
  417. package/src/tool/lsp.txt +24 -0
  418. package/src/tool/mcp-websearch.ts +96 -0
  419. package/src/tool/plan-enter.txt +14 -0
  420. package/src/tool/plan-exit.txt +13 -0
  421. package/src/tool/plan.ts +79 -0
  422. package/src/tool/question.ts +44 -0
  423. package/src/tool/question.txt +10 -0
  424. package/src/tool/read.ts +386 -0
  425. package/src/tool/read.txt +14 -0
  426. package/src/tool/registry.ts +440 -0
  427. package/src/tool/schema.ts +14 -0
  428. package/src/tool/shell/id.ts +19 -0
  429. package/src/tool/shell/prompt.ts +307 -0
  430. package/src/tool/shell/shell.txt +21 -0
  431. package/src/tool/shell.ts +657 -0
  432. package/src/tool/skill.ts +71 -0
  433. package/src/tool/skill.txt +5 -0
  434. package/src/tool/task.ts +346 -0
  435. package/src/tool/task.txt +19 -0
  436. package/src/tool/todo.ts +57 -0
  437. package/src/tool/todowrite.txt +44 -0
  438. package/src/tool/tool.ts +183 -0
  439. package/src/tool/truncate.ts +158 -0
  440. package/src/tool/truncation-dir.ts +4 -0
  441. package/src/tool/webfetch.ts +192 -0
  442. package/src/tool/webfetch.txt +13 -0
  443. package/src/tool/websearch.ts +143 -0
  444. package/src/tool/websearch.txt +14 -0
  445. package/src/tool/write.ts +104 -0
  446. package/src/tool/write.txt +8 -0
  447. package/src/util/archive.ts +17 -0
  448. package/src/util/bom.ts +27 -0
  449. package/src/util/data-url.ts +9 -0
  450. package/src/util/defer.ts +10 -0
  451. package/src/util/effect-http-client.ts +11 -0
  452. package/src/util/error.ts +1 -0
  453. package/src/util/filesystem.ts +251 -0
  454. package/src/util/html.ts +8 -0
  455. package/src/util/iife.ts +3 -0
  456. package/src/util/lazy.ts +20 -0
  457. package/src/util/local-context.ts +25 -0
  458. package/src/util/locale.ts +2 -0
  459. package/src/util/media.ts +26 -0
  460. package/src/util/process.ts +177 -0
  461. package/src/util/proxy-env.ts +72 -0
  462. package/src/util/queue.ts +32 -0
  463. package/src/util/record.ts +1 -0
  464. package/src/util/repository.ts +232 -0
  465. package/src/util/rpc.ts +66 -0
  466. package/src/util/signal.ts +12 -0
  467. package/src/util/timeout.ts +13 -0
  468. package/src/util/token.ts +1 -0
  469. package/src/util/wildcard.ts +59 -0
  470. package/src/worktree/index.ts +654 -0
@@ -0,0 +1,686 @@
1
+ import { Effect, Schema } from "effect"
2
+ import * as path from "path"
3
+ import { FSUtil } from "@rimurucode-ai/core/fs-util"
4
+ import * as Bom from "../util/bom"
5
+
6
+ export const PatchSchema = Schema.Struct({
7
+ patchText: Schema.String.annotate({ description: "The full patch text that describes all changes to be made" }),
8
+ })
9
+
10
+ export type PatchParams = Schema.Schema.Type<typeof PatchSchema>
11
+
12
+ // Core types matching the Rust implementation
13
+ export interface ApplyPatchArgs {
14
+ patch: string
15
+ hunks: Hunk[]
16
+ workdir?: string
17
+ }
18
+
19
+ export type Hunk =
20
+ | { type: "add"; path: string; contents: string }
21
+ | { type: "delete"; path: string }
22
+ | { type: "update"; path: string; move_path?: string; chunks: UpdateFileChunk[] }
23
+
24
+ export interface UpdateFileChunk {
25
+ old_lines: string[]
26
+ new_lines: string[]
27
+ change_context?: string
28
+ is_end_of_file?: boolean
29
+ }
30
+
31
+ export interface ApplyPatchAction {
32
+ changes: Map<string, ApplyPatchFileChange>
33
+ patch: string
34
+ cwd: string
35
+ }
36
+
37
+ export type ApplyPatchFileChange =
38
+ | { type: "add"; content: string }
39
+ | { type: "delete"; content: string }
40
+ | { type: "update"; unified_diff: string; move_path?: string; new_content: string }
41
+
42
+ export interface AffectedPaths {
43
+ added: string[]
44
+ modified: string[]
45
+ deleted: string[]
46
+ }
47
+
48
+ export enum ApplyPatchError {
49
+ ParseError = "ParseError",
50
+ IoError = "IoError",
51
+ ComputeReplacements = "ComputeReplacements",
52
+ ImplicitInvocation = "ImplicitInvocation",
53
+ }
54
+
55
+ export enum MaybeApplyPatch {
56
+ Body = "Body",
57
+ ShellParseError = "ShellParseError",
58
+ PatchParseError = "PatchParseError",
59
+ NotApplyPatch = "NotApplyPatch",
60
+ }
61
+
62
+ export enum MaybeApplyPatchVerified {
63
+ Body = "Body",
64
+ ShellParseError = "ShellParseError",
65
+ CorrectnessError = "CorrectnessError",
66
+ NotApplyPatch = "NotApplyPatch",
67
+ }
68
+
69
+ // Parser implementation
70
+ function parsePatchHeader(
71
+ lines: string[],
72
+ startIdx: number,
73
+ ): { filePath: string; movePath?: string; nextIdx: number } | null {
74
+ const line = lines[startIdx]
75
+
76
+ if (line.startsWith("*** Add File:")) {
77
+ const filePath = line.slice("*** Add File:".length).trim()
78
+ return filePath ? { filePath, nextIdx: startIdx + 1 } : null
79
+ }
80
+
81
+ if (line.startsWith("*** Delete File:")) {
82
+ const filePath = line.slice("*** Delete File:".length).trim()
83
+ return filePath ? { filePath, nextIdx: startIdx + 1 } : null
84
+ }
85
+
86
+ if (line.startsWith("*** Update File:")) {
87
+ const filePath = line.slice("*** Update File:".length).trim()
88
+ let movePath: string | undefined
89
+ let nextIdx = startIdx + 1
90
+
91
+ // Check for move directive
92
+ if (nextIdx < lines.length && lines[nextIdx].startsWith("*** Move to:")) {
93
+ movePath = lines[nextIdx].slice("*** Move to:".length).trim()
94
+ nextIdx++
95
+ }
96
+
97
+ return filePath ? { filePath, movePath, nextIdx } : null
98
+ }
99
+
100
+ return null
101
+ }
102
+
103
+ function parseUpdateFileChunks(lines: string[], startIdx: number): { chunks: UpdateFileChunk[]; nextIdx: number } {
104
+ const chunks: UpdateFileChunk[] = []
105
+ let i = startIdx
106
+
107
+ while (i < lines.length && !lines[i].startsWith("***")) {
108
+ if (lines[i].startsWith("@@")) {
109
+ // Parse context line
110
+ const contextLine = lines[i].substring(2).trim()
111
+ i++
112
+
113
+ const oldLines: string[] = []
114
+ const newLines: string[] = []
115
+ let isEndOfFile = false
116
+
117
+ // Parse change lines
118
+ while (i < lines.length && !lines[i].startsWith("@@") && !lines[i].startsWith("***")) {
119
+ const changeLine = lines[i]
120
+
121
+ if (changeLine === "*** End of File") {
122
+ isEndOfFile = true
123
+ i++
124
+ break
125
+ }
126
+
127
+ if (changeLine.startsWith(" ")) {
128
+ // Keep line - appears in both old and new
129
+ const content = changeLine.substring(1)
130
+ oldLines.push(content)
131
+ newLines.push(content)
132
+ } else if (changeLine.startsWith("-")) {
133
+ // Remove line - only in old
134
+ oldLines.push(changeLine.substring(1))
135
+ } else if (changeLine.startsWith("+")) {
136
+ // Add line - only in new
137
+ newLines.push(changeLine.substring(1))
138
+ }
139
+
140
+ i++
141
+ }
142
+
143
+ chunks.push({
144
+ old_lines: oldLines,
145
+ new_lines: newLines,
146
+ change_context: contextLine || undefined,
147
+ is_end_of_file: isEndOfFile || undefined,
148
+ })
149
+ } else {
150
+ i++
151
+ }
152
+ }
153
+
154
+ return { chunks, nextIdx: i }
155
+ }
156
+
157
+ function parseAddFileContent(lines: string[], startIdx: number): { content: string; nextIdx: number } {
158
+ let content = ""
159
+ let i = startIdx
160
+
161
+ while (i < lines.length && !lines[i].startsWith("***")) {
162
+ if (lines[i].startsWith("+")) {
163
+ content += lines[i].substring(1) + "\n"
164
+ }
165
+ i++
166
+ }
167
+
168
+ // Remove trailing newline
169
+ if (content.endsWith("\n")) {
170
+ content = content.slice(0, -1)
171
+ }
172
+
173
+ return { content, nextIdx: i }
174
+ }
175
+
176
+ function stripHeredoc(input: string): string {
177
+ // Match heredoc patterns like: cat <<'EOF'\n...\nEOF or <<EOF\n...\nEOF
178
+ const heredocMatch = input.match(/^(?:cat\s+)?<<['"]?(\w+)['"]?\s*\n([\s\S]*?)\n\1\s*$/)
179
+ if (heredocMatch) {
180
+ return heredocMatch[2]
181
+ }
182
+ return input
183
+ }
184
+
185
+ export function parsePatch(patchText: string): { hunks: Hunk[] } {
186
+ const cleaned = stripHeredoc(patchText.trim())
187
+ const lines = cleaned.split("\n")
188
+ const hunks: Hunk[] = []
189
+ let i = 0
190
+
191
+ // Look for Begin/End patch markers
192
+ const beginMarker = "*** Begin Patch"
193
+ const endMarker = "*** End Patch"
194
+
195
+ const beginIdx = lines.findIndex((line) => line.trim() === beginMarker)
196
+ const endIdx = lines.findIndex((line) => line.trim() === endMarker)
197
+
198
+ if (beginIdx === -1 || endIdx === -1 || beginIdx >= endIdx) {
199
+ throw new Error("Invalid patch format: missing Begin/End markers")
200
+ }
201
+
202
+ // Parse content between markers
203
+ i = beginIdx + 1
204
+
205
+ while (i < endIdx) {
206
+ const header = parsePatchHeader(lines, i)
207
+ if (!header) {
208
+ i++
209
+ continue
210
+ }
211
+
212
+ if (lines[i].startsWith("*** Add File:")) {
213
+ const { content, nextIdx } = parseAddFileContent(lines, header.nextIdx)
214
+ hunks.push({
215
+ type: "add",
216
+ path: header.filePath,
217
+ contents: content,
218
+ })
219
+ i = nextIdx
220
+ } else if (lines[i].startsWith("*** Delete File:")) {
221
+ hunks.push({
222
+ type: "delete",
223
+ path: header.filePath,
224
+ })
225
+ i = header.nextIdx
226
+ } else if (lines[i].startsWith("*** Update File:")) {
227
+ const { chunks, nextIdx } = parseUpdateFileChunks(lines, header.nextIdx)
228
+ hunks.push({
229
+ type: "update",
230
+ path: header.filePath,
231
+ move_path: header.movePath,
232
+ chunks,
233
+ })
234
+ i = nextIdx
235
+ } else {
236
+ i++
237
+ }
238
+ }
239
+
240
+ return { hunks }
241
+ }
242
+
243
+ // Apply patch functionality
244
+ export function maybeParseApplyPatch(
245
+ argv: string[],
246
+ ):
247
+ | { type: MaybeApplyPatch.Body; args: ApplyPatchArgs }
248
+ | { type: MaybeApplyPatch.PatchParseError; error: Error }
249
+ | { type: MaybeApplyPatch.NotApplyPatch } {
250
+ const APPLY_PATCH_COMMANDS = ["apply_patch", "applypatch"]
251
+
252
+ // Direct invocation: apply_patch <patch>
253
+ if (argv.length === 2 && APPLY_PATCH_COMMANDS.includes(argv[0])) {
254
+ try {
255
+ const { hunks } = parsePatch(argv[1])
256
+ return {
257
+ type: MaybeApplyPatch.Body,
258
+ args: {
259
+ patch: argv[1],
260
+ hunks,
261
+ },
262
+ }
263
+ } catch (error) {
264
+ return {
265
+ type: MaybeApplyPatch.PatchParseError,
266
+ error: error as Error,
267
+ }
268
+ }
269
+ }
270
+
271
+ // Bash heredoc form: bash -lc 'apply_patch <<"EOF" ...'
272
+ if (argv.length === 3 && argv[0] === "bash" && argv[1] === "-lc") {
273
+ // Simple extraction - in real implementation would need proper bash parsing
274
+ const script = argv[2]
275
+ const heredocMatch = script.match(/apply_patch\s*<<['"](\w+)['"]\s*\n([\s\S]*?)\n\1/)
276
+
277
+ if (heredocMatch) {
278
+ const patchContent = heredocMatch[2]
279
+ try {
280
+ const { hunks } = parsePatch(patchContent)
281
+ return {
282
+ type: MaybeApplyPatch.Body,
283
+ args: {
284
+ patch: patchContent,
285
+ hunks,
286
+ },
287
+ }
288
+ } catch (error) {
289
+ return {
290
+ type: MaybeApplyPatch.PatchParseError,
291
+ error: error as Error,
292
+ }
293
+ }
294
+ }
295
+ }
296
+
297
+ return { type: MaybeApplyPatch.NotApplyPatch }
298
+ }
299
+
300
+ // File content manipulation
301
+ interface ApplyPatchFileUpdate {
302
+ unified_diff: string
303
+ content: string
304
+ bom: boolean
305
+ }
306
+
307
+ export function deriveNewContentsFromChunks(
308
+ filePath: string,
309
+ chunks: UpdateFileChunk[],
310
+ originalText: string,
311
+ ): ApplyPatchFileUpdate {
312
+ const originalContent = Bom.split(originalText)
313
+
314
+ let originalLines = originalContent.text.split("\n")
315
+
316
+ // Drop trailing empty element for consistent line counting
317
+ if (originalLines.length > 0 && originalLines[originalLines.length - 1] === "") {
318
+ originalLines.pop()
319
+ }
320
+
321
+ const replacements = computeReplacements(originalLines, filePath, chunks)
322
+ let newLines = applyReplacements(originalLines, replacements)
323
+
324
+ // Ensure trailing newline
325
+ if (newLines.length === 0 || newLines[newLines.length - 1] !== "") {
326
+ newLines.push("")
327
+ }
328
+
329
+ const next = Bom.split(newLines.join("\n"))
330
+ const newContent = next.text
331
+
332
+ // Generate unified diff
333
+ const unifiedDiff = generateUnifiedDiff(originalContent.text, newContent)
334
+
335
+ return {
336
+ unified_diff: unifiedDiff,
337
+ content: newContent,
338
+ bom: originalContent.bom || next.bom,
339
+ }
340
+ }
341
+
342
+ function computeReplacements(
343
+ originalLines: string[],
344
+ filePath: string,
345
+ chunks: UpdateFileChunk[],
346
+ ): Array<[number, number, string[]]> {
347
+ const replacements: Array<[number, number, string[]]> = []
348
+ let lineIndex = 0
349
+
350
+ for (const chunk of chunks) {
351
+ // Handle context-based seeking
352
+ if (chunk.change_context) {
353
+ const contextIdx = seekSequence(originalLines, [chunk.change_context], lineIndex)
354
+ if (contextIdx === -1) {
355
+ throw new Error(`Failed to find context '${chunk.change_context}' in ${filePath}`)
356
+ }
357
+ lineIndex = contextIdx + 1
358
+ }
359
+
360
+ // Handle pure addition (no old lines)
361
+ if (chunk.old_lines.length === 0) {
362
+ const insertionIdx =
363
+ originalLines.length > 0 && originalLines[originalLines.length - 1] === ""
364
+ ? originalLines.length - 1
365
+ : originalLines.length
366
+ replacements.push([insertionIdx, 0, chunk.new_lines])
367
+ continue
368
+ }
369
+
370
+ // Try to match old lines in the file
371
+ let pattern = chunk.old_lines
372
+ let newSlice = chunk.new_lines
373
+ let found = seekSequence(originalLines, pattern, lineIndex, chunk.is_end_of_file)
374
+
375
+ // Retry without trailing empty line if not found
376
+ if (found === -1 && pattern.length > 0 && pattern[pattern.length - 1] === "") {
377
+ pattern = pattern.slice(0, -1)
378
+ if (newSlice.length > 0 && newSlice[newSlice.length - 1] === "") {
379
+ newSlice = newSlice.slice(0, -1)
380
+ }
381
+ found = seekSequence(originalLines, pattern, lineIndex, chunk.is_end_of_file)
382
+ }
383
+
384
+ if (found !== -1) {
385
+ replacements.push([found, pattern.length, newSlice])
386
+ lineIndex = found + pattern.length
387
+ } else {
388
+ throw new Error(`Failed to find expected lines in ${filePath}:\n${chunk.old_lines.join("\n")}`)
389
+ }
390
+ }
391
+
392
+ // Sort replacements by index to apply in order
393
+ replacements.sort((a, b) => a[0] - b[0])
394
+
395
+ return replacements
396
+ }
397
+
398
+ function applyReplacements(lines: string[], replacements: Array<[number, number, string[]]>): string[] {
399
+ // Apply replacements in reverse order to avoid index shifting
400
+ const result = [...lines]
401
+
402
+ for (let i = replacements.length - 1; i >= 0; i--) {
403
+ const [startIdx, oldLen, newSegment] = replacements[i]
404
+
405
+ // Remove old lines
406
+ result.splice(startIdx, oldLen)
407
+
408
+ // Insert new lines
409
+ for (let j = 0; j < newSegment.length; j++) {
410
+ result.splice(startIdx + j, 0, newSegment[j])
411
+ }
412
+ }
413
+
414
+ return result
415
+ }
416
+
417
+ // Normalize Unicode punctuation to ASCII equivalents (like Rust's normalize_unicode)
418
+ function normalizeUnicode(str: string): string {
419
+ return str
420
+ .replace(/[‘’‚‛]/g, "'") // single quotes
421
+ .replace(/[“”„‟]/g, '"') // double quotes
422
+ .replace(/[‐‑‒–—―]/g, "-") // dashes
423
+ .replace(/…/g, "...") // ellipsis
424
+ .replace(/ /g, " ") // non-breaking space
425
+ }
426
+
427
+ type Comparator = (a: string, b: string) => boolean
428
+
429
+ function tryMatch(lines: string[], pattern: string[], startIndex: number, compare: Comparator, eof: boolean): number {
430
+ // If EOF anchor, try matching from end of file first
431
+ if (eof) {
432
+ const fromEnd = lines.length - pattern.length
433
+ if (fromEnd >= startIndex) {
434
+ let matches = true
435
+ for (let j = 0; j < pattern.length; j++) {
436
+ if (!compare(lines[fromEnd + j], pattern[j])) {
437
+ matches = false
438
+ break
439
+ }
440
+ }
441
+ if (matches) return fromEnd
442
+ }
443
+ }
444
+
445
+ // Forward search from startIndex
446
+ for (let i = startIndex; i <= lines.length - pattern.length; i++) {
447
+ let matches = true
448
+ for (let j = 0; j < pattern.length; j++) {
449
+ if (!compare(lines[i + j], pattern[j])) {
450
+ matches = false
451
+ break
452
+ }
453
+ }
454
+ if (matches) return i
455
+ }
456
+
457
+ return -1
458
+ }
459
+
460
+ function seekSequence(lines: string[], pattern: string[], startIndex: number, eof = false): number {
461
+ if (pattern.length === 0) return -1
462
+
463
+ // Pass 1: exact match
464
+ const exact = tryMatch(lines, pattern, startIndex, (a, b) => a === b, eof)
465
+ if (exact !== -1) return exact
466
+
467
+ // Pass 2: rstrip (trim trailing whitespace)
468
+ const rstrip = tryMatch(lines, pattern, startIndex, (a, b) => a.trimEnd() === b.trimEnd(), eof)
469
+ if (rstrip !== -1) return rstrip
470
+
471
+ // Pass 3: trim (both ends)
472
+ const trim = tryMatch(lines, pattern, startIndex, (a, b) => a.trim() === b.trim(), eof)
473
+ if (trim !== -1) return trim
474
+
475
+ // Pass 4: normalized (Unicode punctuation to ASCII)
476
+ const normalized = tryMatch(
477
+ lines,
478
+ pattern,
479
+ startIndex,
480
+ (a, b) => normalizeUnicode(a.trim()) === normalizeUnicode(b.trim()),
481
+ eof,
482
+ )
483
+ return normalized
484
+ }
485
+
486
+ function generateUnifiedDiff(oldContent: string, newContent: string): string {
487
+ const oldLines = oldContent.split("\n")
488
+ const newLines = newContent.split("\n")
489
+
490
+ // Simple diff generation - in a real implementation you'd use a proper diff algorithm
491
+ let diff = "@@ -1 +1 @@\n"
492
+
493
+ // Find changes (simplified approach)
494
+ const maxLen = Math.max(oldLines.length, newLines.length)
495
+ let hasChanges = false
496
+
497
+ for (let i = 0; i < maxLen; i++) {
498
+ const oldLine = oldLines[i] || ""
499
+ const newLine = newLines[i] || ""
500
+
501
+ if (oldLine !== newLine) {
502
+ if (oldLine) diff += `-${oldLine}\n`
503
+ if (newLine) diff += `+${newLine}\n`
504
+ hasChanges = true
505
+ } else if (oldLine) {
506
+ diff += ` ${oldLine}\n`
507
+ }
508
+ }
509
+
510
+ return hasChanges ? diff : ""
511
+ }
512
+
513
+ // Apply hunks to filesystem
514
+ export const applyHunksToFiles = Effect.fn("Patch.applyHunksToFiles")(function* (hunks: Hunk[]) {
515
+ if (hunks.length === 0) {
516
+ return yield* Effect.fail(new Error("No files were modified."))
517
+ }
518
+
519
+ const fs = yield* FSUtil.Service
520
+
521
+ const added: string[] = []
522
+ const modified: string[] = []
523
+ const deleted: string[] = []
524
+
525
+ for (const hunk of hunks) {
526
+ switch (hunk.type) {
527
+ case "add": {
528
+ yield* fs.writeWithDirs(hunk.path, hunk.contents)
529
+ added.push(hunk.path)
530
+ yield* Effect.logInfo(`Added file: ${hunk.path}`)
531
+ break
532
+ }
533
+
534
+ case "delete": {
535
+ yield* fs.remove(hunk.path)
536
+ deleted.push(hunk.path)
537
+ yield* Effect.logInfo(`Deleted file: ${hunk.path}`)
538
+ break
539
+ }
540
+
541
+ case "update": {
542
+ const originalText = yield* fs.readFileString(hunk.path)
543
+ const fileUpdate = deriveNewContentsFromChunks(hunk.path, hunk.chunks, originalText)
544
+
545
+ if (hunk.move_path) {
546
+ yield* fs.writeWithDirs(hunk.move_path, Bom.join(fileUpdate.content, fileUpdate.bom))
547
+ yield* fs.remove(hunk.path)
548
+ modified.push(hunk.move_path)
549
+ yield* Effect.logInfo(`Moved file: ${hunk.path} -> ${hunk.move_path}`)
550
+ } else {
551
+ yield* fs.writeWithDirs(hunk.path, Bom.join(fileUpdate.content, fileUpdate.bom))
552
+ modified.push(hunk.path)
553
+ yield* Effect.logInfo(`Updated file: ${hunk.path}`)
554
+ }
555
+ break
556
+ }
557
+ }
558
+ }
559
+
560
+ return { added, modified, deleted } satisfies AffectedPaths
561
+ })
562
+
563
+ // Main patch application function
564
+ export const applyPatch = Effect.fn("Patch.applyPatch")(function* (patchText: string) {
565
+ const { hunks } = parsePatch(patchText)
566
+ return yield* applyHunksToFiles(hunks)
567
+ })
568
+
569
+ type MaybeApplyPatchVerifiedResult =
570
+ | { type: MaybeApplyPatchVerified.Body; action: ApplyPatchAction }
571
+ | { type: MaybeApplyPatchVerified.CorrectnessError; error: Error }
572
+ | { type: MaybeApplyPatchVerified.NotApplyPatch }
573
+
574
+ // Effectful verified-parse: needs FSUtil.Service to read existing files
575
+ export const maybeParseApplyPatchVerified = Effect.fn("Patch.maybeParseApplyPatchVerified")(function* (
576
+ argv: string[],
577
+ cwd: string,
578
+ ) {
579
+ // Detect implicit patch invocation (raw patch without apply_patch command)
580
+ if (argv.length === 1) {
581
+ try {
582
+ parsePatch(argv[0])
583
+ return {
584
+ type: MaybeApplyPatchVerified.CorrectnessError,
585
+ error: new Error(ApplyPatchError.ImplicitInvocation),
586
+ } satisfies MaybeApplyPatchVerifiedResult
587
+ } catch {
588
+ // Not a patch, continue
589
+ }
590
+ }
591
+
592
+ const result = maybeParseApplyPatch(argv)
593
+
594
+ switch (result.type) {
595
+ case MaybeApplyPatch.Body: {
596
+ const fs = yield* FSUtil.Service
597
+ const args = result.args
598
+ const effectiveCwd = args.workdir ? path.resolve(cwd, args.workdir) : cwd
599
+ const changes = new Map<string, ApplyPatchFileChange>()
600
+
601
+ for (const hunk of args.hunks) {
602
+ const resolvedPath = path.resolve(
603
+ effectiveCwd,
604
+ hunk.type === "update" && hunk.move_path ? hunk.move_path : hunk.path,
605
+ )
606
+
607
+ switch (hunk.type) {
608
+ case "add":
609
+ changes.set(resolvedPath, {
610
+ type: "add",
611
+ content: hunk.contents,
612
+ })
613
+ break
614
+
615
+ case "delete": {
616
+ const deletePath = path.resolve(effectiveCwd, hunk.path)
617
+ const content = yield* fs.readFileString(deletePath).pipe(Effect.catch(() => Effect.succeed(undefined)))
618
+ if (content === undefined) {
619
+ return {
620
+ type: MaybeApplyPatchVerified.CorrectnessError,
621
+ error: new Error(`Failed to read file for deletion: ${deletePath}`),
622
+ } satisfies MaybeApplyPatchVerifiedResult
623
+ }
624
+ changes.set(resolvedPath, {
625
+ type: "delete",
626
+ content,
627
+ })
628
+ break
629
+ }
630
+
631
+ case "update": {
632
+ const updatePath = path.resolve(effectiveCwd, hunk.path)
633
+ const originalText = yield* fs
634
+ .readFileString(updatePath)
635
+ .pipe(
636
+ Effect.catch((cause) =>
637
+ Effect.succeed(new Error(`Failed to read file ${updatePath}: ${cause}`, { cause })),
638
+ ),
639
+ )
640
+ if (originalText instanceof Error) {
641
+ return {
642
+ type: MaybeApplyPatchVerified.CorrectnessError,
643
+ error: originalText,
644
+ } satisfies MaybeApplyPatchVerifiedResult
645
+ }
646
+ try {
647
+ const fileUpdate = deriveNewContentsFromChunks(updatePath, hunk.chunks, originalText)
648
+ changes.set(resolvedPath, {
649
+ type: "update",
650
+ unified_diff: fileUpdate.unified_diff,
651
+ move_path: hunk.move_path ? path.resolve(effectiveCwd, hunk.move_path) : undefined,
652
+ new_content: fileUpdate.content,
653
+ })
654
+ } catch (error) {
655
+ return {
656
+ type: MaybeApplyPatchVerified.CorrectnessError,
657
+ error: error as Error,
658
+ } satisfies MaybeApplyPatchVerifiedResult
659
+ }
660
+ break
661
+ }
662
+ }
663
+ }
664
+
665
+ return {
666
+ type: MaybeApplyPatchVerified.Body,
667
+ action: {
668
+ changes,
669
+ patch: args.patch,
670
+ cwd: effectiveCwd,
671
+ },
672
+ } satisfies MaybeApplyPatchVerifiedResult
673
+ }
674
+
675
+ case MaybeApplyPatch.PatchParseError:
676
+ return {
677
+ type: MaybeApplyPatchVerified.CorrectnessError,
678
+ error: result.error,
679
+ } satisfies MaybeApplyPatchVerifiedResult
680
+
681
+ case MaybeApplyPatch.NotApplyPatch:
682
+ return { type: MaybeApplyPatchVerified.NotApplyPatch } satisfies MaybeApplyPatchVerifiedResult
683
+ }
684
+ })
685
+
686
+ export * as Patch from "."