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,624 @@
1
+ import type { Hooks, PluginInput } from "@opencode-ai/plugin"
2
+ import { Log } from "../util/log"
3
+ import { Installation } from "../installation"
4
+ import { Auth, OAUTH_DUMMY_KEY } from "../auth"
5
+ import os from "os"
6
+ import { ProviderTransform } from "@/provider/transform"
7
+
8
+ const log = Log.create({ service: "plugin.codex" })
9
+
10
+ const CLIENT_ID = "app_EMoamEEZ73f0CkXaXp7hrann"
11
+ const ISSUER = "https://auth.openai.com"
12
+ const CODEX_API_ENDPOINT = "https://chatgpt.com/backend-api/codex/responses"
13
+ const OAUTH_PORT = 1455
14
+ const OAUTH_POLLING_SAFETY_MARGIN_MS = 3000
15
+
16
+ interface PkceCodes {
17
+ verifier: string
18
+ challenge: string
19
+ }
20
+
21
+ async function generatePKCE(): Promise<PkceCodes> {
22
+ const verifier = generateRandomString(43)
23
+ const encoder = new TextEncoder()
24
+ const data = encoder.encode(verifier)
25
+ const hash = await crypto.subtle.digest("SHA-256", data)
26
+ const challenge = base64UrlEncode(hash)
27
+ return { verifier, challenge }
28
+ }
29
+
30
+ function generateRandomString(length: number): string {
31
+ const chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-._~"
32
+ const bytes = crypto.getRandomValues(new Uint8Array(length))
33
+ return Array.from(bytes)
34
+ .map((b) => chars[b % chars.length])
35
+ .join("")
36
+ }
37
+
38
+ function base64UrlEncode(buffer: ArrayBuffer): string {
39
+ const bytes = new Uint8Array(buffer)
40
+ const binary = String.fromCharCode(...bytes)
41
+ return btoa(binary).replace(/\+/g, "-").replace(/\//g, "_").replace(/=+$/, "")
42
+ }
43
+
44
+ function generateState(): string {
45
+ return base64UrlEncode(crypto.getRandomValues(new Uint8Array(32)).buffer)
46
+ }
47
+
48
+ export interface IdTokenClaims {
49
+ chatgpt_account_id?: string
50
+ organizations?: Array<{ id: string }>
51
+ email?: string
52
+ "https://api.openai.com/auth"?: {
53
+ chatgpt_account_id?: string
54
+ }
55
+ }
56
+
57
+ export function parseJwtClaims(token: string): IdTokenClaims | undefined {
58
+ const parts = token.split(".")
59
+ if (parts.length !== 3) return undefined
60
+ try {
61
+ return JSON.parse(Buffer.from(parts[1], "base64url").toString())
62
+ } catch {
63
+ return undefined
64
+ }
65
+ }
66
+
67
+ export function extractAccountIdFromClaims(claims: IdTokenClaims): string | undefined {
68
+ return (
69
+ claims.chatgpt_account_id ||
70
+ claims["https://api.openai.com/auth"]?.chatgpt_account_id ||
71
+ claims.organizations?.[0]?.id
72
+ )
73
+ }
74
+
75
+ export function extractAccountId(tokens: TokenResponse): string | undefined {
76
+ if (tokens.id_token) {
77
+ const claims = parseJwtClaims(tokens.id_token)
78
+ const accountId = claims && extractAccountIdFromClaims(claims)
79
+ if (accountId) return accountId
80
+ }
81
+ if (tokens.access_token) {
82
+ const claims = parseJwtClaims(tokens.access_token)
83
+ return claims ? extractAccountIdFromClaims(claims) : undefined
84
+ }
85
+ return undefined
86
+ }
87
+
88
+ function buildAuthorizeUrl(redirectUri: string, pkce: PkceCodes, state: string): string {
89
+ const params = new URLSearchParams({
90
+ response_type: "code",
91
+ client_id: CLIENT_ID,
92
+ redirect_uri: redirectUri,
93
+ scope: "openid profile email offline_access",
94
+ code_challenge: pkce.challenge,
95
+ code_challenge_method: "S256",
96
+ id_token_add_organizations: "true",
97
+ codex_cli_simplified_flow: "true",
98
+ state,
99
+ originator: "opencode",
100
+ })
101
+ return `${ISSUER}/oauth/authorize?${params.toString()}`
102
+ }
103
+
104
+ interface TokenResponse {
105
+ id_token: string
106
+ access_token: string
107
+ refresh_token: string
108
+ expires_in?: number
109
+ }
110
+
111
+ async function exchangeCodeForTokens(code: string, redirectUri: string, pkce: PkceCodes): Promise<TokenResponse> {
112
+ const response = await fetch(`${ISSUER}/oauth/token`, {
113
+ method: "POST",
114
+ headers: { "Content-Type": "application/x-www-form-urlencoded" },
115
+ body: new URLSearchParams({
116
+ grant_type: "authorization_code",
117
+ code,
118
+ redirect_uri: redirectUri,
119
+ client_id: CLIENT_ID,
120
+ code_verifier: pkce.verifier,
121
+ }).toString(),
122
+ })
123
+ if (!response.ok) {
124
+ throw new Error(`Token exchange failed: ${response.status}`)
125
+ }
126
+ return response.json()
127
+ }
128
+
129
+ async function refreshAccessToken(refreshToken: string): Promise<TokenResponse> {
130
+ const response = await fetch(`${ISSUER}/oauth/token`, {
131
+ method: "POST",
132
+ headers: { "Content-Type": "application/x-www-form-urlencoded" },
133
+ body: new URLSearchParams({
134
+ grant_type: "refresh_token",
135
+ refresh_token: refreshToken,
136
+ client_id: CLIENT_ID,
137
+ }).toString(),
138
+ })
139
+ if (!response.ok) {
140
+ throw new Error(`Token refresh failed: ${response.status}`)
141
+ }
142
+ return response.json()
143
+ }
144
+
145
+ const HTML_SUCCESS = `<!doctype html>
146
+ <html>
147
+ <head>
148
+ <title>OpenCode - Codex Authorization Successful</title>
149
+ <style>
150
+ body {
151
+ font-family:
152
+ system-ui,
153
+ -apple-system,
154
+ sans-serif;
155
+ display: flex;
156
+ justify-content: center;
157
+ align-items: center;
158
+ height: 100vh;
159
+ margin: 0;
160
+ background: #131010;
161
+ color: #f1ecec;
162
+ }
163
+ .container {
164
+ text-align: center;
165
+ padding: 2rem;
166
+ }
167
+ h1 {
168
+ color: #f1ecec;
169
+ margin-bottom: 1rem;
170
+ }
171
+ p {
172
+ color: #b7b1b1;
173
+ }
174
+ </style>
175
+ </head>
176
+ <body>
177
+ <div class="container">
178
+ <h1>Authorization Successful</h1>
179
+ <p>You can close this window and return to OpenCode.</p>
180
+ </div>
181
+ <script>
182
+ setTimeout(() => window.close(), 2000)
183
+ </script>
184
+ </body>
185
+ </html>`
186
+
187
+ const HTML_ERROR = (error: string) => `<!doctype html>
188
+ <html>
189
+ <head>
190
+ <title>OpenCode - Codex Authorization Failed</title>
191
+ <style>
192
+ body {
193
+ font-family:
194
+ system-ui,
195
+ -apple-system,
196
+ sans-serif;
197
+ display: flex;
198
+ justify-content: center;
199
+ align-items: center;
200
+ height: 100vh;
201
+ margin: 0;
202
+ background: #131010;
203
+ color: #f1ecec;
204
+ }
205
+ .container {
206
+ text-align: center;
207
+ padding: 2rem;
208
+ }
209
+ h1 {
210
+ color: #fc533a;
211
+ margin-bottom: 1rem;
212
+ }
213
+ p {
214
+ color: #b7b1b1;
215
+ }
216
+ .error {
217
+ color: #ff917b;
218
+ font-family: monospace;
219
+ margin-top: 1rem;
220
+ padding: 1rem;
221
+ background: #3c140d;
222
+ border-radius: 0.5rem;
223
+ }
224
+ </style>
225
+ </head>
226
+ <body>
227
+ <div class="container">
228
+ <h1>Authorization Failed</h1>
229
+ <p>An error occurred during authorization.</p>
230
+ <div class="error">${error}</div>
231
+ </div>
232
+ </body>
233
+ </html>`
234
+
235
+ interface PendingOAuth {
236
+ pkce: PkceCodes
237
+ state: string
238
+ resolve: (tokens: TokenResponse) => void
239
+ reject: (error: Error) => void
240
+ }
241
+
242
+ let oauthServer: ReturnType<typeof Bun.serve> | undefined
243
+ let pendingOAuth: PendingOAuth | undefined
244
+
245
+ async function startOAuthServer(): Promise<{ port: number; redirectUri: string }> {
246
+ if (oauthServer) {
247
+ return { port: OAUTH_PORT, redirectUri: `http://localhost:${OAUTH_PORT}/auth/callback` }
248
+ }
249
+
250
+ oauthServer = Bun.serve({
251
+ port: OAUTH_PORT,
252
+ fetch(req) {
253
+ const url = new URL(req.url)
254
+
255
+ if (url.pathname === "/auth/callback") {
256
+ const code = url.searchParams.get("code")
257
+ const state = url.searchParams.get("state")
258
+ const error = url.searchParams.get("error")
259
+ const errorDescription = url.searchParams.get("error_description")
260
+
261
+ if (error) {
262
+ const errorMsg = errorDescription || error
263
+ pendingOAuth?.reject(new Error(errorMsg))
264
+ pendingOAuth = undefined
265
+ return new Response(HTML_ERROR(errorMsg), {
266
+ headers: { "Content-Type": "text/html" },
267
+ })
268
+ }
269
+
270
+ if (!code) {
271
+ const errorMsg = "Missing authorization code"
272
+ pendingOAuth?.reject(new Error(errorMsg))
273
+ pendingOAuth = undefined
274
+ return new Response(HTML_ERROR(errorMsg), {
275
+ status: 400,
276
+ headers: { "Content-Type": "text/html" },
277
+ })
278
+ }
279
+
280
+ if (!pendingOAuth || state !== pendingOAuth.state) {
281
+ const errorMsg = "Invalid state - potential CSRF attack"
282
+ pendingOAuth?.reject(new Error(errorMsg))
283
+ pendingOAuth = undefined
284
+ return new Response(HTML_ERROR(errorMsg), {
285
+ status: 400,
286
+ headers: { "Content-Type": "text/html" },
287
+ })
288
+ }
289
+
290
+ const current = pendingOAuth
291
+ pendingOAuth = undefined
292
+
293
+ exchangeCodeForTokens(code, `http://localhost:${OAUTH_PORT}/auth/callback`, current.pkce)
294
+ .then((tokens) => current.resolve(tokens))
295
+ .catch((err) => current.reject(err))
296
+
297
+ return new Response(HTML_SUCCESS, {
298
+ headers: { "Content-Type": "text/html" },
299
+ })
300
+ }
301
+
302
+ if (url.pathname === "/cancel") {
303
+ pendingOAuth?.reject(new Error("Login cancelled"))
304
+ pendingOAuth = undefined
305
+ return new Response("Login cancelled", { status: 200 })
306
+ }
307
+
308
+ return new Response("Not found", { status: 404 })
309
+ },
310
+ })
311
+
312
+ log.info("codex oauth server started", { port: OAUTH_PORT })
313
+ return { port: OAUTH_PORT, redirectUri: `http://localhost:${OAUTH_PORT}/auth/callback` }
314
+ }
315
+
316
+ function stopOAuthServer() {
317
+ if (oauthServer) {
318
+ oauthServer.stop()
319
+ oauthServer = undefined
320
+ log.info("codex oauth server stopped")
321
+ }
322
+ }
323
+
324
+ function waitForOAuthCallback(pkce: PkceCodes, state: string): Promise<TokenResponse> {
325
+ return new Promise((resolve, reject) => {
326
+ const timeout = setTimeout(
327
+ () => {
328
+ if (pendingOAuth) {
329
+ pendingOAuth = undefined
330
+ reject(new Error("OAuth callback timeout - authorization took too long"))
331
+ }
332
+ },
333
+ 5 * 60 * 1000,
334
+ ) // 5 minute timeout
335
+
336
+ pendingOAuth = {
337
+ pkce,
338
+ state,
339
+ resolve: (tokens) => {
340
+ clearTimeout(timeout)
341
+ resolve(tokens)
342
+ },
343
+ reject: (error) => {
344
+ clearTimeout(timeout)
345
+ reject(error)
346
+ },
347
+ }
348
+ })
349
+ }
350
+
351
+ export async function CodexAuthPlugin(input: PluginInput): Promise<Hooks> {
352
+ return {
353
+ auth: {
354
+ provider: "openai",
355
+ async loader(getAuth, provider) {
356
+ const auth = await getAuth()
357
+ if (auth.type !== "oauth") return {}
358
+
359
+ // Filter models to only allowed Codex models for OAuth
360
+ const allowedModels = new Set([
361
+ "gpt-5.1-codex-max",
362
+ "gpt-5.1-codex-mini",
363
+ "gpt-5.2",
364
+ "gpt-5.2-codex",
365
+ "gpt-5.3-codex",
366
+ "gpt-5.1-codex",
367
+ ])
368
+ for (const modelId of Object.keys(provider.models)) {
369
+ if (!allowedModels.has(modelId)) {
370
+ delete provider.models[modelId]
371
+ }
372
+ }
373
+
374
+ if (!provider.models["gpt-5.3-codex"]) {
375
+ const model = {
376
+ id: "gpt-5.3-codex",
377
+ providerID: "openai",
378
+ api: {
379
+ id: "gpt-5.3-codex",
380
+ url: "https://chatgpt.com/backend-api/codex",
381
+ npm: "@ai-sdk/openai",
382
+ },
383
+ name: "GPT-5.3 Codex",
384
+ capabilities: {
385
+ temperature: false,
386
+ reasoning: true,
387
+ attachment: true,
388
+ toolcall: true,
389
+ input: { text: true, audio: false, image: true, video: false, pdf: false },
390
+ output: { text: true, audio: false, image: false, video: false, pdf: false },
391
+ interleaved: false,
392
+ },
393
+ cost: { input: 0, output: 0, cache: { read: 0, write: 0 } },
394
+ limit: { context: 400_000, input: 272_000, output: 128_000 },
395
+ status: "active" as const,
396
+ options: {},
397
+ headers: {},
398
+ release_date: "2026-02-05",
399
+ variants: {} as Record<string, Record<string, any>>,
400
+ family: "gpt-codex",
401
+ }
402
+ model.variants = ProviderTransform.variants(model)
403
+ provider.models["gpt-5.3-codex"] = model
404
+ }
405
+
406
+ // Zero out costs for Codex (included with ChatGPT subscription)
407
+ for (const model of Object.values(provider.models)) {
408
+ model.cost = {
409
+ input: 0,
410
+ output: 0,
411
+ cache: { read: 0, write: 0 },
412
+ }
413
+ }
414
+
415
+ return {
416
+ apiKey: OAUTH_DUMMY_KEY,
417
+ async fetch(requestInput: RequestInfo | URL, init?: RequestInit) {
418
+ // Remove dummy API key authorization header
419
+ if (init?.headers) {
420
+ if (init.headers instanceof Headers) {
421
+ init.headers.delete("authorization")
422
+ init.headers.delete("Authorization")
423
+ } else if (Array.isArray(init.headers)) {
424
+ init.headers = init.headers.filter(([key]) => key.toLowerCase() !== "authorization")
425
+ } else {
426
+ delete init.headers["authorization"]
427
+ delete init.headers["Authorization"]
428
+ }
429
+ }
430
+
431
+ const currentAuth = await getAuth()
432
+ if (currentAuth.type !== "oauth") return fetch(requestInput, init)
433
+
434
+ // Cast to include accountId field
435
+ const authWithAccount = currentAuth as typeof currentAuth & { accountId?: string }
436
+
437
+ // Check if token needs refresh
438
+ if (!currentAuth.access || currentAuth.expires < Date.now()) {
439
+ log.info("refreshing codex access token")
440
+ const tokens = await refreshAccessToken(currentAuth.refresh)
441
+ const newAccountId = extractAccountId(tokens) || authWithAccount.accountId
442
+ await input.client.auth.set({
443
+ path: { id: "openai" },
444
+ body: {
445
+ type: "oauth",
446
+ refresh: tokens.refresh_token,
447
+ access: tokens.access_token,
448
+ expires: Date.now() + (tokens.expires_in ?? 3600) * 1000,
449
+ ...(newAccountId && { accountId: newAccountId }),
450
+ },
451
+ })
452
+ currentAuth.access = tokens.access_token
453
+ authWithAccount.accountId = newAccountId
454
+ }
455
+
456
+ // Build headers
457
+ const headers = new Headers()
458
+ if (init?.headers) {
459
+ if (init.headers instanceof Headers) {
460
+ init.headers.forEach((value, key) => headers.set(key, value))
461
+ } else if (Array.isArray(init.headers)) {
462
+ for (const [key, value] of init.headers) {
463
+ if (value !== undefined) headers.set(key, String(value))
464
+ }
465
+ } else {
466
+ for (const [key, value] of Object.entries(init.headers)) {
467
+ if (value !== undefined) headers.set(key, String(value))
468
+ }
469
+ }
470
+ }
471
+
472
+ // Set authorization header with access token
473
+ headers.set("authorization", `Bearer ${currentAuth.access}`)
474
+
475
+ // Set ChatGPT-Account-Id header for organization subscriptions
476
+ if (authWithAccount.accountId) {
477
+ headers.set("ChatGPT-Account-Id", authWithAccount.accountId)
478
+ }
479
+
480
+ // Rewrite URL to Codex endpoint
481
+ const parsed =
482
+ requestInput instanceof URL
483
+ ? requestInput
484
+ : new URL(typeof requestInput === "string" ? requestInput : requestInput.url)
485
+ const url =
486
+ parsed.pathname.includes("/v1/responses") || parsed.pathname.includes("/chat/completions")
487
+ ? new URL(CODEX_API_ENDPOINT)
488
+ : parsed
489
+
490
+ return fetch(url, {
491
+ ...init,
492
+ headers,
493
+ })
494
+ },
495
+ }
496
+ },
497
+ methods: [
498
+ {
499
+ label: "ChatGPT Pro/Plus (browser)",
500
+ type: "oauth",
501
+ authorize: async () => {
502
+ const { redirectUri } = await startOAuthServer()
503
+ const pkce = await generatePKCE()
504
+ const state = generateState()
505
+ const authUrl = buildAuthorizeUrl(redirectUri, pkce, state)
506
+
507
+ const callbackPromise = waitForOAuthCallback(pkce, state)
508
+
509
+ return {
510
+ url: authUrl,
511
+ instructions: "Complete authorization in your browser. This window will close automatically.",
512
+ method: "auto" as const,
513
+ callback: async () => {
514
+ const tokens = await callbackPromise
515
+ stopOAuthServer()
516
+ const accountId = extractAccountId(tokens)
517
+ return {
518
+ type: "success" as const,
519
+ refresh: tokens.refresh_token,
520
+ access: tokens.access_token,
521
+ expires: Date.now() + (tokens.expires_in ?? 3600) * 1000,
522
+ accountId,
523
+ }
524
+ },
525
+ }
526
+ },
527
+ },
528
+ {
529
+ label: "ChatGPT Pro/Plus (headless)",
530
+ type: "oauth",
531
+ authorize: async () => {
532
+ const deviceResponse = await fetch(`${ISSUER}/api/accounts/deviceauth/usercode`, {
533
+ method: "POST",
534
+ headers: {
535
+ "Content-Type": "application/json",
536
+ "User-Agent": `opencode/${Installation.VERSION}`,
537
+ },
538
+ body: JSON.stringify({ client_id: CLIENT_ID }),
539
+ })
540
+
541
+ if (!deviceResponse.ok) throw new Error("Failed to initiate device authorization")
542
+
543
+ const deviceData = (await deviceResponse.json()) as {
544
+ device_auth_id: string
545
+ user_code: string
546
+ interval: string
547
+ }
548
+ const interval = Math.max(parseInt(deviceData.interval) || 5, 1) * 1000
549
+
550
+ return {
551
+ url: `${ISSUER}/codex/device`,
552
+ instructions: `Enter code: ${deviceData.user_code}`,
553
+ method: "auto" as const,
554
+ async callback() {
555
+ while (true) {
556
+ const response = await fetch(`${ISSUER}/api/accounts/deviceauth/token`, {
557
+ method: "POST",
558
+ headers: {
559
+ "Content-Type": "application/json",
560
+ "User-Agent": `opencode/${Installation.VERSION}`,
561
+ },
562
+ body: JSON.stringify({
563
+ device_auth_id: deviceData.device_auth_id,
564
+ user_code: deviceData.user_code,
565
+ }),
566
+ })
567
+
568
+ if (response.ok) {
569
+ const data = (await response.json()) as {
570
+ authorization_code: string
571
+ code_verifier: string
572
+ }
573
+
574
+ const tokenResponse = await fetch(`${ISSUER}/oauth/token`, {
575
+ method: "POST",
576
+ headers: { "Content-Type": "application/x-www-form-urlencoded" },
577
+ body: new URLSearchParams({
578
+ grant_type: "authorization_code",
579
+ code: data.authorization_code,
580
+ redirect_uri: `${ISSUER}/deviceauth/callback`,
581
+ client_id: CLIENT_ID,
582
+ code_verifier: data.code_verifier,
583
+ }).toString(),
584
+ })
585
+
586
+ if (!tokenResponse.ok) {
587
+ throw new Error(`Token exchange failed: ${tokenResponse.status}`)
588
+ }
589
+
590
+ const tokens: TokenResponse = await tokenResponse.json()
591
+
592
+ return {
593
+ type: "success" as const,
594
+ refresh: tokens.refresh_token,
595
+ access: tokens.access_token,
596
+ expires: Date.now() + (tokens.expires_in ?? 3600) * 1000,
597
+ accountId: extractAccountId(tokens),
598
+ }
599
+ }
600
+
601
+ if (response.status !== 403 && response.status !== 404) {
602
+ return { type: "failed" as const }
603
+ }
604
+
605
+ await Bun.sleep(interval + OAUTH_POLLING_SAFETY_MARGIN_MS)
606
+ }
607
+ },
608
+ }
609
+ },
610
+ },
611
+ {
612
+ label: "Manually enter API Key",
613
+ type: "api",
614
+ },
615
+ ],
616
+ },
617
+ "chat.headers": async (input, output) => {
618
+ if (input.model.providerID !== "openai") return
619
+ output.headers.originator = "opencode"
620
+ output.headers["User-Agent"] = `opencode/${Installation.VERSION} (${os.platform()} ${os.release()}; ${os.arch()})`
621
+ output.headers.session_id = input.sessionID
622
+ },
623
+ }
624
+ }