opencode-v2 1.1.53

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 (439) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/opencode +84 -0
  5. package/bunfig.toml +5 -0
  6. package/package.json +126 -0
  7. package/parsers-config.ts +253 -0
  8. package/script/build.ts +193 -0
  9. package/script/postinstall.mjs +125 -0
  10. package/script/publish.ts +181 -0
  11. package/script/schema.ts +47 -0
  12. package/script/seed-e2e.ts +50 -0
  13. package/src/acp/README.md +164 -0
  14. package/src/acp/agent.ts +1676 -0
  15. package/src/acp/session.ts +117 -0
  16. package/src/acp/types.ts +23 -0
  17. package/src/agent/agent.ts +414 -0
  18. package/src/agent/generate.txt +75 -0
  19. package/src/agent/prompt/compaction.txt +12 -0
  20. package/src/agent/prompt/explore.txt +18 -0
  21. package/src/agent/prompt/summary.txt +11 -0
  22. package/src/agent/prompt/title.txt +44 -0
  23. package/src/auth/index.ts +70 -0
  24. package/src/bun/index.ts +137 -0
  25. package/src/bun/registry.ts +48 -0
  26. package/src/bus/bus-event.ts +43 -0
  27. package/src/bus/global.ts +10 -0
  28. package/src/bus/index.ts +105 -0
  29. package/src/cli/bootstrap.ts +17 -0
  30. package/src/cli/cmd/acp.ts +70 -0
  31. package/src/cli/cmd/agent.ts +257 -0
  32. package/src/cli/cmd/auth.ts +400 -0
  33. package/src/cli/cmd/cmd.ts +7 -0
  34. package/src/cli/cmd/debug/agent.ts +167 -0
  35. package/src/cli/cmd/debug/config.ts +16 -0
  36. package/src/cli/cmd/debug/file.ts +97 -0
  37. package/src/cli/cmd/debug/index.ts +48 -0
  38. package/src/cli/cmd/debug/lsp.ts +52 -0
  39. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  40. package/src/cli/cmd/debug/scrap.ts +16 -0
  41. package/src/cli/cmd/debug/skill.ts +16 -0
  42. package/src/cli/cmd/debug/snapshot.ts +52 -0
  43. package/src/cli/cmd/export.ts +88 -0
  44. package/src/cli/cmd/generate.ts +38 -0
  45. package/src/cli/cmd/github.ts +1540 -0
  46. package/src/cli/cmd/import.ts +147 -0
  47. package/src/cli/cmd/mcp.ts +755 -0
  48. package/src/cli/cmd/models.ts +77 -0
  49. package/src/cli/cmd/pr.ts +112 -0
  50. package/src/cli/cmd/run.ts +617 -0
  51. package/src/cli/cmd/serve.ts +20 -0
  52. package/src/cli/cmd/session.ts +135 -0
  53. package/src/cli/cmd/stats.ts +426 -0
  54. package/src/cli/cmd/tui/app.tsx +801 -0
  55. package/src/cli/cmd/tui/attach.ts +52 -0
  56. package/src/cli/cmd/tui/component/border.tsx +21 -0
  57. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  58. package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
  59. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  60. package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
  61. package/src/cli/cmd/tui/component/dialog-provider.tsx +266 -0
  62. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  63. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  64. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  65. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  66. package/src/cli/cmd/tui/component/dialog-status.tsx +177 -0
  67. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  68. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  69. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  70. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +666 -0
  71. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  72. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  73. package/src/cli/cmd/tui/component/prompt/index.tsx +1132 -0
  74. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  75. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  76. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  77. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  78. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  79. package/src/cli/cmd/tui/context/args.tsx +15 -0
  80. package/src/cli/cmd/tui/context/directory.ts +13 -0
  81. package/src/cli/cmd/tui/context/exit.tsx +52 -0
  82. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  83. package/src/cli/cmd/tui/context/keybind.tsx +100 -0
  84. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  85. package/src/cli/cmd/tui/context/local.tsx +409 -0
  86. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  87. package/src/cli/cmd/tui/context/route.tsx +46 -0
  88. package/src/cli/cmd/tui/context/sdk.tsx +101 -0
  89. package/src/cli/cmd/tui/context/sync.tsx +470 -0
  90. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  91. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  92. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  93. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  94. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  95. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  96. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  97. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  98. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  99. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  100. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  101. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  102. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  103. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  104. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  105. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  106. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  107. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  108. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  109. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  110. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  111. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  112. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  113. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  114. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  115. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  116. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  117. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  118. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  119. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  120. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  121. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  122. package/src/cli/cmd/tui/context/theme.tsx +1152 -0
  123. package/src/cli/cmd/tui/event.ts +48 -0
  124. package/src/cli/cmd/tui/routes/home.tsx +140 -0
  125. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  126. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  128. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  129. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  130. package/src/cli/cmd/tui/routes/session/header.tsx +142 -0
  131. package/src/cli/cmd/tui/routes/session/index.tsx +2126 -0
  132. package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
  133. package/src/cli/cmd/tui/routes/session/question.tsx +466 -0
  134. package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
  135. package/src/cli/cmd/tui/thread.ts +175 -0
  136. package/src/cli/cmd/tui/ui/dialog-alert.tsx +68 -0
  137. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +93 -0
  138. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +215 -0
  139. package/src/cli/cmd/tui/ui/dialog-help.tsx +49 -0
  140. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +88 -0
  141. package/src/cli/cmd/tui/ui/dialog-select.tsx +399 -0
  142. package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
  143. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  144. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  145. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  146. package/src/cli/cmd/tui/util/clipboard.ts +159 -0
  147. package/src/cli/cmd/tui/util/editor.ts +32 -0
  148. package/src/cli/cmd/tui/util/signal.ts +7 -0
  149. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  150. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  151. package/src/cli/cmd/tui/worker.ts +152 -0
  152. package/src/cli/cmd/uninstall.ts +357 -0
  153. package/src/cli/cmd/upgrade.ts +73 -0
  154. package/src/cli/cmd/web.ts +81 -0
  155. package/src/cli/error.ts +57 -0
  156. package/src/cli/logo.ts +6 -0
  157. package/src/cli/network.ts +60 -0
  158. package/src/cli/ui.ts +113 -0
  159. package/src/cli/upgrade.ts +25 -0
  160. package/src/command/index.ts +150 -0
  161. package/src/command/template/initialize.txt +10 -0
  162. package/src/command/template/review.txt +99 -0
  163. package/src/config/config.ts +1477 -0
  164. package/src/config/markdown.ts +98 -0
  165. package/src/env/index.ts +28 -0
  166. package/src/file/ignore.ts +83 -0
  167. package/src/file/index.ts +583 -0
  168. package/src/file/ripgrep.ts +375 -0
  169. package/src/file/time.ts +69 -0
  170. package/src/file/watcher.ts +127 -0
  171. package/src/flag/flag.ts +97 -0
  172. package/src/format/formatter.ts +366 -0
  173. package/src/format/index.ts +137 -0
  174. package/src/global/index.ts +55 -0
  175. package/src/id/id.ts +83 -0
  176. package/src/ide/index.ts +76 -0
  177. package/src/index.ts +159 -0
  178. package/src/installation/index.ts +246 -0
  179. package/src/lsp/client.ts +252 -0
  180. package/src/lsp/index.ts +485 -0
  181. package/src/lsp/language.ts +119 -0
  182. package/src/lsp/server.ts +2046 -0
  183. package/src/mcp/auth.ts +132 -0
  184. package/src/mcp/index.ts +934 -0
  185. package/src/mcp/oauth-callback.ts +200 -0
  186. package/src/mcp/oauth-provider.ts +154 -0
  187. package/src/patch/index.ts +680 -0
  188. package/src/permission/arity.ts +163 -0
  189. package/src/permission/index.ts +210 -0
  190. package/src/permission/next.ts +280 -0
  191. package/src/plugin/codex.ts +624 -0
  192. package/src/plugin/copilot.ts +327 -0
  193. package/src/plugin/index.ts +138 -0
  194. package/src/project/bootstrap.ts +35 -0
  195. package/src/project/instance.ts +114 -0
  196. package/src/project/project.ts +371 -0
  197. package/src/project/state.ts +70 -0
  198. package/src/project/vcs.ts +76 -0
  199. package/src/provider/auth.ts +147 -0
  200. package/src/provider/models.ts +133 -0
  201. package/src/provider/provider.ts +1262 -0
  202. package/src/provider/sdk/copilot/README.md +5 -0
  203. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +164 -0
  204. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  205. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +17 -0
  206. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  207. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +780 -0
  208. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  209. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  210. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +87 -0
  211. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  212. package/src/provider/sdk/copilot/index.ts +2 -0
  213. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  214. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +303 -0
  215. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  216. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  217. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  218. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +207 -0
  219. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1732 -0
  220. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +177 -0
  221. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  222. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +88 -0
  223. package/src/provider/sdk/copilot/responses/tool/file-search.ts +128 -0
  224. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +115 -0
  225. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +65 -0
  226. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +104 -0
  227. package/src/provider/sdk/copilot/responses/tool/web-search.ts +103 -0
  228. package/src/provider/transform.ts +828 -0
  229. package/src/pty/index.ts +250 -0
  230. package/src/question/index.ts +171 -0
  231. package/src/scheduler/index.ts +61 -0
  232. package/src/server/error.ts +36 -0
  233. package/src/server/event.ts +7 -0
  234. package/src/server/mdns.ts +60 -0
  235. package/src/server/routes/config.ts +92 -0
  236. package/src/server/routes/experimental.ts +208 -0
  237. package/src/server/routes/file.ts +197 -0
  238. package/src/server/routes/global.ts +183 -0
  239. package/src/server/routes/mcp.ts +225 -0
  240. package/src/server/routes/permission.ts +68 -0
  241. package/src/server/routes/project.ts +82 -0
  242. package/src/server/routes/provider.ts +165 -0
  243. package/src/server/routes/pty.ts +169 -0
  244. package/src/server/routes/question.ts +98 -0
  245. package/src/server/routes/session.ts +939 -0
  246. package/src/server/routes/tui.ts +379 -0
  247. package/src/server/server.ts +613 -0
  248. package/src/session/compaction.ts +226 -0
  249. package/src/session/index.ts +524 -0
  250. package/src/session/instruction.ts +197 -0
  251. package/src/session/llm.ts +289 -0
  252. package/src/session/message-v2.ts +802 -0
  253. package/src/session/message.ts +189 -0
  254. package/src/session/processor.ts +407 -0
  255. package/src/session/prompt/agent.txt +43 -0
  256. package/src/session/prompt/anthropic-20250930.txt +166 -0
  257. package/src/session/prompt/anthropic.txt +105 -0
  258. package/src/session/prompt/beast.txt +147 -0
  259. package/src/session/prompt/build-switch.txt +5 -0
  260. package/src/session/prompt/codex_header.txt +79 -0
  261. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  262. package/src/session/prompt/gemini.txt +155 -0
  263. package/src/session/prompt/max-steps.txt +16 -0
  264. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  265. package/src/session/prompt/plan.txt +26 -0
  266. package/src/session/prompt/qwen.txt +109 -0
  267. package/src/session/prompt/research.txt +81 -0
  268. package/src/session/prompt/trinity.txt +97 -0
  269. package/src/session/prompt.ts +1952 -0
  270. package/src/session/retry.ts +97 -0
  271. package/src/session/revert.ts +121 -0
  272. package/src/session/status.ts +76 -0
  273. package/src/session/summary.ts +217 -0
  274. package/src/session/system.ts +54 -0
  275. package/src/session/todo.ts +37 -0
  276. package/src/share/share-next.ts +200 -0
  277. package/src/share/share.ts +92 -0
  278. package/src/shell/shell.ts +67 -0
  279. package/src/skill/discovery.ts +97 -0
  280. package/src/skill/index.ts +1 -0
  281. package/src/skill/skill.ts +188 -0
  282. package/src/snapshot/index.ts +255 -0
  283. package/src/storage/storage.ts +227 -0
  284. package/src/tool/agent-enter.txt +1 -0
  285. package/src/tool/agent-exit.txt +1 -0
  286. package/src/tool/agent.ts +237 -0
  287. package/src/tool/apply_patch.ts +281 -0
  288. package/src/tool/apply_patch.txt +33 -0
  289. package/src/tool/bash.ts +269 -0
  290. package/src/tool/bash.txt +115 -0
  291. package/src/tool/batch.ts +175 -0
  292. package/src/tool/batch.txt +24 -0
  293. package/src/tool/chat-enter.txt +15 -0
  294. package/src/tool/chat-exit.txt +7 -0
  295. package/src/tool/chat.ts +217 -0
  296. package/src/tool/codesearch.ts +132 -0
  297. package/src/tool/codesearch.txt +12 -0
  298. package/src/tool/edit.ts +655 -0
  299. package/src/tool/edit.txt +10 -0
  300. package/src/tool/external-directory.ts +32 -0
  301. package/src/tool/glob.ts +78 -0
  302. package/src/tool/glob.txt +6 -0
  303. package/src/tool/grep.ts +147 -0
  304. package/src/tool/grep.txt +8 -0
  305. package/src/tool/invalid.ts +17 -0
  306. package/src/tool/ls.ts +121 -0
  307. package/src/tool/ls.txt +1 -0
  308. package/src/tool/lsp.ts +96 -0
  309. package/src/tool/lsp.txt +19 -0
  310. package/src/tool/multiedit.ts +46 -0
  311. package/src/tool/multiedit.txt +41 -0
  312. package/src/tool/plan-enter.txt +14 -0
  313. package/src/tool/plan-exit.txt +13 -0
  314. package/src/tool/plan.ts +130 -0
  315. package/src/tool/question.ts +33 -0
  316. package/src/tool/question.txt +10 -0
  317. package/src/tool/read.ts +211 -0
  318. package/src/tool/read.txt +12 -0
  319. package/src/tool/registry.ts +167 -0
  320. package/src/tool/research-enter.txt +1 -0
  321. package/src/tool/research-exit.txt +1 -0
  322. package/src/tool/research.ts +134 -0
  323. package/src/tool/skill.ts +123 -0
  324. package/src/tool/task.ts +165 -0
  325. package/src/tool/task.txt +60 -0
  326. package/src/tool/todo.ts +53 -0
  327. package/src/tool/todoread.txt +14 -0
  328. package/src/tool/todowrite.txt +167 -0
  329. package/src/tool/tool.ts +89 -0
  330. package/src/tool/truncation.ts +106 -0
  331. package/src/tool/webfetch.ts +186 -0
  332. package/src/tool/webfetch.txt +13 -0
  333. package/src/tool/websearch.ts +150 -0
  334. package/src/tool/websearch.txt +14 -0
  335. package/src/tool/write.ts +85 -0
  336. package/src/tool/write.txt +8 -0
  337. package/src/util/abort.ts +35 -0
  338. package/src/util/archive.ts +16 -0
  339. package/src/util/color.ts +19 -0
  340. package/src/util/context.ts +25 -0
  341. package/src/util/defer.ts +12 -0
  342. package/src/util/eventloop.ts +20 -0
  343. package/src/util/filesystem.ts +93 -0
  344. package/src/util/fn.ts +11 -0
  345. package/src/util/format.ts +20 -0
  346. package/src/util/iife.ts +3 -0
  347. package/src/util/keybind.ts +103 -0
  348. package/src/util/lazy.ts +18 -0
  349. package/src/util/locale.ts +81 -0
  350. package/src/util/lock.ts +98 -0
  351. package/src/util/log.ts +180 -0
  352. package/src/util/proxied.ts +3 -0
  353. package/src/util/queue.ts +32 -0
  354. package/src/util/rpc.ts +66 -0
  355. package/src/util/scrap.ts +10 -0
  356. package/src/util/signal.ts +12 -0
  357. package/src/util/timeout.ts +14 -0
  358. package/src/util/token.ts +7 -0
  359. package/src/util/wildcard.ts +56 -0
  360. package/src/worktree/index.ts +574 -0
  361. package/sst-env.d.ts +9 -0
  362. package/test/acp/agent-interface.test.ts +51 -0
  363. package/test/acp/event-subscription.test.ts +436 -0
  364. package/test/agent/agent.test.ts +675 -0
  365. package/test/bun.test.ts +53 -0
  366. package/test/cli/github-action.test.ts +161 -0
  367. package/test/cli/github-remote.test.ts +80 -0
  368. package/test/cli/import.test.ts +38 -0
  369. package/test/cli/tui/transcript.test.ts +322 -0
  370. package/test/config/agent-color.test.ts +71 -0
  371. package/test/config/config.test.ts +1802 -0
  372. package/test/config/fixtures/empty-frontmatter.md +4 -0
  373. package/test/config/fixtures/frontmatter.md +28 -0
  374. package/test/config/fixtures/markdown-header.md +11 -0
  375. package/test/config/fixtures/no-frontmatter.md +1 -0
  376. package/test/config/fixtures/weird-model-id.md +13 -0
  377. package/test/config/markdown.test.ts +228 -0
  378. package/test/file/ignore.test.ts +10 -0
  379. package/test/file/path-traversal.test.ts +198 -0
  380. package/test/file/ripgrep.test.ts +39 -0
  381. package/test/fixture/fixture.ts +45 -0
  382. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  383. package/test/ide/ide.test.ts +82 -0
  384. package/test/keybind.test.ts +421 -0
  385. package/test/lsp/client.test.ts +95 -0
  386. package/test/mcp/headers.test.ts +153 -0
  387. package/test/mcp/oauth-browser.test.ts +249 -0
  388. package/test/memory/abort-leak.test.ts +136 -0
  389. package/test/patch/patch.test.ts +348 -0
  390. package/test/permission/arity.test.ts +33 -0
  391. package/test/permission/next.test.ts +690 -0
  392. package/test/permission-task.test.ts +319 -0
  393. package/test/plugin/auth-override.test.ts +44 -0
  394. package/test/plugin/codex.test.ts +123 -0
  395. package/test/preload.ts +63 -0
  396. package/test/project/project.test.ts +120 -0
  397. package/test/provider/amazon-bedrock.test.ts +445 -0
  398. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  399. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  400. package/test/provider/gitlab-duo.test.ts +262 -0
  401. package/test/provider/provider.test.ts +2129 -0
  402. package/test/provider/transform.test.ts +2022 -0
  403. package/test/question/question.test.ts +300 -0
  404. package/test/scheduler.test.ts +73 -0
  405. package/test/server/session-list.test.ts +39 -0
  406. package/test/server/session-select.test.ts +78 -0
  407. package/test/session/compaction.test.ts +293 -0
  408. package/test/session/instruction.test.ts +170 -0
  409. package/test/session/llm.test.ts +691 -0
  410. package/test/session/message-v2.test.ts +786 -0
  411. package/test/session/prompt-missing-file.test.ts +53 -0
  412. package/test/session/prompt-special-chars.test.ts +56 -0
  413. package/test/session/prompt-variant.test.ts +60 -0
  414. package/test/session/retry.test.ts +179 -0
  415. package/test/session/revert-compact.test.ts +285 -0
  416. package/test/session/session.test.ts +71 -0
  417. package/test/skill/discovery.test.ts +60 -0
  418. package/test/skill/skill.test.ts +388 -0
  419. package/test/snapshot/snapshot.test.ts +1040 -0
  420. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  421. package/test/tool/apply_patch.test.ts +559 -0
  422. package/test/tool/bash.test.ts +399 -0
  423. package/test/tool/external-directory.test.ts +127 -0
  424. package/test/tool/fixtures/large-image.png +0 -0
  425. package/test/tool/fixtures/models-api.json +38413 -0
  426. package/test/tool/grep.test.ts +110 -0
  427. package/test/tool/question.test.ts +107 -0
  428. package/test/tool/read.test.ts +358 -0
  429. package/test/tool/registry.test.ts +122 -0
  430. package/test/tool/skill.test.ts +112 -0
  431. package/test/tool/truncation.test.ts +159 -0
  432. package/test/util/filesystem.test.ts +39 -0
  433. package/test/util/format.test.ts +59 -0
  434. package/test/util/iife.test.ts +36 -0
  435. package/test/util/lazy.test.ts +50 -0
  436. package/test/util/lock.test.ts +72 -0
  437. package/test/util/timeout.test.ts +21 -0
  438. package/test/util/wildcard.test.ts +75 -0
  439. package/tsconfig.json +16 -0
