innocode 1.0.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 (434) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/innocode +84 -0
  5. package/bin/opencode +84 -0
  6. package/bunfig.toml +5 -0
  7. package/package.json +126 -0
  8. package/parsers-config.ts +253 -0
  9. package/script/build.ts +198 -0
  10. package/script/postinstall.mjs +125 -0
  11. package/script/publish.ts +186 -0
  12. package/script/schema.ts +47 -0
  13. package/script/seed-e2e.ts +50 -0
  14. package/src/acp/README.md +164 -0
  15. package/src/acp/agent.ts +1676 -0
  16. package/src/acp/session.ts +117 -0
  17. package/src/acp/types.ts +23 -0
  18. package/src/agent/agent.ts +338 -0
  19. package/src/agent/generate.txt +75 -0
  20. package/src/agent/prompt/compaction.txt +14 -0
  21. package/src/agent/prompt/explore.txt +18 -0
  22. package/src/agent/prompt/summary.txt +11 -0
  23. package/src/agent/prompt/title.txt +44 -0
  24. package/src/auth/index.ts +70 -0
  25. package/src/bun/index.ts +137 -0
  26. package/src/bun/registry.ts +48 -0
  27. package/src/bus/bus-event.ts +43 -0
  28. package/src/bus/global.ts +10 -0
  29. package/src/bus/index.ts +105 -0
  30. package/src/cli/bootstrap.ts +17 -0
  31. package/src/cli/cmd/acp.ts +70 -0
  32. package/src/cli/cmd/agent.ts +257 -0
  33. package/src/cli/cmd/auth.ts +400 -0
  34. package/src/cli/cmd/cmd.ts +7 -0
  35. package/src/cli/cmd/debug/agent.ts +167 -0
  36. package/src/cli/cmd/debug/config.ts +16 -0
  37. package/src/cli/cmd/debug/file.ts +97 -0
  38. package/src/cli/cmd/debug/index.ts +48 -0
  39. package/src/cli/cmd/debug/lsp.ts +52 -0
  40. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  41. package/src/cli/cmd/debug/scrap.ts +16 -0
  42. package/src/cli/cmd/debug/skill.ts +16 -0
  43. package/src/cli/cmd/debug/snapshot.ts +52 -0
  44. package/src/cli/cmd/export.ts +88 -0
  45. package/src/cli/cmd/generate.ts +38 -0
  46. package/src/cli/cmd/github.ts +1540 -0
  47. package/src/cli/cmd/import.ts +147 -0
  48. package/src/cli/cmd/mcp.ts +765 -0
  49. package/src/cli/cmd/models.ts +77 -0
  50. package/src/cli/cmd/pr.ts +113 -0
  51. package/src/cli/cmd/run.ts +598 -0
  52. package/src/cli/cmd/serve.ts +20 -0
  53. package/src/cli/cmd/session.ts +135 -0
  54. package/src/cli/cmd/stats.ts +426 -0
  55. package/src/cli/cmd/tui/app.tsx +812 -0
  56. package/src/cli/cmd/tui/attach.ts +60 -0
  57. package/src/cli/cmd/tui/component/border.tsx +21 -0
  58. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  59. package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
  60. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  61. package/src/cli/cmd/tui/component/dialog-model.tsx +165 -0
  62. package/src/cli/cmd/tui/component/dialog-provider.tsx +243 -0
  63. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  64. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  65. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  66. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  67. package/src/cli/cmd/tui/component/dialog-status.tsx +167 -0
  68. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  69. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  70. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  71. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +666 -0
  72. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  73. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  74. package/src/cli/cmd/tui/component/prompt/index.tsx +1153 -0
  75. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  76. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  77. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  78. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  79. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  80. package/src/cli/cmd/tui/context/args.tsx +15 -0
  81. package/src/cli/cmd/tui/context/directory.ts +13 -0
  82. package/src/cli/cmd/tui/context/exit.tsx +54 -0
  83. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  84. package/src/cli/cmd/tui/context/keybind.tsx +100 -0
  85. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  86. package/src/cli/cmd/tui/context/local.tsx +409 -0
  87. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  88. package/src/cli/cmd/tui/context/route.tsx +46 -0
  89. package/src/cli/cmd/tui/context/sdk.tsx +101 -0
  90. package/src/cli/cmd/tui/context/sync.tsx +470 -0
  91. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  92. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  93. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  94. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  95. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  96. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  97. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  98. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  99. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  100. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  101. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  102. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  103. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  104. package/src/cli/cmd/tui/context/theme/innocode.json +245 -0
  105. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  106. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  107. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  108. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  109. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  110. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  111. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  112. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  113. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  114. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  115. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  116. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  117. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  118. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  119. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  120. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  121. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  122. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  123. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  124. package/src/cli/cmd/tui/context/theme.tsx +1154 -0
  125. package/src/cli/cmd/tui/event.ts +48 -0
  126. package/src/cli/cmd/tui/routes/home.tsx +145 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  128. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  129. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  130. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  131. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  132. package/src/cli/cmd/tui/routes/session/header.tsx +135 -0
  133. package/src/cli/cmd/tui/routes/session/index.tsx +2139 -0
  134. package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
  135. package/src/cli/cmd/tui/routes/session/question.tsx +466 -0
  136. package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
  137. package/src/cli/cmd/tui/thread.ts +188 -0
  138. package/src/cli/cmd/tui/ui/dialog-alert.tsx +59 -0
  139. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +85 -0
  140. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +207 -0
  141. package/src/cli/cmd/tui/ui/dialog-help.tsx +40 -0
  142. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +80 -0
  143. package/src/cli/cmd/tui/ui/dialog-select.tsx +401 -0
  144. package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
  145. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  146. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  147. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  148. package/src/cli/cmd/tui/util/clipboard.ts +159 -0
  149. package/src/cli/cmd/tui/util/editor.ts +32 -0
  150. package/src/cli/cmd/tui/util/signal.ts +7 -0
  151. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  152. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  153. package/src/cli/cmd/tui/win32.ts +129 -0
  154. package/src/cli/cmd/tui/worker.ts +152 -0
  155. package/src/cli/cmd/uninstall.ts +363 -0
  156. package/src/cli/cmd/upgrade.ts +73 -0
  157. package/src/cli/cmd/web.ts +81 -0
  158. package/src/cli/error.ts +57 -0
  159. package/src/cli/logo.ts +6 -0
  160. package/src/cli/network.ts +60 -0
  161. package/src/cli/ui.ts +113 -0
  162. package/src/cli/upgrade.ts +25 -0
  163. package/src/command/index.ts +150 -0
  164. package/src/command/template/initialize.txt +10 -0
  165. package/src/command/template/review.txt +101 -0
  166. package/src/config/config.ts +1517 -0
  167. package/src/config/markdown.ts +98 -0
  168. package/src/env/index.ts +28 -0
  169. package/src/file/ignore.ts +83 -0
  170. package/src/file/index.ts +583 -0
  171. package/src/file/ripgrep.ts +375 -0
  172. package/src/file/time.ts +69 -0
  173. package/src/file/watcher.ts +127 -0
  174. package/src/flag/flag.ts +148 -0
  175. package/src/format/formatter.ts +366 -0
  176. package/src/format/index.ts +137 -0
  177. package/src/global/index.ts +80 -0
  178. package/src/id/id.ts +83 -0
  179. package/src/ide/index.ts +76 -0
  180. package/src/index.ts +160 -0
  181. package/src/installation/index.ts +268 -0
  182. package/src/lsp/client.ts +252 -0
  183. package/src/lsp/index.ts +485 -0
  184. package/src/lsp/language.ts +119 -0
  185. package/src/lsp/server.ts +2046 -0
  186. package/src/mcp/auth.ts +132 -0
  187. package/src/mcp/index.ts +937 -0
  188. package/src/mcp/oauth-callback.ts +200 -0
  189. package/src/mcp/oauth-provider.ts +154 -0
  190. package/src/patch/index.ts +680 -0
  191. package/src/permission/arity.ts +163 -0
  192. package/src/permission/index.ts +210 -0
  193. package/src/permission/next.ts +280 -0
  194. package/src/plugin/codex.ts +624 -0
  195. package/src/plugin/copilot.ts +327 -0
  196. package/src/plugin/index.ts +138 -0
  197. package/src/project/bootstrap.ts +35 -0
  198. package/src/project/instance.ts +114 -0
  199. package/src/project/project.ts +371 -0
  200. package/src/project/state.ts +70 -0
  201. package/src/project/vcs.ts +76 -0
  202. package/src/provider/auth.ts +147 -0
  203. package/src/provider/error.ts +189 -0
  204. package/src/provider/models.ts +133 -0
  205. package/src/provider/provider.ts +1370 -0
  206. package/src/provider/sdk/copilot/README.md +5 -0
  207. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +164 -0
  208. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  209. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +17 -0
  210. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  211. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +780 -0
  212. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  213. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  214. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +87 -0
  215. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  216. package/src/provider/sdk/copilot/index.ts +2 -0
  217. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  218. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +303 -0
  219. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  220. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  221. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  222. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +207 -0
  223. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1732 -0
  224. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +177 -0
  225. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  226. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +88 -0
  227. package/src/provider/sdk/copilot/responses/tool/file-search.ts +128 -0
  228. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +115 -0
  229. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +65 -0
  230. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +104 -0
  231. package/src/provider/sdk/copilot/responses/tool/web-search.ts +103 -0
  232. package/src/provider/transform.ts +806 -0
  233. package/src/pty/index.ts +286 -0
  234. package/src/question/index.ts +171 -0
  235. package/src/scheduler/index.ts +61 -0
  236. package/src/server/error.ts +36 -0
  237. package/src/server/event.ts +7 -0
  238. package/src/server/mdns.ts +60 -0
  239. package/src/server/routes/config.ts +92 -0
  240. package/src/server/routes/experimental.ts +208 -0
  241. package/src/server/routes/file.ts +197 -0
  242. package/src/server/routes/global.ts +183 -0
  243. package/src/server/routes/mcp.ts +225 -0
  244. package/src/server/routes/permission.ts +68 -0
  245. package/src/server/routes/project.ts +82 -0
  246. package/src/server/routes/provider.ts +179 -0
  247. package/src/server/routes/pty.ts +176 -0
  248. package/src/server/routes/question.ts +98 -0
  249. package/src/server/routes/session.ts +939 -0
  250. package/src/server/routes/tui.ts +379 -0
  251. package/src/server/server.ts +621 -0
  252. package/src/session/compaction.ts +261 -0
  253. package/src/session/index.ts +543 -0
  254. package/src/session/instruction.ts +197 -0
  255. package/src/session/llm.ts +283 -0
  256. package/src/session/message-v2.ts +841 -0
  257. package/src/session/message.ts +189 -0
  258. package/src/session/processor.ts +410 -0
  259. package/src/session/prompt/anthropic-20250930.txt +166 -0
  260. package/src/session/prompt/anthropic.txt +105 -0
  261. package/src/session/prompt/beast.txt +147 -0
  262. package/src/session/prompt/build-switch.txt +5 -0
  263. package/src/session/prompt/codex_header.txt +79 -0
  264. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  265. package/src/session/prompt/gemini.txt +155 -0
  266. package/src/session/prompt/max-steps.txt +16 -0
  267. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  268. package/src/session/prompt/plan.txt +26 -0
  269. package/src/session/prompt/qwen.txt +109 -0
  270. package/src/session/prompt/trinity.txt +97 -0
  271. package/src/session/prompt.ts +1964 -0
  272. package/src/session/retry.ts +101 -0
  273. package/src/session/revert.ts +121 -0
  274. package/src/session/status.ts +76 -0
  275. package/src/session/summary.ts +203 -0
  276. package/src/session/system.ts +54 -0
  277. package/src/session/todo.ts +37 -0
  278. package/src/share/share-next.ts +200 -0
  279. package/src/share/share.ts +92 -0
  280. package/src/shell/shell.ts +67 -0
  281. package/src/skill/discovery.ts +97 -0
  282. package/src/skill/index.ts +1 -0
  283. package/src/skill/skill.ts +188 -0
  284. package/src/snapshot/index.ts +255 -0
  285. package/src/storage/storage.ts +227 -0
  286. package/src/tool/apply_patch.ts +281 -0
  287. package/src/tool/apply_patch.txt +33 -0
  288. package/src/tool/bash.ts +269 -0
  289. package/src/tool/bash.txt +115 -0
  290. package/src/tool/batch.ts +175 -0
  291. package/src/tool/batch.txt +24 -0
  292. package/src/tool/codesearch.ts +132 -0
  293. package/src/tool/codesearch.txt +12 -0
  294. package/src/tool/edit.ts +655 -0
  295. package/src/tool/edit.txt +10 -0
  296. package/src/tool/external-directory.ts +32 -0
  297. package/src/tool/glob.ts +80 -0
  298. package/src/tool/glob.txt +6 -0
  299. package/src/tool/grep.ts +150 -0
  300. package/src/tool/grep.txt +8 -0
  301. package/src/tool/invalid.ts +17 -0
  302. package/src/tool/ls.ts +121 -0
  303. package/src/tool/ls.txt +1 -0
  304. package/src/tool/lsp.ts +96 -0
  305. package/src/tool/lsp.txt +19 -0
  306. package/src/tool/multiedit.ts +46 -0
  307. package/src/tool/multiedit.txt +41 -0
  308. package/src/tool/plan-enter.txt +14 -0
  309. package/src/tool/plan-exit.txt +13 -0
  310. package/src/tool/plan.ts +130 -0
  311. package/src/tool/question.ts +33 -0
  312. package/src/tool/question.txt +10 -0
  313. package/src/tool/read.ts +261 -0
  314. package/src/tool/read.txt +14 -0
  315. package/src/tool/registry.ts +160 -0
  316. package/src/tool/skill.ts +123 -0
  317. package/src/tool/task.ts +165 -0
  318. package/src/tool/task.txt +60 -0
  319. package/src/tool/todo.ts +53 -0
  320. package/src/tool/todoread.txt +14 -0
  321. package/src/tool/todowrite.txt +167 -0
  322. package/src/tool/tool.ts +89 -0
  323. package/src/tool/truncation.ts +106 -0
  324. package/src/tool/webfetch.ts +186 -0
  325. package/src/tool/webfetch.txt +13 -0
  326. package/src/tool/websearch.ts +150 -0
  327. package/src/tool/websearch.txt +14 -0
  328. package/src/tool/write.ts +85 -0
  329. package/src/tool/write.txt +8 -0
  330. package/src/util/abort.ts +35 -0
  331. package/src/util/archive.ts +16 -0
  332. package/src/util/color.ts +19 -0
  333. package/src/util/context.ts +25 -0
  334. package/src/util/defer.ts +12 -0
  335. package/src/util/eventloop.ts +20 -0
  336. package/src/util/filesystem.ts +93 -0
  337. package/src/util/fn.ts +11 -0
  338. package/src/util/format.ts +20 -0
  339. package/src/util/iife.ts +3 -0
  340. package/src/util/keybind.ts +103 -0
  341. package/src/util/lazy.ts +18 -0
  342. package/src/util/locale.ts +81 -0
  343. package/src/util/lock.ts +98 -0
  344. package/src/util/log.ts +180 -0
  345. package/src/util/proxied.ts +3 -0
  346. package/src/util/queue.ts +32 -0
  347. package/src/util/rpc.ts +66 -0
  348. package/src/util/scrap.ts +10 -0
  349. package/src/util/signal.ts +12 -0
  350. package/src/util/timeout.ts +14 -0
  351. package/src/util/token.ts +7 -0
  352. package/src/util/wildcard.ts +56 -0
  353. package/src/worktree/index.ts +612 -0
  354. package/sst-env.d.ts +9 -0
  355. package/test/acp/agent-interface.test.ts +51 -0
  356. package/test/acp/event-subscription.test.ts +436 -0
  357. package/test/agent/agent.test.ts +675 -0
  358. package/test/bun.test.ts +53 -0
  359. package/test/cli/github-action.test.ts +161 -0
  360. package/test/cli/github-remote.test.ts +80 -0
  361. package/test/cli/import.test.ts +38 -0
  362. package/test/cli/tui/transcript.test.ts +322 -0
  363. package/test/config/agent-color.test.ts +71 -0
  364. package/test/config/config.test.ts +1802 -0
  365. package/test/config/fixtures/empty-frontmatter.md +4 -0
  366. package/test/config/fixtures/frontmatter.md +28 -0
  367. package/test/config/fixtures/markdown-header.md +11 -0
  368. package/test/config/fixtures/no-frontmatter.md +1 -0
  369. package/test/config/fixtures/weird-model-id.md +13 -0
  370. package/test/config/markdown.test.ts +228 -0
  371. package/test/file/ignore.test.ts +10 -0
  372. package/test/file/path-traversal.test.ts +198 -0
  373. package/test/file/ripgrep.test.ts +39 -0
  374. package/test/fixture/fixture.ts +45 -0
  375. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  376. package/test/ide/ide.test.ts +82 -0
  377. package/test/keybind.test.ts +421 -0
  378. package/test/lsp/client.test.ts +95 -0
  379. package/test/mcp/headers.test.ts +153 -0
  380. package/test/mcp/oauth-browser.test.ts +249 -0
  381. package/test/memory/abort-leak.test.ts +136 -0
  382. package/test/patch/patch.test.ts +348 -0
  383. package/test/permission/arity.test.ts +33 -0
  384. package/test/permission/next.test.ts +690 -0
  385. package/test/permission-task.test.ts +319 -0
  386. package/test/plugin/auth-override.test.ts +44 -0
  387. package/test/plugin/codex.test.ts +123 -0
  388. package/test/preload.ts +63 -0
  389. package/test/project/project.test.ts +120 -0
  390. package/test/provider/amazon-bedrock.test.ts +445 -0
  391. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  392. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  393. package/test/provider/gitlab-duo.test.ts +262 -0
  394. package/test/provider/provider.test.ts +2129 -0
  395. package/test/provider/transform.test.ts +1928 -0
  396. package/test/question/question.test.ts +300 -0
  397. package/test/scheduler.test.ts +73 -0
  398. package/test/server/session-list.test.ts +39 -0
  399. package/test/server/session-select.test.ts +78 -0
  400. package/test/session/compaction.test.ts +423 -0
  401. package/test/session/instruction.test.ts +170 -0
  402. package/test/session/llm.test.ts +667 -0
  403. package/test/session/message-v2.test.ts +924 -0
  404. package/test/session/prompt-missing-file.test.ts +53 -0
  405. package/test/session/prompt-special-chars.test.ts +56 -0
  406. package/test/session/prompt-variant.test.ts +68 -0
  407. package/test/session/retry.test.ts +188 -0
  408. package/test/session/revert-compact.test.ts +285 -0
  409. package/test/session/session.test.ts +71 -0
  410. package/test/session/structured-output-integration.test.ts +233 -0
  411. package/test/session/structured-output.test.ts +385 -0
  412. package/test/skill/discovery.test.ts +60 -0
  413. package/test/skill/skill.test.ts +388 -0
  414. package/test/snapshot/snapshot.test.ts +1040 -0
  415. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  416. package/test/tool/apply_patch.test.ts +559 -0
  417. package/test/tool/bash.test.ts +399 -0
  418. package/test/tool/external-directory.test.ts +127 -0
  419. package/test/tool/fixtures/large-image.png +0 -0
  420. package/test/tool/fixtures/models-api.json +38413 -0
  421. package/test/tool/grep.test.ts +110 -0
  422. package/test/tool/question.test.ts +107 -0
  423. package/test/tool/read.test.ts +421 -0
  424. package/test/tool/registry.test.ts +122 -0
  425. package/test/tool/skill.test.ts +112 -0
  426. package/test/tool/truncation.test.ts +159 -0
  427. package/test/util/filesystem.test.ts +39 -0
  428. package/test/util/format.test.ts +59 -0
  429. package/test/util/iife.test.ts +36 -0
  430. package/test/util/lazy.test.ts +50 -0
  431. package/test/util/lock.test.ts +72 -0
  432. package/test/util/timeout.test.ts +21 -0
  433. package/test/util/wildcard.test.ts +75 -0
  434. package/tsconfig.json +16 -0
