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,802 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import z from "zod"
3
+ import { NamedError } from "@opencode-ai/util/error"
4
+ import { APICallError, convertToModelMessages, LoadAPIKeyError, type ModelMessage, type UIMessage } from "ai"
5
+ import { Identifier } from "../id/id"
6
+ import { LSP } from "../lsp"
7
+ import { Snapshot } from "@/snapshot"
8
+ import { fn } from "@/util/fn"
9
+ import { Storage } from "@/storage/storage"
10
+ import { ProviderTransform } from "@/provider/transform"
11
+ import { STATUS_CODES } from "http"
12
+ import { iife } from "@/util/iife"
13
+ import { type SystemError } from "bun"
14
+ import type { Provider } from "@/provider/provider"
15
+
16
+ export namespace MessageV2 {
17
+ export const OutputLengthError = NamedError.create("MessageOutputLengthError", z.object({}))
18
+ export const AbortedError = NamedError.create("MessageAbortedError", z.object({ message: z.string() }))
19
+ export const AuthError = NamedError.create(
20
+ "ProviderAuthError",
21
+ z.object({
22
+ providerID: z.string(),
23
+ message: z.string(),
24
+ }),
25
+ )
26
+ export const APIError = NamedError.create(
27
+ "APIError",
28
+ z.object({
29
+ message: z.string(),
30
+ statusCode: z.number().optional(),
31
+ isRetryable: z.boolean(),
32
+ responseHeaders: z.record(z.string(), z.string()).optional(),
33
+ responseBody: z.string().optional(),
34
+ metadata: z.record(z.string(), z.string()).optional(),
35
+ }),
36
+ )
37
+ export type APIError = z.infer<typeof APIError.Schema>
38
+
39
+ const PartBase = z.object({
40
+ id: z.string(),
41
+ sessionID: z.string(),
42
+ messageID: z.string(),
43
+ })
44
+
45
+ export const SnapshotPart = PartBase.extend({
46
+ type: z.literal("snapshot"),
47
+ snapshot: z.string(),
48
+ }).meta({
49
+ ref: "SnapshotPart",
50
+ })
51
+ export type SnapshotPart = z.infer<typeof SnapshotPart>
52
+
53
+ export const PatchPart = PartBase.extend({
54
+ type: z.literal("patch"),
55
+ hash: z.string(),
56
+ files: z.string().array(),
57
+ }).meta({
58
+ ref: "PatchPart",
59
+ })
60
+ export type PatchPart = z.infer<typeof PatchPart>
61
+
62
+ export const TextPart = PartBase.extend({
63
+ type: z.literal("text"),
64
+ text: z.string(),
65
+ synthetic: z.boolean().optional(),
66
+ ignored: z.boolean().optional(),
67
+ time: z
68
+ .object({
69
+ start: z.number(),
70
+ end: z.number().optional(),
71
+ })
72
+ .optional(),
73
+ metadata: z.record(z.string(), z.any()).optional(),
74
+ }).meta({
75
+ ref: "TextPart",
76
+ })
77
+ export type TextPart = z.infer<typeof TextPart>
78
+
79
+ export const ReasoningPart = PartBase.extend({
80
+ type: z.literal("reasoning"),
81
+ text: z.string(),
82
+ metadata: z.record(z.string(), z.any()).optional(),
83
+ time: z.object({
84
+ start: z.number(),
85
+ end: z.number().optional(),
86
+ }),
87
+ }).meta({
88
+ ref: "ReasoningPart",
89
+ })
90
+ export type ReasoningPart = z.infer<typeof ReasoningPart>
91
+
92
+ const FilePartSourceBase = z.object({
93
+ text: z
94
+ .object({
95
+ value: z.string(),
96
+ start: z.number().int(),
97
+ end: z.number().int(),
98
+ })
99
+ .meta({
100
+ ref: "FilePartSourceText",
101
+ }),
102
+ })
103
+
104
+ export const FileSource = FilePartSourceBase.extend({
105
+ type: z.literal("file"),
106
+ path: z.string(),
107
+ }).meta({
108
+ ref: "FileSource",
109
+ })
110
+
111
+ export const SymbolSource = FilePartSourceBase.extend({
112
+ type: z.literal("symbol"),
113
+ path: z.string(),
114
+ range: LSP.Range,
115
+ name: z.string(),
116
+ kind: z.number().int(),
117
+ }).meta({
118
+ ref: "SymbolSource",
119
+ })
120
+
121
+ export const ResourceSource = FilePartSourceBase.extend({
122
+ type: z.literal("resource"),
123
+ clientName: z.string(),
124
+ uri: z.string(),
125
+ }).meta({
126
+ ref: "ResourceSource",
127
+ })
128
+
129
+ export const FilePartSource = z.discriminatedUnion("type", [FileSource, SymbolSource, ResourceSource]).meta({
130
+ ref: "FilePartSource",
131
+ })
132
+
133
+ export const FilePart = PartBase.extend({
134
+ type: z.literal("file"),
135
+ mime: z.string(),
136
+ filename: z.string().optional(),
137
+ url: z.string(),
138
+ source: FilePartSource.optional(),
139
+ }).meta({
140
+ ref: "FilePart",
141
+ })
142
+ export type FilePart = z.infer<typeof FilePart>
143
+
144
+ export const AgentPart = PartBase.extend({
145
+ type: z.literal("agent"),
146
+ name: z.string(),
147
+ source: z
148
+ .object({
149
+ value: z.string(),
150
+ start: z.number().int(),
151
+ end: z.number().int(),
152
+ })
153
+ .optional(),
154
+ }).meta({
155
+ ref: "AgentPart",
156
+ })
157
+ export type AgentPart = z.infer<typeof AgentPart>
158
+
159
+ export const CompactionPart = PartBase.extend({
160
+ type: z.literal("compaction"),
161
+ auto: z.boolean(),
162
+ }).meta({
163
+ ref: "CompactionPart",
164
+ })
165
+ export type CompactionPart = z.infer<typeof CompactionPart>
166
+
167
+ export const SubtaskPart = PartBase.extend({
168
+ type: z.literal("subtask"),
169
+ prompt: z.string(),
170
+ description: z.string(),
171
+ agent: z.string(),
172
+ model: z
173
+ .object({
174
+ providerID: z.string(),
175
+ modelID: z.string(),
176
+ })
177
+ .optional(),
178
+ command: z.string().optional(),
179
+ }).meta({
180
+ ref: "SubtaskPart",
181
+ })
182
+ export type SubtaskPart = z.infer<typeof SubtaskPart>
183
+
184
+ export const RetryPart = PartBase.extend({
185
+ type: z.literal("retry"),
186
+ attempt: z.number(),
187
+ error: APIError.Schema,
188
+ time: z.object({
189
+ created: z.number(),
190
+ }),
191
+ }).meta({
192
+ ref: "RetryPart",
193
+ })
194
+ export type RetryPart = z.infer<typeof RetryPart>
195
+
196
+ export const StepStartPart = PartBase.extend({
197
+ type: z.literal("step-start"),
198
+ snapshot: z.string().optional(),
199
+ }).meta({
200
+ ref: "StepStartPart",
201
+ })
202
+ export type StepStartPart = z.infer<typeof StepStartPart>
203
+
204
+ export const StepFinishPart = PartBase.extend({
205
+ type: z.literal("step-finish"),
206
+ reason: z.string(),
207
+ snapshot: z.string().optional(),
208
+ cost: z.number(),
209
+ tokens: z.object({
210
+ input: z.number(),
211
+ output: z.number(),
212
+ reasoning: z.number(),
213
+ cache: z.object({
214
+ read: z.number(),
215
+ write: z.number(),
216
+ }),
217
+ }),
218
+ }).meta({
219
+ ref: "StepFinishPart",
220
+ })
221
+ export type StepFinishPart = z.infer<typeof StepFinishPart>
222
+
223
+ export const ToolStatePending = z
224
+ .object({
225
+ status: z.literal("pending"),
226
+ input: z.record(z.string(), z.any()),
227
+ raw: z.string(),
228
+ })
229
+ .meta({
230
+ ref: "ToolStatePending",
231
+ })
232
+
233
+ export type ToolStatePending = z.infer<typeof ToolStatePending>
234
+
235
+ export const ToolStateRunning = z
236
+ .object({
237
+ status: z.literal("running"),
238
+ input: z.record(z.string(), z.any()),
239
+ title: z.string().optional(),
240
+ metadata: z.record(z.string(), z.any()).optional(),
241
+ time: z.object({
242
+ start: z.number(),
243
+ }),
244
+ })
245
+ .meta({
246
+ ref: "ToolStateRunning",
247
+ })
248
+ export type ToolStateRunning = z.infer<typeof ToolStateRunning>
249
+
250
+ export const ToolStateCompleted = z
251
+ .object({
252
+ status: z.literal("completed"),
253
+ input: z.record(z.string(), z.any()),
254
+ output: z.string(),
255
+ title: z.string(),
256
+ metadata: z.record(z.string(), z.any()),
257
+ time: z.object({
258
+ start: z.number(),
259
+ end: z.number(),
260
+ compacted: z.number().optional(),
261
+ }),
262
+ attachments: FilePart.array().optional(),
263
+ })
264
+ .meta({
265
+ ref: "ToolStateCompleted",
266
+ })
267
+ export type ToolStateCompleted = z.infer<typeof ToolStateCompleted>
268
+
269
+ export const ToolStateError = z
270
+ .object({
271
+ status: z.literal("error"),
272
+ input: z.record(z.string(), z.any()),
273
+ error: z.string(),
274
+ metadata: z.record(z.string(), z.any()).optional(),
275
+ time: z.object({
276
+ start: z.number(),
277
+ end: z.number(),
278
+ }),
279
+ })
280
+ .meta({
281
+ ref: "ToolStateError",
282
+ })
283
+ export type ToolStateError = z.infer<typeof ToolStateError>
284
+
285
+ export const ToolState = z
286
+ .discriminatedUnion("status", [ToolStatePending, ToolStateRunning, ToolStateCompleted, ToolStateError])
287
+ .meta({
288
+ ref: "ToolState",
289
+ })
290
+
291
+ export const ToolPart = PartBase.extend({
292
+ type: z.literal("tool"),
293
+ callID: z.string(),
294
+ tool: z.string(),
295
+ state: ToolState,
296
+ metadata: z.record(z.string(), z.any()).optional(),
297
+ }).meta({
298
+ ref: "ToolPart",
299
+ })
300
+ export type ToolPart = z.infer<typeof ToolPart>
301
+
302
+ const Base = z.object({
303
+ id: z.string(),
304
+ sessionID: z.string(),
305
+ })
306
+
307
+ export const User = Base.extend({
308
+ role: z.literal("user"),
309
+ time: z.object({
310
+ created: z.number(),
311
+ }),
312
+ summary: z
313
+ .object({
314
+ title: z.string().optional(),
315
+ body: z.string().optional(),
316
+ diffs: Snapshot.FileDiff.array(),
317
+ })
318
+ .optional(),
319
+ agent: z.string(),
320
+ model: z.object({
321
+ providerID: z.string(),
322
+ modelID: z.string(),
323
+ }),
324
+ system: z.string().optional(),
325
+ tools: z.record(z.string(), z.boolean()).optional(),
326
+ variant: z.string().optional(),
327
+ }).meta({
328
+ ref: "UserMessage",
329
+ })
330
+ export type User = z.infer<typeof User>
331
+
332
+ export const Part = z
333
+ .discriminatedUnion("type", [
334
+ TextPart,
335
+ SubtaskPart,
336
+ ReasoningPart,
337
+ FilePart,
338
+ ToolPart,
339
+ StepStartPart,
340
+ StepFinishPart,
341
+ SnapshotPart,
342
+ PatchPart,
343
+ AgentPart,
344
+ RetryPart,
345
+ CompactionPart,
346
+ ])
347
+ .meta({
348
+ ref: "Part",
349
+ })
350
+ export type Part = z.infer<typeof Part>
351
+
352
+ export const Assistant = Base.extend({
353
+ role: z.literal("assistant"),
354
+ time: z.object({
355
+ created: z.number(),
356
+ completed: z.number().optional(),
357
+ }),
358
+ error: z
359
+ .discriminatedUnion("name", [
360
+ AuthError.Schema,
361
+ NamedError.Unknown.Schema,
362
+ OutputLengthError.Schema,
363
+ AbortedError.Schema,
364
+ APIError.Schema,
365
+ ])
366
+ .optional(),
367
+ parentID: z.string(),
368
+ modelID: z.string(),
369
+ providerID: z.string(),
370
+ /**
371
+ * @deprecated
372
+ */
373
+ mode: z.string(),
374
+ agent: z.string(),
375
+ path: z.object({
376
+ cwd: z.string(),
377
+ root: z.string(),
378
+ }),
379
+ summary: z.boolean().optional(),
380
+ cost: z.number(),
381
+ tokens: z.object({
382
+ input: z.number(),
383
+ output: z.number(),
384
+ reasoning: z.number(),
385
+ cache: z.object({
386
+ read: z.number(),
387
+ write: z.number(),
388
+ }),
389
+ }),
390
+ variant: z.string().optional(),
391
+ finish: z.string().optional(),
392
+ }).meta({
393
+ ref: "AssistantMessage",
394
+ })
395
+ export type Assistant = z.infer<typeof Assistant>
396
+
397
+ export const Info = z.discriminatedUnion("role", [User, Assistant]).meta({
398
+ ref: "Message",
399
+ })
400
+ export type Info = z.infer<typeof Info>
401
+
402
+ export const Event = {
403
+ Updated: BusEvent.define(
404
+ "message.updated",
405
+ z.object({
406
+ info: Info,
407
+ }),
408
+ ),
409
+ Removed: BusEvent.define(
410
+ "message.removed",
411
+ z.object({
412
+ sessionID: z.string(),
413
+ messageID: z.string(),
414
+ }),
415
+ ),
416
+ PartUpdated: BusEvent.define(
417
+ "message.part.updated",
418
+ z.object({
419
+ part: Part,
420
+ delta: z.string().optional(),
421
+ }),
422
+ ),
423
+ PartRemoved: BusEvent.define(
424
+ "message.part.removed",
425
+ z.object({
426
+ sessionID: z.string(),
427
+ messageID: z.string(),
428
+ partID: z.string(),
429
+ }),
430
+ ),
431
+ }
432
+
433
+ export const WithParts = z.object({
434
+ info: Info,
435
+ parts: z.array(Part),
436
+ })
437
+ export type WithParts = z.infer<typeof WithParts>
438
+
439
+ export function toModelMessages(input: WithParts[], model: Provider.Model): ModelMessage[] {
440
+ const result: UIMessage[] = []
441
+ const toolNames = new Set<string>()
442
+ // Track media from tool results that need to be injected as user messages
443
+ // for providers that don't support media in tool results.
444
+ //
445
+ // OpenAI-compatible APIs only support string content in tool results, so we need
446
+ // to extract media and inject as user messages. Other SDKs (anthropic, google,
447
+ // bedrock) handle type: "content" with media parts natively.
448
+ //
449
+ // Only apply this workaround if the model actually supports image input -
450
+ // otherwise there's no point extracting images.
451
+ const supportsMediaInToolResults = (() => {
452
+ if (model.api.npm === "@ai-sdk/anthropic") return true
453
+ if (model.api.npm === "@ai-sdk/openai") return true
454
+ if (model.api.npm === "@ai-sdk/amazon-bedrock") return true
455
+ if (model.api.npm === "@ai-sdk/google-vertex/anthropic") return true
456
+ if (model.api.npm === "@ai-sdk/google") {
457
+ const id = model.api.id.toLowerCase()
458
+ return id.includes("gemini-3") && !id.includes("gemini-2")
459
+ }
460
+ return false
461
+ })()
462
+
463
+ const toModelOutput = (output: unknown) => {
464
+ if (typeof output === "string") {
465
+ return { type: "text", value: output }
466
+ }
467
+
468
+ if (typeof output === "object") {
469
+ const outputObject = output as {
470
+ text: string
471
+ attachments?: Array<{ mime: string; url: string }>
472
+ }
473
+ const attachments = (outputObject.attachments ?? []).filter((attachment) => {
474
+ return attachment.url.startsWith("data:") && attachment.url.includes(",")
475
+ })
476
+
477
+ return {
478
+ type: "content",
479
+ value: [
480
+ { type: "text", text: outputObject.text },
481
+ ...attachments.map((attachment) => ({
482
+ type: "media",
483
+ mediaType: attachment.mime,
484
+ data: iife(() => {
485
+ const commaIndex = attachment.url.indexOf(",")
486
+ return commaIndex === -1 ? attachment.url : attachment.url.slice(commaIndex + 1)
487
+ }),
488
+ })),
489
+ ],
490
+ }
491
+ }
492
+
493
+ return { type: "json", value: output as never }
494
+ }
495
+
496
+ for (const msg of input) {
497
+ if (msg.parts.length === 0) continue
498
+
499
+ if (msg.info.role === "user") {
500
+ const userMessage: UIMessage = {
501
+ id: msg.info.id,
502
+ role: "user",
503
+ parts: [],
504
+ }
505
+ result.push(userMessage)
506
+ for (const part of msg.parts) {
507
+ if (part.type === "text" && !part.ignored)
508
+ userMessage.parts.push({
509
+ type: "text",
510
+ text: part.text,
511
+ })
512
+ // text/plain and directory files are converted into text parts, ignore them
513
+ if (part.type === "file" && part.mime !== "text/plain" && part.mime !== "application/x-directory")
514
+ userMessage.parts.push({
515
+ type: "file",
516
+ url: part.url,
517
+ mediaType: part.mime,
518
+ filename: part.filename,
519
+ })
520
+
521
+ if (part.type === "compaction") {
522
+ userMessage.parts.push({
523
+ type: "text",
524
+ text: "What did we do so far?",
525
+ })
526
+ }
527
+ if (part.type === "subtask") {
528
+ userMessage.parts.push({
529
+ type: "text",
530
+ text: "The following tool was executed by the user",
531
+ })
532
+ }
533
+ }
534
+ }
535
+
536
+ if (msg.info.role === "assistant") {
537
+ const differentModel = `${model.providerID}/${model.id}` !== `${msg.info.providerID}/${msg.info.modelID}`
538
+ const media: Array<{ mime: string; url: string }> = []
539
+
540
+ if (
541
+ msg.info.error &&
542
+ !(
543
+ MessageV2.AbortedError.isInstance(msg.info.error) &&
544
+ msg.parts.some((part) => part.type !== "step-start" && part.type !== "reasoning")
545
+ )
546
+ ) {
547
+ continue
548
+ }
549
+ const assistantMessage: UIMessage = {
550
+ id: msg.info.id,
551
+ role: "assistant",
552
+ parts: [],
553
+ }
554
+ for (const part of msg.parts) {
555
+ if (part.type === "text")
556
+ assistantMessage.parts.push({
557
+ type: "text",
558
+ text: part.text,
559
+ ...(differentModel ? {} : { providerMetadata: part.metadata }),
560
+ })
561
+ if (part.type === "step-start")
562
+ assistantMessage.parts.push({
563
+ type: "step-start",
564
+ })
565
+ if (part.type === "tool") {
566
+ toolNames.add(part.tool)
567
+ if (part.state.status === "completed") {
568
+ const outputText = part.state.time.compacted ? "[Old tool result content cleared]" : part.state.output
569
+ const attachments = part.state.time.compacted ? [] : (part.state.attachments ?? [])
570
+
571
+ // For providers that don't support media in tool results, extract media files
572
+ // (images, PDFs) to be sent as a separate user message
573
+ const isMediaAttachment = (a: { mime: string }) =>
574
+ a.mime.startsWith("image/") || a.mime === "application/pdf"
575
+ const mediaAttachments = attachments.filter(isMediaAttachment)
576
+ const nonMediaAttachments = attachments.filter((a) => !isMediaAttachment(a))
577
+ if (!supportsMediaInToolResults && mediaAttachments.length > 0) {
578
+ media.push(...mediaAttachments)
579
+ }
580
+ const finalAttachments = supportsMediaInToolResults ? attachments : nonMediaAttachments
581
+
582
+ const output =
583
+ finalAttachments.length > 0
584
+ ? {
585
+ text: outputText,
586
+ attachments: finalAttachments,
587
+ }
588
+ : outputText
589
+
590
+ assistantMessage.parts.push({
591
+ type: ("tool-" + part.tool) as `tool-${string}`,
592
+ state: "output-available",
593
+ toolCallId: part.callID,
594
+ input: part.state.input,
595
+ output,
596
+ ...(differentModel ? {} : { callProviderMetadata: part.metadata }),
597
+ })
598
+ }
599
+ if (part.state.status === "error")
600
+ assistantMessage.parts.push({
601
+ type: ("tool-" + part.tool) as `tool-${string}`,
602
+ state: "output-error",
603
+ toolCallId: part.callID,
604
+ input: part.state.input,
605
+ errorText: part.state.error,
606
+ ...(differentModel ? {} : { callProviderMetadata: part.metadata }),
607
+ })
608
+ // Handle pending/running tool calls to prevent dangling tool_use blocks
609
+ // Anthropic/Claude APIs require every tool_use to have a corresponding tool_result
610
+ if (part.state.status === "pending" || part.state.status === "running")
611
+ assistantMessage.parts.push({
612
+ type: ("tool-" + part.tool) as `tool-${string}`,
613
+ state: "output-error",
614
+ toolCallId: part.callID,
615
+ input: part.state.input,
616
+ errorText: "[Tool execution was interrupted]",
617
+ ...(differentModel ? {} : { callProviderMetadata: part.metadata }),
618
+ })
619
+ }
620
+ if (part.type === "reasoning") {
621
+ assistantMessage.parts.push({
622
+ type: "reasoning",
623
+ text: part.text,
624
+ ...(differentModel ? {} : { providerMetadata: part.metadata }),
625
+ })
626
+ }
627
+ }
628
+ if (assistantMessage.parts.length > 0) {
629
+ result.push(assistantMessage)
630
+ // Inject pending media as a user message for providers that don't support
631
+ // media (images, PDFs) in tool results
632
+ if (media.length > 0) {
633
+ result.push({
634
+ id: Identifier.ascending("message"),
635
+ role: "user",
636
+ parts: [
637
+ {
638
+ type: "text" as const,
639
+ text: "Attached image(s) from tool result:",
640
+ },
641
+ ...media.map((attachment) => ({
642
+ type: "file" as const,
643
+ url: attachment.url,
644
+ mediaType: attachment.mime,
645
+ })),
646
+ ],
647
+ })
648
+ }
649
+ }
650
+ }
651
+ }
652
+
653
+ const tools = Object.fromEntries(Array.from(toolNames).map((toolName) => [toolName, { toModelOutput }]))
654
+
655
+ return convertToModelMessages(
656
+ result.filter((msg) => msg.parts.some((part) => part.type !== "step-start")),
657
+ {
658
+ //@ts-expect-error (convertToModelMessages expects a ToolSet but only actually needs tools[name]?.toModelOutput)
659
+ tools,
660
+ },
661
+ )
662
+ }
663
+
664
+ export const stream = fn(Identifier.schema("session"), async function* (sessionID) {
665
+ const list = await Array.fromAsync(await Storage.list(["message", sessionID]))
666
+ for (let i = list.length - 1; i >= 0; i--) {
667
+ yield await get({
668
+ sessionID,
669
+ messageID: list[i][2],
670
+ })
671
+ }
672
+ })
673
+
674
+ export const parts = fn(Identifier.schema("message"), async (messageID) => {
675
+ const result = [] as MessageV2.Part[]
676
+ for (const item of await Storage.list(["part", messageID])) {
677
+ const read = await Storage.read<MessageV2.Part>(item)
678
+ result.push(read)
679
+ }
680
+ result.sort((a, b) => (a.id > b.id ? 1 : -1))
681
+ return result
682
+ })
683
+
684
+ export const get = fn(
685
+ z.object({
686
+ sessionID: Identifier.schema("session"),
687
+ messageID: Identifier.schema("message"),
688
+ }),
689
+ async (input): Promise<WithParts> => {
690
+ return {
691
+ info: await Storage.read<MessageV2.Info>(["message", input.sessionID, input.messageID]),
692
+ parts: await parts(input.messageID),
693
+ }
694
+ },
695
+ )
696
+
697
+ export async function filterCompacted(stream: AsyncIterable<MessageV2.WithParts>) {
698
+ const result = [] as MessageV2.WithParts[]
699
+ const completed = new Set<string>()
700
+ for await (const msg of stream) {
701
+ result.push(msg)
702
+ if (
703
+ msg.info.role === "user" &&
704
+ completed.has(msg.info.id) &&
705
+ msg.parts.some((part) => part.type === "compaction")
706
+ )
707
+ break
708
+ if (msg.info.role === "assistant" && msg.info.summary && msg.info.finish) completed.add(msg.info.parentID)
709
+ }
710
+ result.reverse()
711
+ return result
712
+ }
713
+
714
+ const isOpenAiErrorRetryable = (e: APICallError) => {
715
+ const status = e.statusCode
716
+ if (!status) return e.isRetryable
717
+ // openai sometimes returns 404 for models that are actually available
718
+ return status === 404 || e.isRetryable
719
+ }
720
+
721
+ export function fromError(e: unknown, ctx: { providerID: string }) {
722
+ switch (true) {
723
+ case e instanceof DOMException && e.name === "AbortError":
724
+ return new MessageV2.AbortedError(
725
+ { message: e.message },
726
+ {
727
+ cause: e,
728
+ },
729
+ ).toObject()
730
+ case MessageV2.OutputLengthError.isInstance(e):
731
+ return e
732
+ case LoadAPIKeyError.isInstance(e):
733
+ return new MessageV2.AuthError(
734
+ {
735
+ providerID: ctx.providerID,
736
+ message: e.message,
737
+ },
738
+ { cause: e },
739
+ ).toObject()
740
+ case (e as SystemError)?.code === "ECONNRESET":
741
+ return new MessageV2.APIError(
742
+ {
743
+ message: "Connection reset by server",
744
+ isRetryable: true,
745
+ metadata: {
746
+ code: (e as SystemError).code ?? "",
747
+ syscall: (e as SystemError).syscall ?? "",
748
+ message: (e as SystemError).message ?? "",
749
+ },
750
+ },
751
+ { cause: e },
752
+ ).toObject()
753
+ case APICallError.isInstance(e):
754
+ const message = iife(() => {
755
+ let msg = e.message
756
+ if (msg === "") {
757
+ if (e.responseBody) return e.responseBody
758
+ if (e.statusCode) {
759
+ const err = STATUS_CODES[e.statusCode]
760
+ if (err) return err
761
+ }
762
+ return "Unknown error"
763
+ }
764
+ const transformed = ProviderTransform.error(ctx.providerID, e)
765
+ if (transformed !== msg) {
766
+ return transformed
767
+ }
768
+ if (!e.responseBody || (e.statusCode && msg !== STATUS_CODES[e.statusCode])) {
769
+ return msg
770
+ }
771
+
772
+ try {
773
+ const body = JSON.parse(e.responseBody)
774
+ // try to extract common error message fields
775
+ const errMsg = body.message || body.error || body.error?.message
776
+ if (errMsg && typeof errMsg === "string") {
777
+ return `${msg}: ${errMsg}`
778
+ }
779
+ } catch {}
780
+
781
+ return `${msg}: ${e.responseBody}`
782
+ }).trim()
783
+
784
+ const metadata = e.url ? { url: e.url } : undefined
785
+ return new MessageV2.APIError(
786
+ {
787
+ message,
788
+ statusCode: e.statusCode,
789
+ isRetryable: ctx.providerID.startsWith("openai") ? isOpenAiErrorRetryable(e) : e.isRetryable,
790
+ responseHeaders: e.responseHeaders,
791
+ responseBody: e.responseBody,
792
+ metadata,
793
+ },
794
+ { cause: e },
795
+ ).toObject()
796
+ case e instanceof Error:
797
+ return new NamedError.Unknown({ message: e.toString() }, { cause: e }).toObject()
798
+ default:
799
+ return new NamedError.Unknown({ message: JSON.stringify(e) }, { cause: e })
800
+ }
801
+ }
802
+ }