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,780 @@
1
+ import {
2
+ APICallError,
3
+ InvalidResponseDataError,
4
+ type LanguageModelV2,
5
+ type LanguageModelV2CallWarning,
6
+ type LanguageModelV2Content,
7
+ type LanguageModelV2FinishReason,
8
+ type LanguageModelV2StreamPart,
9
+ type SharedV2ProviderMetadata,
10
+ } from "@ai-sdk/provider"
11
+ import {
12
+ combineHeaders,
13
+ createEventSourceResponseHandler,
14
+ createJsonErrorResponseHandler,
15
+ createJsonResponseHandler,
16
+ type FetchFunction,
17
+ generateId,
18
+ isParsableJson,
19
+ parseProviderOptions,
20
+ type ParseResult,
21
+ postJsonToApi,
22
+ type ResponseHandler,
23
+ } from "@ai-sdk/provider-utils"
24
+ import { z } from "zod/v4"
25
+ import { convertToOpenAICompatibleChatMessages } from "./convert-to-openai-compatible-chat-messages"
26
+ import { getResponseMetadata } from "./get-response-metadata"
27
+ import { mapOpenAICompatibleFinishReason } from "./map-openai-compatible-finish-reason"
28
+ import { type OpenAICompatibleChatModelId, openaiCompatibleProviderOptions } from "./openai-compatible-chat-options"
29
+ import { defaultOpenAICompatibleErrorStructure, type ProviderErrorStructure } from "../openai-compatible-error"
30
+ import type { MetadataExtractor } from "./openai-compatible-metadata-extractor"
31
+ import { prepareTools } from "./openai-compatible-prepare-tools"
32
+
33
+ export type OpenAICompatibleChatConfig = {
34
+ provider: string
35
+ headers: () => Record<string, string | undefined>
36
+ url: (options: { modelId: string; path: string }) => string
37
+ fetch?: FetchFunction
38
+ includeUsage?: boolean
39
+ errorStructure?: ProviderErrorStructure<any>
40
+ metadataExtractor?: MetadataExtractor
41
+
42
+ /**
43
+ * Whether the model supports structured outputs.
44
+ */
45
+ supportsStructuredOutputs?: boolean
46
+
47
+ /**
48
+ * The supported URLs for the model.
49
+ */
50
+ supportedUrls?: () => LanguageModelV2["supportedUrls"]
51
+ }
52
+
53
+ export class OpenAICompatibleChatLanguageModel implements LanguageModelV2 {
54
+ readonly specificationVersion = "v2"
55
+
56
+ readonly supportsStructuredOutputs: boolean
57
+
58
+ readonly modelId: OpenAICompatibleChatModelId
59
+ private readonly config: OpenAICompatibleChatConfig
60
+ private readonly failedResponseHandler: ResponseHandler<APICallError>
61
+ private readonly chunkSchema // type inferred via constructor
62
+
63
+ constructor(modelId: OpenAICompatibleChatModelId, config: OpenAICompatibleChatConfig) {
64
+ this.modelId = modelId
65
+ this.config = config
66
+
67
+ // initialize error handling:
68
+ const errorStructure = config.errorStructure ?? defaultOpenAICompatibleErrorStructure
69
+ this.chunkSchema = createOpenAICompatibleChatChunkSchema(errorStructure.errorSchema)
70
+ this.failedResponseHandler = createJsonErrorResponseHandler(errorStructure)
71
+
72
+ this.supportsStructuredOutputs = config.supportsStructuredOutputs ?? false
73
+ }
74
+
75
+ get provider(): string {
76
+ return this.config.provider
77
+ }
78
+
79
+ private get providerOptionsName(): string {
80
+ return this.config.provider.split(".")[0].trim()
81
+ }
82
+
83
+ get supportedUrls() {
84
+ return this.config.supportedUrls?.() ?? {}
85
+ }
86
+
87
+ private async getArgs({
88
+ prompt,
89
+ maxOutputTokens,
90
+ temperature,
91
+ topP,
92
+ topK,
93
+ frequencyPenalty,
94
+ presencePenalty,
95
+ providerOptions,
96
+ stopSequences,
97
+ responseFormat,
98
+ seed,
99
+ toolChoice,
100
+ tools,
101
+ }: Parameters<LanguageModelV2["doGenerate"]>[0]) {
102
+ const warnings: LanguageModelV2CallWarning[] = []
103
+
104
+ // Parse provider options
105
+ const compatibleOptions = Object.assign(
106
+ (await parseProviderOptions({
107
+ provider: "copilot",
108
+ providerOptions,
109
+ schema: openaiCompatibleProviderOptions,
110
+ })) ?? {},
111
+ (await parseProviderOptions({
112
+ provider: this.providerOptionsName,
113
+ providerOptions,
114
+ schema: openaiCompatibleProviderOptions,
115
+ })) ?? {},
116
+ )
117
+
118
+ if (topK != null) {
119
+ warnings.push({ type: "unsupported-setting", setting: "topK" })
120
+ }
121
+
122
+ if (responseFormat?.type === "json" && responseFormat.schema != null && !this.supportsStructuredOutputs) {
123
+ warnings.push({
124
+ type: "unsupported-setting",
125
+ setting: "responseFormat",
126
+ details: "JSON response format schema is only supported with structuredOutputs",
127
+ })
128
+ }
129
+
130
+ const {
131
+ tools: openaiTools,
132
+ toolChoice: openaiToolChoice,
133
+ toolWarnings,
134
+ } = prepareTools({
135
+ tools,
136
+ toolChoice,
137
+ })
138
+
139
+ return {
140
+ args: {
141
+ // model id:
142
+ model: this.modelId,
143
+
144
+ // model specific settings:
145
+ user: compatibleOptions.user,
146
+
147
+ // standardized settings:
148
+ max_tokens: maxOutputTokens,
149
+ temperature,
150
+ top_p: topP,
151
+ frequency_penalty: frequencyPenalty,
152
+ presence_penalty: presencePenalty,
153
+ response_format:
154
+ responseFormat?.type === "json"
155
+ ? this.supportsStructuredOutputs === true && responseFormat.schema != null
156
+ ? {
157
+ type: "json_schema",
158
+ json_schema: {
159
+ schema: responseFormat.schema,
160
+ name: responseFormat.name ?? "response",
161
+ description: responseFormat.description,
162
+ },
163
+ }
164
+ : { type: "json_object" }
165
+ : undefined,
166
+
167
+ stop: stopSequences,
168
+ seed,
169
+ ...Object.fromEntries(
170
+ Object.entries(providerOptions?.[this.providerOptionsName] ?? {}).filter(
171
+ ([key]) => !Object.keys(openaiCompatibleProviderOptions.shape).includes(key),
172
+ ),
173
+ ),
174
+
175
+ reasoning_effort: compatibleOptions.reasoningEffort,
176
+ verbosity: compatibleOptions.textVerbosity,
177
+
178
+ // messages:
179
+ messages: convertToOpenAICompatibleChatMessages(prompt),
180
+
181
+ // tools:
182
+ tools: openaiTools,
183
+ tool_choice: openaiToolChoice,
184
+
185
+ // thinking_budget
186
+ thinking_budget: compatibleOptions.thinking_budget,
187
+ },
188
+ warnings: [...warnings, ...toolWarnings],
189
+ }
190
+ }
191
+
192
+ async doGenerate(
193
+ options: Parameters<LanguageModelV2["doGenerate"]>[0],
194
+ ): Promise<Awaited<ReturnType<LanguageModelV2["doGenerate"]>>> {
195
+ const { args, warnings } = await this.getArgs({ ...options })
196
+
197
+ const body = JSON.stringify(args)
198
+
199
+ const {
200
+ responseHeaders,
201
+ value: responseBody,
202
+ rawValue: rawResponse,
203
+ } = await postJsonToApi({
204
+ url: this.config.url({
205
+ path: "/chat/completions",
206
+ modelId: this.modelId,
207
+ }),
208
+ headers: combineHeaders(this.config.headers(), options.headers),
209
+ body: args,
210
+ failedResponseHandler: this.failedResponseHandler,
211
+ successfulResponseHandler: createJsonResponseHandler(OpenAICompatibleChatResponseSchema),
212
+ abortSignal: options.abortSignal,
213
+ fetch: this.config.fetch,
214
+ })
215
+
216
+ const choice = responseBody.choices[0]
217
+ const content: Array<LanguageModelV2Content> = []
218
+
219
+ // text content:
220
+ const text = choice.message.content
221
+ if (text != null && text.length > 0) {
222
+ content.push({
223
+ type: "text",
224
+ text,
225
+ providerMetadata: choice.message.reasoning_opaque
226
+ ? { copilot: { reasoningOpaque: choice.message.reasoning_opaque } }
227
+ : undefined,
228
+ })
229
+ }
230
+
231
+ // reasoning content (Copilot uses reasoning_text):
232
+ const reasoning = choice.message.reasoning_text
233
+ if (reasoning != null && reasoning.length > 0) {
234
+ content.push({
235
+ type: "reasoning",
236
+ text: reasoning,
237
+ // Include reasoning_opaque for Copilot multi-turn reasoning
238
+ providerMetadata: choice.message.reasoning_opaque
239
+ ? { copilot: { reasoningOpaque: choice.message.reasoning_opaque } }
240
+ : undefined,
241
+ })
242
+ }
243
+
244
+ // tool calls:
245
+ if (choice.message.tool_calls != null) {
246
+ for (const toolCall of choice.message.tool_calls) {
247
+ content.push({
248
+ type: "tool-call",
249
+ toolCallId: toolCall.id ?? generateId(),
250
+ toolName: toolCall.function.name,
251
+ input: toolCall.function.arguments!,
252
+ providerMetadata: choice.message.reasoning_opaque
253
+ ? { copilot: { reasoningOpaque: choice.message.reasoning_opaque } }
254
+ : undefined,
255
+ })
256
+ }
257
+ }
258
+
259
+ // provider metadata:
260
+ const providerMetadata: SharedV2ProviderMetadata = {
261
+ [this.providerOptionsName]: {},
262
+ ...(await this.config.metadataExtractor?.extractMetadata?.({
263
+ parsedBody: rawResponse,
264
+ })),
265
+ }
266
+ const completionTokenDetails = responseBody.usage?.completion_tokens_details
267
+ if (completionTokenDetails?.accepted_prediction_tokens != null) {
268
+ providerMetadata[this.providerOptionsName].acceptedPredictionTokens =
269
+ completionTokenDetails?.accepted_prediction_tokens
270
+ }
271
+ if (completionTokenDetails?.rejected_prediction_tokens != null) {
272
+ providerMetadata[this.providerOptionsName].rejectedPredictionTokens =
273
+ completionTokenDetails?.rejected_prediction_tokens
274
+ }
275
+
276
+ return {
277
+ content,
278
+ finishReason: mapOpenAICompatibleFinishReason(choice.finish_reason),
279
+ usage: {
280
+ inputTokens: responseBody.usage?.prompt_tokens ?? undefined,
281
+ outputTokens: responseBody.usage?.completion_tokens ?? undefined,
282
+ totalTokens: responseBody.usage?.total_tokens ?? undefined,
283
+ reasoningTokens: responseBody.usage?.completion_tokens_details?.reasoning_tokens ?? undefined,
284
+ cachedInputTokens: responseBody.usage?.prompt_tokens_details?.cached_tokens ?? undefined,
285
+ },
286
+ providerMetadata,
287
+ request: { body },
288
+ response: {
289
+ ...getResponseMetadata(responseBody),
290
+ headers: responseHeaders,
291
+ body: rawResponse,
292
+ },
293
+ warnings,
294
+ }
295
+ }
296
+
297
+ async doStream(
298
+ options: Parameters<LanguageModelV2["doStream"]>[0],
299
+ ): Promise<Awaited<ReturnType<LanguageModelV2["doStream"]>>> {
300
+ const { args, warnings } = await this.getArgs({ ...options })
301
+
302
+ const body = {
303
+ ...args,
304
+ stream: true,
305
+
306
+ // only include stream_options when in strict compatibility mode:
307
+ stream_options: this.config.includeUsage ? { include_usage: true } : undefined,
308
+ }
309
+
310
+ const metadataExtractor = this.config.metadataExtractor?.createStreamExtractor()
311
+
312
+ const { responseHeaders, value: response } = await postJsonToApi({
313
+ url: this.config.url({
314
+ path: "/chat/completions",
315
+ modelId: this.modelId,
316
+ }),
317
+ headers: combineHeaders(this.config.headers(), options.headers),
318
+ body,
319
+ failedResponseHandler: this.failedResponseHandler,
320
+ successfulResponseHandler: createEventSourceResponseHandler(this.chunkSchema),
321
+ abortSignal: options.abortSignal,
322
+ fetch: this.config.fetch,
323
+ })
324
+
325
+ const toolCalls: Array<{
326
+ id: string
327
+ type: "function"
328
+ function: {
329
+ name: string
330
+ arguments: string
331
+ }
332
+ hasFinished: boolean
333
+ }> = []
334
+
335
+ let finishReason: LanguageModelV2FinishReason = "unknown"
336
+ const usage: {
337
+ completionTokens: number | undefined
338
+ completionTokensDetails: {
339
+ reasoningTokens: number | undefined
340
+ acceptedPredictionTokens: number | undefined
341
+ rejectedPredictionTokens: number | undefined
342
+ }
343
+ promptTokens: number | undefined
344
+ promptTokensDetails: {
345
+ cachedTokens: number | undefined
346
+ }
347
+ totalTokens: number | undefined
348
+ } = {
349
+ completionTokens: undefined,
350
+ completionTokensDetails: {
351
+ reasoningTokens: undefined,
352
+ acceptedPredictionTokens: undefined,
353
+ rejectedPredictionTokens: undefined,
354
+ },
355
+ promptTokens: undefined,
356
+ promptTokensDetails: {
357
+ cachedTokens: undefined,
358
+ },
359
+ totalTokens: undefined,
360
+ }
361
+ let isFirstChunk = true
362
+ const providerOptionsName = this.providerOptionsName
363
+ let isActiveReasoning = false
364
+ let isActiveText = false
365
+ let reasoningOpaque: string | undefined
366
+
367
+ return {
368
+ stream: response.pipeThrough(
369
+ new TransformStream<ParseResult<z.infer<typeof this.chunkSchema>>, LanguageModelV2StreamPart>({
370
+ start(controller) {
371
+ controller.enqueue({ type: "stream-start", warnings })
372
+ },
373
+
374
+ // TODO we lost type safety on Chunk, most likely due to the error schema. MUST FIX
375
+ transform(chunk, controller) {
376
+ // Emit raw chunk if requested (before anything else)
377
+ if (options.includeRawChunks) {
378
+ controller.enqueue({ type: "raw", rawValue: chunk.rawValue })
379
+ }
380
+
381
+ // handle failed chunk parsing / validation:
382
+ if (!chunk.success) {
383
+ finishReason = "error"
384
+ controller.enqueue({ type: "error", error: chunk.error })
385
+ return
386
+ }
387
+ const value = chunk.value
388
+
389
+ metadataExtractor?.processChunk(chunk.rawValue)
390
+
391
+ // handle error chunks:
392
+ if ("error" in value) {
393
+ finishReason = "error"
394
+ controller.enqueue({ type: "error", error: value.error.message })
395
+ return
396
+ }
397
+
398
+ if (isFirstChunk) {
399
+ isFirstChunk = false
400
+
401
+ controller.enqueue({
402
+ type: "response-metadata",
403
+ ...getResponseMetadata(value),
404
+ })
405
+ }
406
+
407
+ if (value.usage != null) {
408
+ const {
409
+ prompt_tokens,
410
+ completion_tokens,
411
+ total_tokens,
412
+ prompt_tokens_details,
413
+ completion_tokens_details,
414
+ } = value.usage
415
+
416
+ usage.promptTokens = prompt_tokens ?? undefined
417
+ usage.completionTokens = completion_tokens ?? undefined
418
+ usage.totalTokens = total_tokens ?? undefined
419
+ if (completion_tokens_details?.reasoning_tokens != null) {
420
+ usage.completionTokensDetails.reasoningTokens = completion_tokens_details?.reasoning_tokens
421
+ }
422
+ if (completion_tokens_details?.accepted_prediction_tokens != null) {
423
+ usage.completionTokensDetails.acceptedPredictionTokens =
424
+ completion_tokens_details?.accepted_prediction_tokens
425
+ }
426
+ if (completion_tokens_details?.rejected_prediction_tokens != null) {
427
+ usage.completionTokensDetails.rejectedPredictionTokens =
428
+ completion_tokens_details?.rejected_prediction_tokens
429
+ }
430
+ if (prompt_tokens_details?.cached_tokens != null) {
431
+ usage.promptTokensDetails.cachedTokens = prompt_tokens_details?.cached_tokens
432
+ }
433
+ }
434
+
435
+ const choice = value.choices[0]
436
+
437
+ if (choice?.finish_reason != null) {
438
+ finishReason = mapOpenAICompatibleFinishReason(choice.finish_reason)
439
+ }
440
+
441
+ if (choice?.delta == null) {
442
+ return
443
+ }
444
+
445
+ const delta = choice.delta
446
+
447
+ // Capture reasoning_opaque for Copilot multi-turn reasoning
448
+ if (delta.reasoning_opaque) {
449
+ if (reasoningOpaque != null) {
450
+ throw new InvalidResponseDataError({
451
+ data: delta,
452
+ message:
453
+ "Multiple reasoning_opaque values received in a single response. Only one thinking part per response is supported.",
454
+ })
455
+ }
456
+ reasoningOpaque = delta.reasoning_opaque
457
+ }
458
+
459
+ // enqueue reasoning before text deltas (Copilot uses reasoning_text):
460
+ const reasoningContent = delta.reasoning_text
461
+ if (reasoningContent) {
462
+ if (!isActiveReasoning) {
463
+ controller.enqueue({
464
+ type: "reasoning-start",
465
+ id: "reasoning-0",
466
+ })
467
+ isActiveReasoning = true
468
+ }
469
+
470
+ controller.enqueue({
471
+ type: "reasoning-delta",
472
+ id: "reasoning-0",
473
+ delta: reasoningContent,
474
+ })
475
+ }
476
+
477
+ if (delta.content) {
478
+ // If reasoning was active and we're starting text, end reasoning first
479
+ // This handles the case where reasoning_opaque and content come in the same chunk
480
+ if (isActiveReasoning && !isActiveText) {
481
+ controller.enqueue({
482
+ type: "reasoning-end",
483
+ id: "reasoning-0",
484
+ providerMetadata: reasoningOpaque ? { copilot: { reasoningOpaque } } : undefined,
485
+ })
486
+ isActiveReasoning = false
487
+ }
488
+
489
+ if (!isActiveText) {
490
+ controller.enqueue({
491
+ type: "text-start",
492
+ id: "txt-0",
493
+ providerMetadata: reasoningOpaque ? { copilot: { reasoningOpaque } } : undefined,
494
+ })
495
+ isActiveText = true
496
+ }
497
+
498
+ controller.enqueue({
499
+ type: "text-delta",
500
+ id: "txt-0",
501
+ delta: delta.content,
502
+ })
503
+ }
504
+
505
+ if (delta.tool_calls != null) {
506
+ // If reasoning was active and we're starting tool calls, end reasoning first
507
+ // This handles the case where reasoning goes directly to tool calls with no content
508
+ if (isActiveReasoning) {
509
+ controller.enqueue({
510
+ type: "reasoning-end",
511
+ id: "reasoning-0",
512
+ providerMetadata: reasoningOpaque ? { copilot: { reasoningOpaque } } : undefined,
513
+ })
514
+ isActiveReasoning = false
515
+ }
516
+ for (const toolCallDelta of delta.tool_calls) {
517
+ const index = toolCallDelta.index
518
+
519
+ if (toolCalls[index] == null) {
520
+ if (toolCallDelta.id == null) {
521
+ throw new InvalidResponseDataError({
522
+ data: toolCallDelta,
523
+ message: `Expected 'id' to be a string.`,
524
+ })
525
+ }
526
+
527
+ if (toolCallDelta.function?.name == null) {
528
+ throw new InvalidResponseDataError({
529
+ data: toolCallDelta,
530
+ message: `Expected 'function.name' to be a string.`,
531
+ })
532
+ }
533
+
534
+ controller.enqueue({
535
+ type: "tool-input-start",
536
+ id: toolCallDelta.id,
537
+ toolName: toolCallDelta.function.name,
538
+ })
539
+
540
+ toolCalls[index] = {
541
+ id: toolCallDelta.id,
542
+ type: "function",
543
+ function: {
544
+ name: toolCallDelta.function.name,
545
+ arguments: toolCallDelta.function.arguments ?? "",
546
+ },
547
+ hasFinished: false,
548
+ }
549
+
550
+ const toolCall = toolCalls[index]
551
+
552
+ if (toolCall.function?.name != null && toolCall.function?.arguments != null) {
553
+ // send delta if the argument text has already started:
554
+ if (toolCall.function.arguments.length > 0) {
555
+ controller.enqueue({
556
+ type: "tool-input-delta",
557
+ id: toolCall.id,
558
+ delta: toolCall.function.arguments,
559
+ })
560
+ }
561
+
562
+ // check if tool call is complete
563
+ // (some providers send the full tool call in one chunk):
564
+ if (isParsableJson(toolCall.function.arguments)) {
565
+ controller.enqueue({
566
+ type: "tool-input-end",
567
+ id: toolCall.id,
568
+ })
569
+
570
+ controller.enqueue({
571
+ type: "tool-call",
572
+ toolCallId: toolCall.id ?? generateId(),
573
+ toolName: toolCall.function.name,
574
+ input: toolCall.function.arguments,
575
+ providerMetadata: reasoningOpaque ? { copilot: { reasoningOpaque } } : undefined,
576
+ })
577
+ toolCall.hasFinished = true
578
+ }
579
+ }
580
+
581
+ continue
582
+ }
583
+
584
+ // existing tool call, merge if not finished
585
+ const toolCall = toolCalls[index]
586
+
587
+ if (toolCall.hasFinished) {
588
+ continue
589
+ }
590
+
591
+ if (toolCallDelta.function?.arguments != null) {
592
+ toolCall.function!.arguments += toolCallDelta.function?.arguments ?? ""
593
+ }
594
+
595
+ // send delta
596
+ controller.enqueue({
597
+ type: "tool-input-delta",
598
+ id: toolCall.id,
599
+ delta: toolCallDelta.function.arguments ?? "",
600
+ })
601
+
602
+ // check if tool call is complete
603
+ if (
604
+ toolCall.function?.name != null &&
605
+ toolCall.function?.arguments != null &&
606
+ isParsableJson(toolCall.function.arguments)
607
+ ) {
608
+ controller.enqueue({
609
+ type: "tool-input-end",
610
+ id: toolCall.id,
611
+ })
612
+
613
+ controller.enqueue({
614
+ type: "tool-call",
615
+ toolCallId: toolCall.id ?? generateId(),
616
+ toolName: toolCall.function.name,
617
+ input: toolCall.function.arguments,
618
+ providerMetadata: reasoningOpaque ? { copilot: { reasoningOpaque } } : undefined,
619
+ })
620
+ toolCall.hasFinished = true
621
+ }
622
+ }
623
+ }
624
+ },
625
+
626
+ flush(controller) {
627
+ if (isActiveReasoning) {
628
+ controller.enqueue({
629
+ type: "reasoning-end",
630
+ id: "reasoning-0",
631
+ // Include reasoning_opaque for Copilot multi-turn reasoning
632
+ providerMetadata: reasoningOpaque ? { copilot: { reasoningOpaque } } : undefined,
633
+ })
634
+ }
635
+
636
+ if (isActiveText) {
637
+ controller.enqueue({ type: "text-end", id: "txt-0" })
638
+ }
639
+
640
+ // go through all tool calls and send the ones that are not finished
641
+ for (const toolCall of toolCalls.filter((toolCall) => !toolCall.hasFinished)) {
642
+ controller.enqueue({
643
+ type: "tool-input-end",
644
+ id: toolCall.id,
645
+ })
646
+
647
+ controller.enqueue({
648
+ type: "tool-call",
649
+ toolCallId: toolCall.id ?? generateId(),
650
+ toolName: toolCall.function.name,
651
+ input: toolCall.function.arguments,
652
+ })
653
+ }
654
+
655
+ const providerMetadata: SharedV2ProviderMetadata = {
656
+ [providerOptionsName]: {},
657
+ // Include reasoning_opaque for Copilot multi-turn reasoning
658
+ ...(reasoningOpaque ? { copilot: { reasoningOpaque } } : {}),
659
+ ...metadataExtractor?.buildMetadata(),
660
+ }
661
+ if (usage.completionTokensDetails.acceptedPredictionTokens != null) {
662
+ providerMetadata[providerOptionsName].acceptedPredictionTokens =
663
+ usage.completionTokensDetails.acceptedPredictionTokens
664
+ }
665
+ if (usage.completionTokensDetails.rejectedPredictionTokens != null) {
666
+ providerMetadata[providerOptionsName].rejectedPredictionTokens =
667
+ usage.completionTokensDetails.rejectedPredictionTokens
668
+ }
669
+
670
+ controller.enqueue({
671
+ type: "finish",
672
+ finishReason,
673
+ usage: {
674
+ inputTokens: usage.promptTokens ?? undefined,
675
+ outputTokens: usage.completionTokens ?? undefined,
676
+ totalTokens: usage.totalTokens ?? undefined,
677
+ reasoningTokens: usage.completionTokensDetails.reasoningTokens ?? undefined,
678
+ cachedInputTokens: usage.promptTokensDetails.cachedTokens ?? undefined,
679
+ },
680
+ providerMetadata,
681
+ })
682
+ },
683
+ }),
684
+ ),
685
+ request: { body },
686
+ response: { headers: responseHeaders },
687
+ }
688
+ }
689
+ }
690
+
691
+ const openaiCompatibleTokenUsageSchema = z
692
+ .object({
693
+ prompt_tokens: z.number().nullish(),
694
+ completion_tokens: z.number().nullish(),
695
+ total_tokens: z.number().nullish(),
696
+ prompt_tokens_details: z
697
+ .object({
698
+ cached_tokens: z.number().nullish(),
699
+ })
700
+ .nullish(),
701
+ completion_tokens_details: z
702
+ .object({
703
+ reasoning_tokens: z.number().nullish(),
704
+ accepted_prediction_tokens: z.number().nullish(),
705
+ rejected_prediction_tokens: z.number().nullish(),
706
+ })
707
+ .nullish(),
708
+ })
709
+ .nullish()
710
+
711
+ // limited version of the schema, focussed on what is needed for the implementation
712
+ // this approach limits breakages when the API changes and increases efficiency
713
+ const OpenAICompatibleChatResponseSchema = z.object({
714
+ id: z.string().nullish(),
715
+ created: z.number().nullish(),
716
+ model: z.string().nullish(),
717
+ choices: z.array(
718
+ z.object({
719
+ message: z.object({
720
+ role: z.literal("assistant").nullish(),
721
+ content: z.string().nullish(),
722
+ // Copilot-specific reasoning fields
723
+ reasoning_text: z.string().nullish(),
724
+ reasoning_opaque: z.string().nullish(),
725
+ tool_calls: z
726
+ .array(
727
+ z.object({
728
+ id: z.string().nullish(),
729
+ function: z.object({
730
+ name: z.string(),
731
+ arguments: z.string(),
732
+ }),
733
+ }),
734
+ )
735
+ .nullish(),
736
+ }),
737
+ finish_reason: z.string().nullish(),
738
+ }),
739
+ ),
740
+ usage: openaiCompatibleTokenUsageSchema,
741
+ })
742
+
743
+ // limited version of the schema, focussed on what is needed for the implementation
744
+ // this approach limits breakages when the API changes and increases efficiency
745
+ const createOpenAICompatibleChatChunkSchema = <ERROR_SCHEMA extends z.core.$ZodType>(errorSchema: ERROR_SCHEMA) =>
746
+ z.union([
747
+ z.object({
748
+ id: z.string().nullish(),
749
+ created: z.number().nullish(),
750
+ model: z.string().nullish(),
751
+ choices: z.array(
752
+ z.object({
753
+ delta: z
754
+ .object({
755
+ role: z.enum(["assistant"]).nullish(),
756
+ content: z.string().nullish(),
757
+ // Copilot-specific reasoning fields
758
+ reasoning_text: z.string().nullish(),
759
+ reasoning_opaque: z.string().nullish(),
760
+ tool_calls: z
761
+ .array(
762
+ z.object({
763
+ index: z.number(),
764
+ id: z.string().nullish(),
765
+ function: z.object({
766
+ name: z.string().nullish(),
767
+ arguments: z.string().nullish(),
768
+ }),
769
+ }),
770
+ )
771
+ .nullish(),
772
+ })
773
+ .nullish(),
774
+ finish_reason: z.string().nullish(),
775
+ }),
776
+ ),
777
+ usage: openaiCompatibleTokenUsageSchema,
778
+ }),
779
+ errorSchema,
780
+ ])