@@ -0,0 +1,237 @@
1
+ import z from "zod"
2
+ import path from "path"
3
+ import { Tool } from "./tool"
4
+ import { Question } from "../question"
5
+ import { Session } from "../session"
6
+ import { MessageV2 } from "../session/message-v2"
7
+ import { Identifier } from "../id/id"
8
+ import { Provider } from "../provider/provider"
9
+ import { Instance } from "../project/instance"
10
+ import { PermissionNext } from "@/permission/next"
11
+ import EXIT_DESCRIPTION from "./agent-exit.txt"
12
+ import ENTER_DESCRIPTION from "./agent-enter.txt"
13
+
14
+ async function getLastModel(sessionID: string) {
15
+ for await (const item of MessageV2.stream(sessionID)) {
16
+ if (item.info.role === "user" && item.info.model) return item.info.model
17
+ }
18
+ return Provider.defaultModel()
19
+ }
20
+
21
+ export const AgentExitTool = Tool.define("agent_exit", {
22
+ description: EXIT_DESCRIPTION,
23
+ parameters: z.object({}),
24
+ async execute(_params, ctx) {
25
+ const session = await Session.get(ctx.sessionID)
26
+ const plan = path.relative(Instance.worktree, Session.plan(session))
27
+
28
+ const answers = await Question.ask({
29
+ sessionID: ctx.sessionID,
30
+ questions: [
31
+ {
32
+ question: "Would you like to exit agent mode and switch to a different mode?",
33
+ header: "Exit Agent Mode",
34
+ custom: false,
35
+ options: [
36
+ { label: "Chat", description: "Switch to chat agent for general conversation" },
37
+ { label: "Research", description: "Switch to research agent for web research" },
38
+ ],
39
+ },
40
+ ],
41
+ tool: ctx.callID ? { messageID: ctx.messageID, callID: ctx.callID } : undefined,
42
+ })
43
+
44
+ const answer = answers[0]?.[0]
45
+ if (!answer) throw new Question.RejectedError()
46
+
47
+ const targetAgent = answer.toLowerCase()
48
+ const model = await getLastModel(ctx.sessionID)
49
+
50
+ const userMsg: MessageV2.User = {
51
+ id: Identifier.ascending("message"),
52
+ sessionID: ctx.sessionID,
53
+ role: "user",
54
+ time: {
55
+ created: Date.now(),
56
+ },
57
+ agent: targetAgent,
58
+ model,
59
+ }
60
+ await Session.updateMessage(userMsg)
61
+ await Session.updatePart({
62
+ id: Identifier.ascending("part"),
63
+ messageID: userMsg.id,
64
+ sessionID: ctx.sessionID,
65
+ type: "text",
66
+ text: `User has requested to exit agent mode and switch to ${targetAgent} mode.`,
67
+ synthetic: true,
68
+ } satisfies MessageV2.TextPart)
69
+
70
+ return {
71
+ title: `Switching to ${targetAgent} agent`,
72
+ output: `User confirmed to switch to ${targetAgent} mode.`,
73
+ metadata: {},
74
+ }
75
+ },
76
+ })
77
+
78
+ export const AgentEnterTool = Tool.define("agent_enter", {
79
+ description: ENTER_DESCRIPTION,
80
+ parameters: z.object({}),
81
+ async execute(_params, ctx) {
82
+ const session = await Session.get(ctx.sessionID)
83
+ const messages = await Session.messages({ sessionID: session.id })
84
+
85
+ const hasAssistantMessages = messages.some((msg) => msg.info.role === "assistant")
86
+
87
+ if (hasAssistantMessages) {
88
+ return {
89
+ title: "Cannot enter agent mode",
90
+ output: "Agent mode can only be entered when starting a new conversation, not in an existing thread.",
91
+ metadata: {},
92
+ }
93
+ }
94
+
95
+ const answers = await Question.ask({
96
+ sessionID: ctx.sessionID,
97
+ questions: [
98
+ {
99
+ question: "Would you like to switch to agent mode? Agent mode combines planning and execution - you plan first, then approve execution.",
100
+ header: "Agent Mode",
101
+ custom: false,
102
+ options: [
103
+ { label: "Yes", description: "Switch to agent mode for unified planning and execution" },
104
+ { label: "No", description: "Stay with current agent" },
105
+ ],
106
+ },
107
+ ],
108
+ tool: ctx.callID ? { messageID: ctx.messageID, callID: ctx.callID } : undefined,
109
+ })
110
+
111
+ const answer = answers[0]?.[0]
112
+
113
+ if (answer === "No") throw new Question.RejectedError()
114
+
115
+ const model = await getLastModel(ctx.sessionID)
116
+
117
+ const userMsg: MessageV2.User = {
118
+ id: Identifier.ascending("message"),
119
+ sessionID: ctx.sessionID,
120
+ role: "user",
121
+ time: {
122
+ created: Date.now(),
123
+ },
124
+ agent: "agent",
125
+ model,
126
+ }
127
+ await Session.updateMessage(userMsg)
128
+ await Session.updatePart({
129
+ id: Identifier.ascending("part"),
130
+ messageID: userMsg.id,
131
+ sessionID: ctx.sessionID,
132
+ type: "text",
133
+ text: "User has requested to enter agent mode. Begin with planning phase - understand the request, ask clarifying questions, then create a plan for user approval.",
134
+ synthetic: true,
135
+ } satisfies MessageV2.TextPart)
136
+
137
+ return {
138
+ title: "Switching to agent mode",
139
+ output: "User confirmed to switch to agent mode. Begin with the planning workflow.",
140
+ metadata: {},
141
+ }
142
+ },
143
+ })
144
+
145
+ export const AgentApprovePlanTool = Tool.define("agent_approve_plan", {
146
+ description: "Approve the plan and switch from planning phase to execution phase. This grants full file system access for implementation.",
147
+ parameters: z.object({}),
148
+ async execute(_params, ctx) {
149
+ const session = await Session.get(ctx.sessionID)
150
+ const plan = path.relative(Instance.worktree, Session.plan(session))
151
+
152
+ const answers = await Question.ask({
153
+ sessionID: ctx.sessionID,
154
+ questions: [
155
+ {
156
+ question: `Approve the plan at ${plan} and switch to execution mode? You'll have full file system access after approval.`,
157
+ header: "Approve Plan",
158
+ custom: false,
159
+ options: [
160
+ { label: "Yes", description: "Approve plan and grant full execution permissions" },
161
+ { label: "No", description: "Continue refining the plan" },
162
+ ],
163
+ },
164
+ ],
165
+ tool: ctx.callID ? { messageID: ctx.messageID, callID: ctx.callID } : undefined,
166
+ })
167
+
168
+ const answer = answers[0]?.[0]
169
+ if (answer === "No") throw new Question.RejectedError()
170
+
171
+ const model = await getLastModel(ctx.sessionID)
172
+
173
+ const userMsg: MessageV2.User = {
174
+ id: Identifier.ascending("message"),
175
+ sessionID: ctx.sessionID,
176
+ role: "user",
177
+ time: {
178
+ created: Date.now(),
179
+ },
180
+ agent: "agent",
181
+ model,
182
+ }
183
+ await Session.updateMessage(userMsg)
184
+ await Session.updatePart({
185
+ id: Identifier.ascending("part"),
186
+ messageID: userMsg.id,
187
+ sessionID: ctx.sessionID,
188
+ type: "text",
189
+ text: `Plan at ${plan} has been approved. You now have full file system access. Create a todo list and begin implementation.`,
190
+ synthetic: true,
191
+ } satisfies MessageV2.TextPart)
192
+
193
+ return {
194
+ title: "Plan approved - execution mode enabled",
195
+ output: `Plan at ${plan} approved. You now have full file system access. Begin by creating a todo list from the plan.`,
196
+ metadata: {},
197
+ }
198
+ },
199
+ })
200
+
201
+ export const AgentDenyPlanTool = Tool.define("agent_deny_plan", {
202
+ description: "Deny the current plan and return to planning phase. Use this to request changes to the plan before execution.",
203
+ parameters: z.object({
204
+ reason: z.string().optional().describe("Reason for denying the plan"),
205
+ }),
206
+ async execute(params, ctx) {
207
+ const session = await Session.get(ctx.sessionID)
208
+
209
+ const model = await getLastModel(ctx.sessionID)
210
+
211
+ const userMsg: MessageV2.User = {
212
+ id: Identifier.ascending("message"),
213
+ sessionID: ctx.sessionID,
214
+ role: "user",
215
+ time: {
216
+ created: Date.now(),
217
+ },
218
+ agent: "agent",
219
+ model,
220
+ }
221
+ await Session.updateMessage(userMsg)
222
+ await Session.updatePart({
223
+ id: Identifier.ascending("part"),
224
+ messageID: userMsg.id,
225
+ sessionID: ctx.sessionID,
226
+ type: "text",
227
+ text: `Plan has been denied. Reason: ${params.reason ?? "No reason provided"}. Continue planning and revise the plan based on feedback.`,
228
+ synthetic: true,
229
+ } satisfies MessageV2.TextPart)
230
+
231
+ return {
232
+ title: "Plan denied - returning to planning",
233
+ output: "Continue refining the plan based on user feedback.",
234
+ metadata: {},
235
+ }
236
+ },
237
+ })
@@ -0,0 +1,281 @@
1
+ import z from "zod"
2
+ import * as path from "path"
3
+ import * as fs from "fs/promises"
4
+ import { Tool } from "./tool"
5
+ import { Bus } from "../bus"
6
+ import { FileWatcher } from "../file/watcher"
7
+ import { Instance } from "../project/instance"
8
+ import { Patch } from "../patch"
9
+ import { createTwoFilesPatch, diffLines } from "diff"
10
+ import { assertExternalDirectory } from "./external-directory"
11
+ import { trimDiff } from "./edit"
12
+ import { LSP } from "../lsp"
13
+ import { Filesystem } from "../util/filesystem"
14
+ import DESCRIPTION from "./apply_patch.txt"
15
+ import { File } from "../file"
16
+
17
+ const PatchParams = z.object({
18
+ patchText: z.string().describe("The full patch text that describes all changes to be made"),
19
+ })
20
+
21
+ export const ApplyPatchTool = Tool.define("apply_patch", {
22
+ description: DESCRIPTION,
23
+ parameters: PatchParams,
24
+ async execute(params, ctx) {
25
+ if (!params.patchText) {
26
+ throw new Error("patchText is required")
27
+ }
28
+
29
+ // Parse the patch to get hunks
30
+ let hunks: Patch.Hunk[]
31
+ try {
32
+ const parseResult = Patch.parsePatch(params.patchText)
33
+ hunks = parseResult.hunks
34
+ } catch (error) {
35
+ throw new Error(`apply_patch verification failed: ${error}`)
36
+ }
37
+
38
+ if (hunks.length === 0) {
39
+ const normalized = params.patchText.replace(/\r\n/g, "\n").replace(/\r/g, "\n").trim()
40
+ if (normalized === "*** Begin Patch\n*** End Patch") {
41
+ throw new Error("patch rejected: empty patch")
42
+ }
43
+ throw new Error("apply_patch verification failed: no hunks found")
44
+ }
45
+
46
+ // Validate file paths and check permissions
47
+ const fileChanges: Array<{
48
+ filePath: string
49
+ oldContent: string
50
+ newContent: string
51
+ type: "add" | "update" | "delete" | "move"
52
+ movePath?: string
53
+ diff: string
54
+ additions: number
55
+ deletions: number
56
+ }> = []
57
+
58
+ let totalDiff = ""
59
+
60
+ for (const hunk of hunks) {
61
+ const filePath = path.resolve(Instance.directory, hunk.path)
62
+ await assertExternalDirectory(ctx, filePath)
63
+
64
+ switch (hunk.type) {
65
+ case "add": {
66
+ const oldContent = ""
67
+ const newContent =
68
+ hunk.contents.length === 0 || hunk.contents.endsWith("\n") ? hunk.contents : `${hunk.contents}\n`
69
+ const diff = trimDiff(createTwoFilesPatch(filePath, filePath, oldContent, newContent))
70
+
71
+ let additions = 0
72
+ let deletions = 0
73
+ for (const change of diffLines(oldContent, newContent)) {
74
+ if (change.added) additions += change.count || 0
75
+ if (change.removed) deletions += change.count || 0
76
+ }
77
+
78
+ fileChanges.push({
79
+ filePath,
80
+ oldContent,
81
+ newContent,
82
+ type: "add",
83
+ diff,
84
+ additions,
85
+ deletions,
86
+ })
87
+
88
+ totalDiff += diff + "\n"
89
+ break
90
+ }
91
+
92
+ case "update": {
93
+ // Check if file exists for update
94
+ const stats = await fs.stat(filePath).catch(() => null)
95
+ if (!stats || stats.isDirectory()) {
96
+ throw new Error(`apply_patch verification failed: Failed to read file to update: ${filePath}`)
97
+ }
98
+
99
+ const oldContent = await fs.readFile(filePath, "utf-8")
100
+ let newContent = oldContent
101
+
102
+ // Apply the update chunks to get new content
103
+ try {
104
+ const fileUpdate = Patch.deriveNewContentsFromChunks(filePath, hunk.chunks)
105
+ newContent = fileUpdate.content
106
+ } catch (error) {
107
+ throw new Error(`apply_patch verification failed: ${error}`)
108
+ }
109
+
110
+ const diff = trimDiff(createTwoFilesPatch(filePath, filePath, oldContent, newContent))
111
+
112
+ let additions = 0
113
+ let deletions = 0
114
+ for (const change of diffLines(oldContent, newContent)) {
115
+ if (change.added) additions += change.count || 0
116
+ if (change.removed) deletions += change.count || 0
117
+ }
118
+
119
+ const movePath = hunk.move_path ? path.resolve(Instance.directory, hunk.move_path) : undefined
120
+ await assertExternalDirectory(ctx, movePath)
121
+
122
+ fileChanges.push({
123
+ filePath,
124
+ oldContent,
125
+ newContent,
126
+ type: hunk.move_path ? "move" : "update",
127
+ movePath,
128
+ diff,
129
+ additions,
130
+ deletions,
131
+ })
132
+
133
+ totalDiff += diff + "\n"
134
+ break
135
+ }
136
+
137
+ case "delete": {
138
+ const contentToDelete = await fs.readFile(filePath, "utf-8").catch((error) => {
139
+ throw new Error(`apply_patch verification failed: ${error}`)
140
+ })
141
+ const deleteDiff = trimDiff(createTwoFilesPatch(filePath, filePath, contentToDelete, ""))
142
+
143
+ const deletions = contentToDelete.split("\n").length
144
+
145
+ fileChanges.push({
146
+ filePath,
147
+ oldContent: contentToDelete,
148
+ newContent: "",
149
+ type: "delete",
150
+ diff: deleteDiff,
151
+ additions: 0,
152
+ deletions,
153
+ })
154
+
155
+ totalDiff += deleteDiff + "\n"
156
+ break
157
+ }
158
+ }
159
+ }
160
+
161
+ // Build per-file metadata for UI rendering (used for both permission and result)
162
+ const files = fileChanges.map((change) => ({
163
+ filePath: change.filePath,
164
+ relativePath: path.relative(Instance.worktree, change.movePath ?? change.filePath),
165
+ type: change.type,
166
+ diff: change.diff,
167
+ before: change.oldContent,
168
+ after: change.newContent,
169
+ additions: change.additions,
170
+ deletions: change.deletions,
171
+ movePath: change.movePath,
172
+ }))
173
+
174
+ // Check permissions if needed
175
+ const relativePaths = fileChanges.map((c) => path.relative(Instance.worktree, c.filePath))
176
+ await ctx.ask({
177
+ permission: "edit",
178
+ patterns: relativePaths,
179
+ always: ["*"],
180
+ metadata: {
181
+ filepath: relativePaths.join(", "),
182
+ diff: totalDiff,
183
+ files,
184
+ },
185
+ })
186
+
187
+ // Apply the changes
188
+ const updates: Array<{ file: string; event: "add" | "change" | "unlink" }> = []
189
+
190
+ for (const change of fileChanges) {
191
+ const edited = change.type === "delete" ? undefined : (change.movePath ?? change.filePath)
192
+ switch (change.type) {
193
+ case "add":
194
+ // Create parent directories (recursive: true is safe on existing/root dirs)
195
+ await fs.mkdir(path.dirname(change.filePath), { recursive: true })
196
+ await fs.writeFile(change.filePath, change.newContent, "utf-8")
197
+ updates.push({ file: change.filePath, event: "add" })
198
+ break
199
+
200
+ case "update":
201
+ await fs.writeFile(change.filePath, change.newContent, "utf-8")
202
+ updates.push({ file: change.filePath, event: "change" })
203
+ break
204
+
205
+ case "move":
206
+ if (change.movePath) {
207
+ // Create parent directories (recursive: true is safe on existing/root dirs)
208
+ await fs.mkdir(path.dirname(change.movePath), { recursive: true })
209
+ await fs.writeFile(change.movePath, change.newContent, "utf-8")
210
+ await fs.unlink(change.filePath)
211
+ updates.push({ file: change.filePath, event: "unlink" })
212
+ updates.push({ file: change.movePath, event: "add" })
213
+ }
214
+ break
215
+
216
+ case "delete":
217
+ await fs.unlink(change.filePath)
218
+ updates.push({ file: change.filePath, event: "unlink" })
219
+ break
220
+ }
221
+
222
+ if (edited) {
223
+ await Bus.publish(File.Event.Edited, {
224
+ file: edited,
225
+ })
226
+ }
227
+ }
228
+
229
+ // Publish file change events
230
+ for (const update of updates) {
231
+ await Bus.publish(FileWatcher.Event.Updated, update)
232
+ }
233
+
234
+ // Notify LSP of file changes and collect diagnostics
235
+ for (const change of fileChanges) {
236
+ if (change.type === "delete") continue
237
+ const target = change.movePath ?? change.filePath
238
+ await LSP.touchFile(target, true)
239
+ }
240
+ const diagnostics = await LSP.diagnostics()
241
+
242
+ // Generate output summary
243
+ const summaryLines = fileChanges.map((change) => {
244
+ if (change.type === "add") {
245
+ return `A ${path.relative(Instance.worktree, change.filePath)}`
246
+ }
247
+ if (change.type === "delete") {
248
+ return `D ${path.relative(Instance.worktree, change.filePath)}`
249
+ }
250
+ const target = change.movePath ?? change.filePath
251
+ return `M ${path.relative(Instance.worktree, target)}`
252
+ })
253
+ let output = `Success. Updated the following files:\n${summaryLines.join("\n")}`
254
+
255
+ // Report LSP errors for changed files
256
+ const MAX_DIAGNOSTICS_PER_FILE = 20
257
+ for (const change of fileChanges) {
258
+ if (change.type === "delete") continue
259
+ const target = change.movePath ?? change.filePath
260
+ const normalized = Filesystem.normalizePath(target)
261
+ const issues = diagnostics[normalized] ?? []
262
+ const errors = issues.filter((item) => item.severity === 1)
263
+ if (errors.length > 0) {
264
+ const limited = errors.slice(0, MAX_DIAGNOSTICS_PER_FILE)
265
+ const suffix =
266
+ errors.length > MAX_DIAGNOSTICS_PER_FILE ? `\n... and ${errors.length - MAX_DIAGNOSTICS_PER_FILE} more` : ""
267
+ output += `\n\nLSP errors detected in ${path.relative(Instance.worktree, target)}, please fix:\n<diagnostics file="${target}">\n${limited.map(LSP.Diagnostic.pretty).join("\n")}${suffix}\n</diagnostics>`
268
+ }
269
+ }
270
+
271
+ return {
272
+ title: output,
273
+ metadata: {
274
+ diff: totalDiff,
275
+ files,
276
+ diagnostics,
277
+ },
278
+ output,
279
+ }
280
+ },
281
+ })
@@ -0,0 +1,33 @@
1
+ Use the `apply_patch` tool to edit files. Your patch language is a stripped‑down, file‑oriented diff format designed to be easy to parse and safe to apply. You can think of it as a high‑level envelope:
2
+
3
+ *** Begin Patch
4
+ [ one or more file sections ]
5
+ *** End Patch
6
+
7
+ Within that envelope, you get a sequence of file operations.
8
+ You MUST include a header to specify the action you are taking.
9
+ Each operation starts with one of three headers:
10
+
11
+ *** Add File: <path> - create a new file. Every following line is a + line (the initial contents).
12
+ *** Delete File: <path> - remove an existing file. Nothing follows.
13
+ *** Update File: <path> - patch an existing file in place (optionally with a rename).
14
+
15
+ Example patch:
16
+
17
+ ```
18
+ *** Begin Patch
19
+ *** Add File: hello.txt
20
+ +Hello world
21
+ *** Update File: src/app.py
22
+ *** Move to: src/main.py
23
+ @@ def greet():
24
+ -print("Hi")
25
+ +print("Hello, world!")
26
+ *** Delete File: obsolete.txt
27
+ *** End Patch
28
+ ```
29
+
30
+ It is important to remember:
31
+
32
+ - You must include a header with your intended action (Add/Delete/Update)
33
+ - You must prefix new lines with `+` even when creating a new file