@@ -0,0 +1,841 @@
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 { ProviderError } from "@/provider/error"
11
+ import { iife } from "@/util/iife"
12
+ import { type SystemError } from "bun"
13
+ import type { Provider } from "@/provider/provider"
14
+
15
+ export namespace MessageV2 {
16
+ export const OutputLengthError = NamedError.create("MessageOutputLengthError", z.object({}))
17
+ export const AbortedError = NamedError.create("MessageAbortedError", z.object({ message: z.string() }))
18
+ export const StructuredOutputError = NamedError.create(
19
+ "StructuredOutputError",
20
+ z.object({
21
+ message: z.string(),
22
+ retries: z.number(),
23
+ }),
24
+ )
25
+ export const AuthError = NamedError.create(
26
+ "ProviderAuthError",
27
+ z.object({
28
+ providerID: z.string(),
29
+ message: z.string(),
30
+ }),
31
+ )
32
+ export const APIError = NamedError.create(
33
+ "APIError",
34
+ z.object({
35
+ message: z.string(),
36
+ statusCode: z.number().optional(),
37
+ isRetryable: z.boolean(),
38
+ responseHeaders: z.record(z.string(), z.string()).optional(),
39
+ responseBody: z.string().optional(),
40
+ metadata: z.record(z.string(), z.string()).optional(),
41
+ }),
42
+ )
43
+ export type APIError = z.infer<typeof APIError.Schema>
44
+ export const ContextOverflowError = NamedError.create(
45
+ "ContextOverflowError",
46
+ z.object({ message: z.string(), responseBody: z.string().optional() }),
47
+ )
48
+
49
+ export const OutputFormatText = z
50
+ .object({
51
+ type: z.literal("text"),
52
+ })
53
+ .meta({
54
+ ref: "OutputFormatText",
55
+ })
56
+
57
+ export const OutputFormatJsonSchema = z
58
+ .object({
59
+ type: z.literal("json_schema"),
60
+ schema: z.record(z.string(), z.any()).meta({ ref: "JSONSchema" }),
61
+ retryCount: z.number().int().min(0).default(2),
62
+ })
63
+ .meta({
64
+ ref: "OutputFormatJsonSchema",
65
+ })
66
+
67
+ export const Format = z.discriminatedUnion("type", [OutputFormatText, OutputFormatJsonSchema]).meta({
68
+ ref: "OutputFormat",
69
+ })
70
+ export type OutputFormat = z.infer<typeof Format>
71
+
72
+ const PartBase = z.object({
73
+ id: z.string(),
74
+ sessionID: z.string(),
75
+ messageID: z.string(),
76
+ })
77
+
78
+ export const SnapshotPart = PartBase.extend({
79
+ type: z.literal("snapshot"),
80
+ snapshot: z.string(),
81
+ }).meta({
82
+ ref: "SnapshotPart",
83
+ })
84
+ export type SnapshotPart = z.infer<typeof SnapshotPart>
85
+
86
+ export const PatchPart = PartBase.extend({
87
+ type: z.literal("patch"),
88
+ hash: z.string(),
89
+ files: z.string().array(),
90
+ }).meta({
91
+ ref: "PatchPart",
92
+ })
93
+ export type PatchPart = z.infer<typeof PatchPart>
94
+
95
+ export const TextPart = PartBase.extend({
96
+ type: z.literal("text"),
97
+ text: z.string(),
98
+ synthetic: z.boolean().optional(),
99
+ ignored: z.boolean().optional(),
100
+ time: z
101
+ .object({
102
+ start: z.number(),
103
+ end: z.number().optional(),
104
+ })
105
+ .optional(),
106
+ metadata: z.record(z.string(), z.any()).optional(),
107
+ }).meta({
108
+ ref: "TextPart",
109
+ })
110
+ export type TextPart = z.infer<typeof TextPart>
111
+
112
+ export const ReasoningPart = PartBase.extend({
113
+ type: z.literal("reasoning"),
114
+ text: z.string(),
115
+ metadata: z.record(z.string(), z.any()).optional(),
116
+ time: z.object({
117
+ start: z.number(),
118
+ end: z.number().optional(),
119
+ }),
120
+ }).meta({
121
+ ref: "ReasoningPart",
122
+ })
123
+ export type ReasoningPart = z.infer<typeof ReasoningPart>
124
+
125
+ const FilePartSourceBase = z.object({
126
+ text: z
127
+ .object({
128
+ value: z.string(),
129
+ start: z.number().int(),
130
+ end: z.number().int(),
131
+ })
132
+ .meta({
133
+ ref: "FilePartSourceText",
134
+ }),
135
+ })
136
+
137
+ export const FileSource = FilePartSourceBase.extend({
138
+ type: z.literal("file"),
139
+ path: z.string(),
140
+ }).meta({
141
+ ref: "FileSource",
142
+ })
143
+
144
+ export const SymbolSource = FilePartSourceBase.extend({
145
+ type: z.literal("symbol"),
146
+ path: z.string(),
147
+ range: LSP.Range,
148
+ name: z.string(),
149
+ kind: z.number().int(),
150
+ }).meta({
151
+ ref: "SymbolSource",
152
+ })
153
+
154
+ export const ResourceSource = FilePartSourceBase.extend({
155
+ type: z.literal("resource"),
156
+ clientName: z.string(),
157
+ uri: z.string(),
158
+ }).meta({
159
+ ref: "ResourceSource",
160
+ })
161
+
162
+ export const FilePartSource = z.discriminatedUnion("type", [FileSource, SymbolSource, ResourceSource]).meta({
163
+ ref: "FilePartSource",
164
+ })
165
+
166
+ export const FilePart = PartBase.extend({
167
+ type: z.literal("file"),
168
+ mime: z.string(),
169
+ filename: z.string().optional(),
170
+ url: z.string(),
171
+ source: FilePartSource.optional(),
172
+ }).meta({
173
+ ref: "FilePart",
174
+ })
175
+ export type FilePart = z.infer<typeof FilePart>
176
+
177
+ export const AgentPart = PartBase.extend({
178
+ type: z.literal("agent"),
179
+ name: z.string(),
180
+ source: z
181
+ .object({
182
+ value: z.string(),
183
+ start: z.number().int(),
184
+ end: z.number().int(),
185
+ })
186
+ .optional(),
187
+ }).meta({
188
+ ref: "AgentPart",
189
+ })
190
+ export type AgentPart = z.infer<typeof AgentPart>
191
+
192
+ export const CompactionPart = PartBase.extend({
193
+ type: z.literal("compaction"),
194
+ auto: z.boolean(),
195
+ }).meta({
196
+ ref: "CompactionPart",
197
+ })
198
+ export type CompactionPart = z.infer<typeof CompactionPart>
199
+
200
+ export const SubtaskPart = PartBase.extend({
201
+ type: z.literal("subtask"),
202
+ prompt: z.string(),
203
+ description: z.string(),
204
+ agent: z.string(),
205
+ model: z
206
+ .object({
207
+ providerID: z.string(),
208
+ modelID: z.string(),
209
+ })
210
+ .optional(),
211
+ command: z.string().optional(),
212
+ }).meta({
213
+ ref: "SubtaskPart",
214
+ })
215
+ export type SubtaskPart = z.infer<typeof SubtaskPart>
216
+
217
+ export const RetryPart = PartBase.extend({
218
+ type: z.literal("retry"),
219
+ attempt: z.number(),
220
+ error: APIError.Schema,
221
+ time: z.object({
222
+ created: z.number(),
223
+ }),
224
+ }).meta({
225
+ ref: "RetryPart",
226
+ })
227
+ export type RetryPart = z.infer<typeof RetryPart>
228
+
229
+ export const StepStartPart = PartBase.extend({
230
+ type: z.literal("step-start"),
231
+ snapshot: z.string().optional(),
232
+ }).meta({
233
+ ref: "StepStartPart",
234
+ })
235
+ export type StepStartPart = z.infer<typeof StepStartPart>
236
+
237
+ export const StepFinishPart = PartBase.extend({
238
+ type: z.literal("step-finish"),
239
+ reason: z.string(),
240
+ snapshot: z.string().optional(),
241
+ cost: z.number(),
242
+ tokens: z.object({
243
+ total: z.number().optional(),
244
+ input: z.number(),
245
+ output: z.number(),
246
+ reasoning: z.number(),
247
+ cache: z.object({
248
+ read: z.number(),
249
+ write: z.number(),
250
+ }),
251
+ }),
252
+ }).meta({
253
+ ref: "StepFinishPart",
254
+ })
255
+ export type StepFinishPart = z.infer<typeof StepFinishPart>
256
+
257
+ export const ToolStatePending = z
258
+ .object({
259
+ status: z.literal("pending"),
260
+ input: z.record(z.string(), z.any()),
261
+ raw: z.string(),
262
+ })
263
+ .meta({
264
+ ref: "ToolStatePending",
265
+ })
266
+
267
+ export type ToolStatePending = z.infer<typeof ToolStatePending>
268
+
269
+ export const ToolStateRunning = z
270
+ .object({
271
+ status: z.literal("running"),
272
+ input: z.record(z.string(), z.any()),
273
+ title: z.string().optional(),
274
+ metadata: z.record(z.string(), z.any()).optional(),
275
+ time: z.object({
276
+ start: z.number(),
277
+ }),
278
+ })
279
+ .meta({
280
+ ref: "ToolStateRunning",
281
+ })
282
+ export type ToolStateRunning = z.infer<typeof ToolStateRunning>
283
+
284
+ export const ToolStateCompleted = z
285
+ .object({
286
+ status: z.literal("completed"),
287
+ input: z.record(z.string(), z.any()),
288
+ output: z.string(),
289
+ title: z.string(),
290
+ metadata: z.record(z.string(), z.any()),
291
+ time: z.object({
292
+ start: z.number(),
293
+ end: z.number(),
294
+ compacted: z.number().optional(),
295
+ }),
296
+ attachments: FilePart.array().optional(),
297
+ })
298
+ .meta({
299
+ ref: "ToolStateCompleted",
300
+ })
301
+ export type ToolStateCompleted = z.infer<typeof ToolStateCompleted>
302
+
303
+ export const ToolStateError = z
304
+ .object({
305
+ status: z.literal("error"),
306
+ input: z.record(z.string(), z.any()),
307
+ error: z.string(),
308
+ metadata: z.record(z.string(), z.any()).optional(),
309
+ time: z.object({
310
+ start: z.number(),
311
+ end: z.number(),
312
+ }),
313
+ })
314
+ .meta({
315
+ ref: "ToolStateError",
316
+ })
317
+ export type ToolStateError = z.infer<typeof ToolStateError>
318
+
319
+ export const ToolState = z
320
+ .discriminatedUnion("status", [ToolStatePending, ToolStateRunning, ToolStateCompleted, ToolStateError])
321
+ .meta({
322
+ ref: "ToolState",
323
+ })
324
+
325
+ export const ToolPart = PartBase.extend({
326
+ type: z.literal("tool"),
327
+ callID: z.string(),
328
+ tool: z.string(),
329
+ state: ToolState,
330
+ metadata: z.record(z.string(), z.any()).optional(),
331
+ }).meta({
332
+ ref: "ToolPart",
333
+ })
334
+ export type ToolPart = z.infer<typeof ToolPart>
335
+
336
+ const Base = z.object({
337
+ id: z.string(),
338
+ sessionID: z.string(),
339
+ })
340
+
341
+ export const User = Base.extend({
342
+ role: z.literal("user"),
343
+ time: z.object({
344
+ created: z.number(),
345
+ }),
346
+ format: Format.optional(),
347
+ summary: z
348
+ .object({
349
+ title: z.string().optional(),
350
+ body: z.string().optional(),
351
+ diffs: Snapshot.FileDiff.array(),
352
+ })
353
+ .optional(),
354
+ agent: z.string(),
355
+ model: z.object({
356
+ providerID: z.string(),
357
+ modelID: z.string(),
358
+ }),
359
+ system: z.string().optional(),
360
+ tools: z.record(z.string(), z.boolean()).optional(),
361
+ variant: z.string().optional(),
362
+ }).meta({
363
+ ref: "UserMessage",
364
+ })
365
+ export type User = z.infer<typeof User>
366
+
367
+ export const Part = z
368
+ .discriminatedUnion("type", [
369
+ TextPart,
370
+ SubtaskPart,
371
+ ReasoningPart,
372
+ FilePart,
373
+ ToolPart,
374
+ StepStartPart,
375
+ StepFinishPart,
376
+ SnapshotPart,
377
+ PatchPart,
378
+ AgentPart,
379
+ RetryPart,
380
+ CompactionPart,
381
+ ])
382
+ .meta({
383
+ ref: "Part",
384
+ })
385
+ export type Part = z.infer<typeof Part>
386
+
387
+ export const Assistant = Base.extend({
388
+ role: z.literal("assistant"),
389
+ time: z.object({
390
+ created: z.number(),
391
+ completed: z.number().optional(),
392
+ }),
393
+ error: z
394
+ .discriminatedUnion("name", [
395
+ AuthError.Schema,
396
+ NamedError.Unknown.Schema,
397
+ OutputLengthError.Schema,
398
+ AbortedError.Schema,
399
+ StructuredOutputError.Schema,
400
+ ContextOverflowError.Schema,
401
+ APIError.Schema,
402
+ ])
403
+ .optional(),
404
+ parentID: z.string(),
405
+ modelID: z.string(),
406
+ providerID: z.string(),
407
+ /**
408
+ * @deprecated
409
+ */
410
+ mode: z.string(),
411
+ agent: z.string(),
412
+ path: z.object({
413
+ cwd: z.string(),
414
+ root: z.string(),
415
+ }),
416
+ summary: z.boolean().optional(),
417
+ cost: z.number(),
418
+ tokens: z.object({
419
+ total: z.number().optional(),
420
+ input: z.number(),
421
+ output: z.number(),
422
+ reasoning: z.number(),
423
+ cache: z.object({
424
+ read: z.number(),
425
+ write: z.number(),
426
+ }),
427
+ }),
428
+ structured: z.any().optional(),
429
+ variant: z.string().optional(),
430
+ finish: z.string().optional(),
431
+ }).meta({
432
+ ref: "AssistantMessage",
433
+ })
434
+ export type Assistant = z.infer<typeof Assistant>
435
+
436
+ export const Info = z.discriminatedUnion("role", [User, Assistant]).meta({
437
+ ref: "Message",
438
+ })
439
+ export type Info = z.infer<typeof Info>
440
+
441
+ export const Event = {
442
+ Updated: BusEvent.define(
443
+ "message.updated",
444
+ z.object({
445
+ info: Info,
446
+ }),
447
+ ),
448
+ Removed: BusEvent.define(
449
+ "message.removed",
450
+ z.object({
451
+ sessionID: z.string(),
452
+ messageID: z.string(),
453
+ }),
454
+ ),
455
+ PartUpdated: BusEvent.define(
456
+ "message.part.updated",
457
+ z.object({
458
+ part: Part,
459
+ delta: z.string().optional(),
460
+ }),
461
+ ),
462
+ PartRemoved: BusEvent.define(
463
+ "message.part.removed",
464
+ z.object({
465
+ sessionID: z.string(),
466
+ messageID: z.string(),
467
+ partID: z.string(),
468
+ }),
469
+ ),
470
+ }
471
+
472
+ export const WithParts = z.object({
473
+ info: Info,
474
+ parts: z.array(Part),
475
+ })
476
+ export type WithParts = z.infer<typeof WithParts>
477
+
478
+ export function toModelMessages(input: WithParts[], model: Provider.Model): ModelMessage[] {
479
+ const result: UIMessage[] = []
480
+ const toolNames = new Set<string>()
481
+ // Track media from tool results that need to be injected as user messages
482
+ // for providers that don't support media in tool results.
483
+ //
484
+ // OpenAI-compatible APIs only support string content in tool results, so we need
485
+ // to extract media and inject as user messages. Other SDKs (anthropic, google,
486
+ // bedrock) handle type: "content" with media parts natively.
487
+ //
488
+ // Only apply this workaround if the model actually supports image input -
489
+ // otherwise there's no point extracting images.
490
+ const supportsMediaInToolResults = (() => {
491
+ if (model.api.npm === "@ai-sdk/anthropic") return true
492
+ if (model.api.npm === "@ai-sdk/openai") return true
493
+ if (model.api.npm === "@ai-sdk/amazon-bedrock") return true
494
+ if (model.api.npm === "@ai-sdk/google-vertex/anthropic") return true
495
+ if (model.api.npm === "@ai-sdk/google") {
496
+ const id = model.api.id.toLowerCase()
497
+ return id.includes("gemini-3") && !id.includes("gemini-2")
498
+ }
499
+ return false
500
+ })()
501
+
502
+ const toModelOutput = (output: unknown) => {
503
+ if (typeof output === "string") {
504
+ return { type: "text", value: output }
505
+ }
506
+
507
+ if (typeof output === "object") {
508
+ const outputObject = output as {
509
+ text: string
510
+ attachments?: Array<{ mime: string; url: string }>
511
+ }
512
+ const attachments = (outputObject.attachments ?? []).filter((attachment) => {
513
+ return attachment.url.startsWith("data:") && attachment.url.includes(",")
514
+ })
515
+
516
+ return {
517
+ type: "content",
518
+ value: [
519
+ { type: "text", text: outputObject.text },
520
+ ...attachments.map((attachment) => ({
521
+ type: "media",
522
+ mediaType: attachment.mime,
523
+ data: iife(() => {
524
+ const commaIndex = attachment.url.indexOf(",")
525
+ return commaIndex === -1 ? attachment.url : attachment.url.slice(commaIndex + 1)
526
+ }),
527
+ })),
528
+ ],
529
+ }
530
+ }
531
+
532
+ return { type: "json", value: output as never }
533
+ }
534
+
535
+ for (const msg of input) {
536
+ if (msg.parts.length === 0) continue
537
+
538
+ if (msg.info.role === "user") {
539
+ const userMessage: UIMessage = {
540
+ id: msg.info.id,
541
+ role: "user",
542
+ parts: [],
543
+ }
544
+ result.push(userMessage)
545
+ for (const part of msg.parts) {
546
+ if (part.type === "text" && !part.ignored)
547
+ userMessage.parts.push({
548
+ type: "text",
549
+ text: part.text,
550
+ })
551
+ // text/plain and directory files are converted into text parts, ignore them
552
+ if (part.type === "file" && part.mime !== "text/plain" && part.mime !== "application/x-directory")
553
+ userMessage.parts.push({
554
+ type: "file",
555
+ url: part.url,
556
+ mediaType: part.mime,
557
+ filename: part.filename,
558
+ })
559
+
560
+ if (part.type === "compaction") {
561
+ userMessage.parts.push({
562
+ type: "text",
563
+ text: "What did we do so far?",
564
+ })
565
+ }
566
+ if (part.type === "subtask") {
567
+ userMessage.parts.push({
568
+ type: "text",
569
+ text: "The following tool was executed by the user",
570
+ })
571
+ }
572
+ }
573
+ }
574
+
575
+ if (msg.info.role === "assistant") {
576
+ const differentModel = `${model.providerID}/${model.id}` !== `${msg.info.providerID}/${msg.info.modelID}`
577
+ const media: Array<{ mime: string; url: string }> = []
578
+
579
+ if (
580
+ msg.info.error &&
581
+ !(
582
+ MessageV2.AbortedError.isInstance(msg.info.error) &&
583
+ msg.parts.some((part) => part.type !== "step-start" && part.type !== "reasoning")
584
+ )
585
+ ) {
586
+ continue
587
+ }
588
+ const assistantMessage: UIMessage = {
589
+ id: msg.info.id,
590
+ role: "assistant",
591
+ parts: [],
592
+ }
593
+ for (const part of msg.parts) {
594
+ if (part.type === "text")
595
+ assistantMessage.parts.push({
596
+ type: "text",
597
+ text: part.text,
598
+ ...(differentModel ? {} : { providerMetadata: part.metadata }),
599
+ })
600
+ if (part.type === "step-start")
601
+ assistantMessage.parts.push({
602
+ type: "step-start",
603
+ })
604
+ if (part.type === "tool") {
605
+ toolNames.add(part.tool)
606
+ if (part.state.status === "completed") {
607
+ const outputText = part.state.time.compacted ? "[Old tool result content cleared]" : part.state.output
608
+ const attachments = part.state.time.compacted ? [] : (part.state.attachments ?? [])
609
+
610
+ // For providers that don't support media in tool results, extract media files
611
+ // (images, PDFs) to be sent as a separate user message
612
+ const isMediaAttachment = (a: { mime: string }) =>
613
+ a.mime.startsWith("image/") || a.mime === "application/pdf"
614
+ const mediaAttachments = attachments.filter(isMediaAttachment)
615
+ const nonMediaAttachments = attachments.filter((a) => !isMediaAttachment(a))
616
+ if (!supportsMediaInToolResults && mediaAttachments.length > 0) {
617
+ media.push(...mediaAttachments)
618
+ }
619
+ const finalAttachments = supportsMediaInToolResults ? attachments : nonMediaAttachments
620
+
621
+ const output =
622
+ finalAttachments.length > 0
623
+ ? {
624
+ text: outputText,
625
+ attachments: finalAttachments,
626
+ }
627
+ : outputText
628
+
629
+ assistantMessage.parts.push({
630
+ type: ("tool-" + part.tool) as `tool-${string}`,
631
+ state: "output-available",
632
+ toolCallId: part.callID,
633
+ input: part.state.input,
634
+ output,
635
+ ...(differentModel ? {} : { callProviderMetadata: part.metadata }),
636
+ })
637
+ }
638
+ if (part.state.status === "error")
639
+ assistantMessage.parts.push({
640
+ type: ("tool-" + part.tool) as `tool-${string}`,
641
+ state: "output-error",
642
+ toolCallId: part.callID,
643
+ input: part.state.input,
644
+ errorText: part.state.error,
645
+ ...(differentModel ? {} : { callProviderMetadata: part.metadata }),
646
+ })
647
+ // Handle pending/running tool calls to prevent dangling tool_use blocks
648
+ // Anthropic/Claude APIs require every tool_use to have a corresponding tool_result
649
+ if (part.state.status === "pending" || part.state.status === "running")
650
+ assistantMessage.parts.push({
651
+ type: ("tool-" + part.tool) as `tool-${string}`,
652
+ state: "output-error",
653
+ toolCallId: part.callID,
654
+ input: part.state.input,
655
+ errorText: "[Tool execution was interrupted]",
656
+ ...(differentModel ? {} : { callProviderMetadata: part.metadata }),
657
+ })
658
+ }
659
+ if (part.type === "reasoning") {
660
+ assistantMessage.parts.push({
661
+ type: "reasoning",
662
+ text: part.text,
663
+ ...(differentModel ? {} : { providerMetadata: part.metadata }),
664
+ })
665
+ }
666
+ }
667
+ if (assistantMessage.parts.length > 0) {
668
+ result.push(assistantMessage)
669
+ // Inject pending media as a user message for providers that don't support
670
+ // media (images, PDFs) in tool results
671
+ if (media.length > 0) {
672
+ result.push({
673
+ id: Identifier.ascending("message"),
674
+ role: "user",
675
+ parts: [
676
+ {
677
+ type: "text" as const,
678
+ text: "Attached image(s) from tool result:",
679
+ },
680
+ ...media.map((attachment) => ({
681
+ type: "file" as const,
682
+ url: attachment.url,
683
+ mediaType: attachment.mime,
684
+ })),
685
+ ],
686
+ })
687
+ }
688
+ }
689
+ }
690
+ }
691
+
692
+ const tools = Object.fromEntries(Array.from(toolNames).map((toolName) => [toolName, { toModelOutput }]))
693
+
694
+ return convertToModelMessages(
695
+ result.filter((msg) => msg.parts.some((part) => part.type !== "step-start")),
696
+ {
697
+ //@ts-expect-error (convertToModelMessages expects a ToolSet but only actually needs tools[name]?.toModelOutput)
698
+ tools,
699
+ },
700
+ )
701
+ }
702
+
703
+ export const stream = fn(Identifier.schema("session"), async function* (sessionID) {
704
+ const list = await Array.fromAsync(await Storage.list(["message", sessionID]))
705
+ for (let i = list.length - 1; i >= 0; i--) {
706
+ yield await get({
707
+ sessionID,
708
+ messageID: list[i][2],
709
+ })
710
+ }
711
+ })
712
+
713
+ export const parts = fn(Identifier.schema("message"), async (messageID) => {
714
+ const result = [] as MessageV2.Part[]
715
+ for (const item of await Storage.list(["part", messageID])) {
716
+ const read = await Storage.read<MessageV2.Part>(item)
717
+ result.push(read)
718
+ }
719
+ result.sort((a, b) => (a.id > b.id ? 1 : -1))
720
+ return result
721
+ })
722
+
723
+ export const get = fn(
724
+ z.object({
725
+ sessionID: Identifier.schema("session"),
726
+ messageID: Identifier.schema("message"),
727
+ }),
728
+ async (input): Promise<WithParts> => {
729
+ return {
730
+ info: await Storage.read<MessageV2.Info>(["message", input.sessionID, input.messageID]),
731
+ parts: await parts(input.messageID),
732
+ }
733
+ },
734
+ )
735
+
736
+ export async function filterCompacted(stream: AsyncIterable<MessageV2.WithParts>) {
737
+ const result = [] as MessageV2.WithParts[]
738
+ const completed = new Set<string>()
739
+ for await (const msg of stream) {
740
+ result.push(msg)
741
+ if (
742
+ msg.info.role === "user" &&
743
+ completed.has(msg.info.id) &&
744
+ msg.parts.some((part) => part.type === "compaction")
745
+ )
746
+ break
747
+ if (msg.info.role === "assistant" && msg.info.summary && msg.info.finish) completed.add(msg.info.parentID)
748
+ }
749
+ result.reverse()
750
+ return result
751
+ }
752
+
753
+ export function fromError(e: unknown, ctx: { providerID: string }) {
754
+ switch (true) {
755
+ case e instanceof DOMException && e.name === "AbortError":
756
+ return new MessageV2.AbortedError(
757
+ { message: e.message },
758
+ {
759
+ cause: e,
760
+ },
761
+ ).toObject()
762
+ case MessageV2.OutputLengthError.isInstance(e):
763
+ return e
764
+ case LoadAPIKeyError.isInstance(e):
765
+ return new MessageV2.AuthError(
766
+ {
767
+ providerID: ctx.providerID,
768
+ message: e.message,
769
+ },
770
+ { cause: e },
771
+ ).toObject()
772
+ case (e as SystemError)?.code === "ECONNRESET":
773
+ return new MessageV2.APIError(
774
+ {
775
+ message: "Connection reset by server",
776
+ isRetryable: true,
777
+ metadata: {
778
+ code: (e as SystemError).code ?? "",
779
+ syscall: (e as SystemError).syscall ?? "",
780
+ message: (e as SystemError).message ?? "",
781
+ },
782
+ },
783
+ { cause: e },
784
+ ).toObject()
785
+ case APICallError.isInstance(e):
786
+ const parsed = ProviderError.parseAPICallError({
787
+ providerID: ctx.providerID,
788
+ error: e,
789
+ })
790
+ if (parsed.type === "context_overflow") {
791
+ return new MessageV2.ContextOverflowError(
792
+ {
793
+ message: parsed.message,
794
+ responseBody: parsed.responseBody,
795
+ },
796
+ { cause: e },
797
+ ).toObject()
798
+ }
799
+
800
+ return new MessageV2.APIError(
801
+ {
802
+ message: parsed.message,
803
+ statusCode: parsed.statusCode,
804
+ isRetryable: parsed.isRetryable,
805
+ responseHeaders: parsed.responseHeaders,
806
+ responseBody: parsed.responseBody,
807
+ metadata: parsed.metadata,
808
+ },
809
+ { cause: e },
810
+ ).toObject()
811
+ case e instanceof Error:
812
+ return new NamedError.Unknown({ message: e.toString() }, { cause: e }).toObject()
813
+ default:
814
+ try {
815
+ const parsed = ProviderError.parseStreamError(e)
816
+ if (parsed) {
817
+ if (parsed.type === "context_overflow") {
818
+ return new MessageV2.ContextOverflowError(
819
+ {
820
+ message: parsed.message,
821
+ responseBody: parsed.responseBody,
822
+ },
823
+ { cause: e },
824
+ ).toObject()
825
+ }
826
+ return new MessageV2.APIError(
827
+ {
828
+ message: parsed.message,
829
+ isRetryable: parsed.isRetryable,
830
+ responseBody: parsed.responseBody,
831
+ },
832
+ {
833
+ cause: e,
834
+ },
835
+ ).toObject()
836
+ }
837
+ } catch {}
838
+ return new NamedError.Unknown({ message: JSON.stringify(e) }, { cause: e }).toObject()
839
+ }
840
+ }
841
+ }