@code0123/opencode-android-arm64 1.1.54 → 1.1.56

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 (359) hide show
  1. package/bin/opencode +3 -1
  2. package/package.json +1 -80
  3. package/runtime/highlights-eq9cgrbb.scm +604 -0
  4. package/runtime/highlights-ghv9g403.scm +205 -0
  5. package/runtime/highlights-hk7bwhj4.scm +284 -0
  6. package/runtime/highlights-r812a2qc.scm +150 -0
  7. package/runtime/highlights-x6tmsnaa.scm +115 -0
  8. package/runtime/index.js +287124 -0
  9. package/runtime/injections-73j83es3.scm +27 -0
  10. package/runtime/parser.worker.js +4081 -0
  11. package/runtime/tree-sitter-3jzf13jk.wasm +0 -0
  12. package/runtime/tree-sitter-bash-hq5s6fxb.wasm +0 -0
  13. package/runtime/tree-sitter-javascript-nd0q4pe9.wasm +0 -0
  14. package/runtime/tree-sitter-markdown-411r6y9b.wasm +0 -0
  15. package/runtime/tree-sitter-markdown_inline-j5349f42.wasm +0 -0
  16. package/runtime/tree-sitter-typescript-zxjzwt75.wasm +0 -0
  17. package/runtime/tree-sitter-zig-e78zbjpm.wasm +0 -0
  18. package/runtime/worker.js +208338 -0
  19. package/parsers-config.ts +0 -253
  20. package/src/acp/README.md +0 -164
  21. package/src/acp/agent.ts +0 -1676
  22. package/src/acp/session.ts +0 -117
  23. package/src/acp/types.ts +0 -23
  24. package/src/agent/agent.ts +0 -338
  25. package/src/agent/generate.txt +0 -75
  26. package/src/agent/prompt/compaction.txt +0 -12
  27. package/src/agent/prompt/explore.txt +0 -18
  28. package/src/agent/prompt/summary.txt +0 -11
  29. package/src/agent/prompt/title.txt +0 -44
  30. package/src/auth/index.ts +0 -70
  31. package/src/bun/index.ts +0 -137
  32. package/src/bun/registry.ts +0 -48
  33. package/src/bus/bus-event.ts +0 -43
  34. package/src/bus/global.ts +0 -10
  35. package/src/bus/index.ts +0 -105
  36. package/src/cli/bootstrap.ts +0 -17
  37. package/src/cli/cmd/acp.ts +0 -70
  38. package/src/cli/cmd/agent.ts +0 -257
  39. package/src/cli/cmd/auth.ts +0 -400
  40. package/src/cli/cmd/cmd.ts +0 -7
  41. package/src/cli/cmd/debug/agent.ts +0 -167
  42. package/src/cli/cmd/debug/config.ts +0 -16
  43. package/src/cli/cmd/debug/file.ts +0 -97
  44. package/src/cli/cmd/debug/index.ts +0 -48
  45. package/src/cli/cmd/debug/lsp.ts +0 -52
  46. package/src/cli/cmd/debug/ripgrep.ts +0 -87
  47. package/src/cli/cmd/debug/scrap.ts +0 -16
  48. package/src/cli/cmd/debug/skill.ts +0 -16
  49. package/src/cli/cmd/debug/snapshot.ts +0 -52
  50. package/src/cli/cmd/export.ts +0 -88
  51. package/src/cli/cmd/generate.ts +0 -38
  52. package/src/cli/cmd/github.ts +0 -1540
  53. package/src/cli/cmd/import.ts +0 -147
  54. package/src/cli/cmd/mcp.ts +0 -755
  55. package/src/cli/cmd/models.ts +0 -77
  56. package/src/cli/cmd/pr.ts +0 -112
  57. package/src/cli/cmd/run.ts +0 -598
  58. package/src/cli/cmd/serve.ts +0 -20
  59. package/src/cli/cmd/session.ts +0 -135
  60. package/src/cli/cmd/stats.ts +0 -426
  61. package/src/cli/cmd/tui/app.tsx +0 -801
  62. package/src/cli/cmd/tui/attach.ts +0 -52
  63. package/src/cli/cmd/tui/component/border.tsx +0 -21
  64. package/src/cli/cmd/tui/component/dialog-agent.tsx +0 -31
  65. package/src/cli/cmd/tui/component/dialog-command.tsx +0 -148
  66. package/src/cli/cmd/tui/component/dialog-mcp.tsx +0 -86
  67. package/src/cli/cmd/tui/component/dialog-model.tsx +0 -234
  68. package/src/cli/cmd/tui/component/dialog-provider.tsx +0 -266
  69. package/src/cli/cmd/tui/component/dialog-session-list.tsx +0 -108
  70. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +0 -31
  71. package/src/cli/cmd/tui/component/dialog-skill.tsx +0 -36
  72. package/src/cli/cmd/tui/component/dialog-stash.tsx +0 -87
  73. package/src/cli/cmd/tui/component/dialog-status.tsx +0 -177
  74. package/src/cli/cmd/tui/component/dialog-tag.tsx +0 -44
  75. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +0 -50
  76. package/src/cli/cmd/tui/component/logo.tsx +0 -85
  77. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +0 -666
  78. package/src/cli/cmd/tui/component/prompt/frecency.tsx +0 -89
  79. package/src/cli/cmd/tui/component/prompt/history.tsx +0 -108
  80. package/src/cli/cmd/tui/component/prompt/index.tsx +0 -1132
  81. package/src/cli/cmd/tui/component/prompt/stash.tsx +0 -101
  82. package/src/cli/cmd/tui/component/spinner.tsx +0 -24
  83. package/src/cli/cmd/tui/component/textarea-keybindings.ts +0 -73
  84. package/src/cli/cmd/tui/component/tips.tsx +0 -153
  85. package/src/cli/cmd/tui/component/todo-item.tsx +0 -32
  86. package/src/cli/cmd/tui/context/args.tsx +0 -15
  87. package/src/cli/cmd/tui/context/directory.ts +0 -13
  88. package/src/cli/cmd/tui/context/exit.tsx +0 -52
  89. package/src/cli/cmd/tui/context/helper.tsx +0 -25
  90. package/src/cli/cmd/tui/context/keybind.tsx +0 -100
  91. package/src/cli/cmd/tui/context/kv.tsx +0 -52
  92. package/src/cli/cmd/tui/context/local.tsx +0 -409
  93. package/src/cli/cmd/tui/context/prompt.tsx +0 -18
  94. package/src/cli/cmd/tui/context/route.tsx +0 -46
  95. package/src/cli/cmd/tui/context/sdk.tsx +0 -101
  96. package/src/cli/cmd/tui/context/sync.tsx +0 -470
  97. package/src/cli/cmd/tui/context/theme/aura.json +0 -69
  98. package/src/cli/cmd/tui/context/theme/ayu.json +0 -80
  99. package/src/cli/cmd/tui/context/theme/carbonfox.json +0 -248
  100. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +0 -233
  101. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +0 -233
  102. package/src/cli/cmd/tui/context/theme/catppuccin.json +0 -112
  103. package/src/cli/cmd/tui/context/theme/cobalt2.json +0 -228
  104. package/src/cli/cmd/tui/context/theme/cursor.json +0 -249
  105. package/src/cli/cmd/tui/context/theme/dracula.json +0 -219
  106. package/src/cli/cmd/tui/context/theme/everforest.json +0 -241
  107. package/src/cli/cmd/tui/context/theme/flexoki.json +0 -237
  108. package/src/cli/cmd/tui/context/theme/github.json +0 -233
  109. package/src/cli/cmd/tui/context/theme/gruvbox.json +0 -242
  110. package/src/cli/cmd/tui/context/theme/kanagawa.json +0 -77
  111. package/src/cli/cmd/tui/context/theme/lucent-orng.json +0 -237
  112. package/src/cli/cmd/tui/context/theme/material.json +0 -235
  113. package/src/cli/cmd/tui/context/theme/matrix.json +0 -77
  114. package/src/cli/cmd/tui/context/theme/mercury.json +0 -252
  115. package/src/cli/cmd/tui/context/theme/monokai.json +0 -221
  116. package/src/cli/cmd/tui/context/theme/nightowl.json +0 -221
  117. package/src/cli/cmd/tui/context/theme/nord.json +0 -223
  118. package/src/cli/cmd/tui/context/theme/one-dark.json +0 -84
  119. package/src/cli/cmd/tui/context/theme/opencode.json +0 -245
  120. package/src/cli/cmd/tui/context/theme/orng.json +0 -249
  121. package/src/cli/cmd/tui/context/theme/osaka-jade.json +0 -93
  122. package/src/cli/cmd/tui/context/theme/palenight.json +0 -222
  123. package/src/cli/cmd/tui/context/theme/rosepine.json +0 -234
  124. package/src/cli/cmd/tui/context/theme/solarized.json +0 -223
  125. package/src/cli/cmd/tui/context/theme/synthwave84.json +0 -226
  126. package/src/cli/cmd/tui/context/theme/tokyonight.json +0 -243
  127. package/src/cli/cmd/tui/context/theme/vercel.json +0 -245
  128. package/src/cli/cmd/tui/context/theme/vesper.json +0 -218
  129. package/src/cli/cmd/tui/context/theme/zenburn.json +0 -223
  130. package/src/cli/cmd/tui/context/theme.tsx +0 -1152
  131. package/src/cli/cmd/tui/event.ts +0 -48
  132. package/src/cli/cmd/tui/routes/home.tsx +0 -140
  133. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +0 -64
  134. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +0 -109
  135. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +0 -26
  136. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +0 -47
  137. package/src/cli/cmd/tui/routes/session/footer.tsx +0 -91
  138. package/src/cli/cmd/tui/routes/session/header.tsx +0 -142
  139. package/src/cli/cmd/tui/routes/session/index.tsx +0 -2126
  140. package/src/cli/cmd/tui/routes/session/permission.tsx +0 -508
  141. package/src/cli/cmd/tui/routes/session/question.tsx +0 -466
  142. package/src/cli/cmd/tui/routes/session/sidebar.tsx +0 -313
  143. package/src/cli/cmd/tui/thread.ts +0 -175
  144. package/src/cli/cmd/tui/ui/dialog-alert.tsx +0 -68
  145. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +0 -93
  146. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +0 -215
  147. package/src/cli/cmd/tui/ui/dialog-help.tsx +0 -49
  148. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +0 -88
  149. package/src/cli/cmd/tui/ui/dialog-select.tsx +0 -399
  150. package/src/cli/cmd/tui/ui/dialog.tsx +0 -167
  151. package/src/cli/cmd/tui/ui/link.tsx +0 -28
  152. package/src/cli/cmd/tui/ui/spinner.ts +0 -368
  153. package/src/cli/cmd/tui/ui/toast.tsx +0 -100
  154. package/src/cli/cmd/tui/util/clipboard.ts +0 -159
  155. package/src/cli/cmd/tui/util/editor.ts +0 -32
  156. package/src/cli/cmd/tui/util/signal.ts +0 -7
  157. package/src/cli/cmd/tui/util/terminal.ts +0 -114
  158. package/src/cli/cmd/tui/util/transcript.ts +0 -98
  159. package/src/cli/cmd/tui/worker.ts +0 -152
  160. package/src/cli/cmd/uninstall.ts +0 -357
  161. package/src/cli/cmd/upgrade.ts +0 -73
  162. package/src/cli/cmd/web.ts +0 -81
  163. package/src/cli/error.ts +0 -57
  164. package/src/cli/logo.ts +0 -6
  165. package/src/cli/network.ts +0 -60
  166. package/src/cli/ui.ts +0 -113
  167. package/src/cli/upgrade.ts +0 -25
  168. package/src/command/index.ts +0 -150
  169. package/src/command/template/initialize.txt +0 -10
  170. package/src/command/template/review.txt +0 -99
  171. package/src/config/config.ts +0 -1477
  172. package/src/config/markdown.ts +0 -98
  173. package/src/env/index.ts +0 -28
  174. package/src/file/ignore.ts +0 -83
  175. package/src/file/index.ts +0 -583
  176. package/src/file/ripgrep.ts +0 -384
  177. package/src/file/time.ts +0 -69
  178. package/src/file/watcher.ts +0 -148
  179. package/src/flag/flag.ts +0 -93
  180. package/src/format/formatter.ts +0 -366
  181. package/src/format/index.ts +0 -137
  182. package/src/global/index.ts +0 -55
  183. package/src/id/id.ts +0 -83
  184. package/src/ide/index.ts +0 -76
  185. package/src/index.ts +0 -159
  186. package/src/installation/index.ts +0 -246
  187. package/src/lsp/client.ts +0 -252
  188. package/src/lsp/index.ts +0 -485
  189. package/src/lsp/language.ts +0 -119
  190. package/src/lsp/server.ts +0 -2046
  191. package/src/mcp/auth.ts +0 -132
  192. package/src/mcp/index.ts +0 -934
  193. package/src/mcp/oauth-callback.ts +0 -200
  194. package/src/mcp/oauth-provider.ts +0 -154
  195. package/src/patch/index.ts +0 -680
  196. package/src/permission/arity.ts +0 -163
  197. package/src/permission/index.ts +0 -210
  198. package/src/permission/next.ts +0 -280
  199. package/src/plugin/codex.ts +0 -624
  200. package/src/plugin/copilot.ts +0 -327
  201. package/src/plugin/index.ts +0 -138
  202. package/src/project/bootstrap.ts +0 -35
  203. package/src/project/instance.ts +0 -114
  204. package/src/project/project.ts +0 -371
  205. package/src/project/state.ts +0 -70
  206. package/src/project/vcs.ts +0 -76
  207. package/src/provider/auth.ts +0 -147
  208. package/src/provider/models-snapshot.ts +0 -38414
  209. package/src/provider/models.ts +0 -133
  210. package/src/provider/provider.ts +0 -1262
  211. package/src/provider/sdk/copilot/README.md +0 -5
  212. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +0 -164
  213. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +0 -15
  214. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +0 -17
  215. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +0 -64
  216. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +0 -780
  217. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +0 -28
  218. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +0 -44
  219. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +0 -87
  220. package/src/provider/sdk/copilot/copilot-provider.ts +0 -100
  221. package/src/provider/sdk/copilot/index.ts +0 -2
  222. package/src/provider/sdk/copilot/openai-compatible-error.ts +0 -27
  223. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +0 -303
  224. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +0 -22
  225. package/src/provider/sdk/copilot/responses/openai-config.ts +0 -18
  226. package/src/provider/sdk/copilot/responses/openai-error.ts +0 -22
  227. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +0 -207
  228. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +0 -1732
  229. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +0 -177
  230. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +0 -1
  231. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +0 -88
  232. package/src/provider/sdk/copilot/responses/tool/file-search.ts +0 -128
  233. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +0 -115
  234. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +0 -65
  235. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +0 -104
  236. package/src/provider/sdk/copilot/responses/tool/web-search.ts +0 -103
  237. package/src/provider/transform.ts +0 -828
  238. package/src/pty/index.ts +0 -250
  239. package/src/question/index.ts +0 -171
  240. package/src/scheduler/index.ts +0 -61
  241. package/src/server/error.ts +0 -36
  242. package/src/server/event.ts +0 -7
  243. package/src/server/mdns.ts +0 -60
  244. package/src/server/routes/config.ts +0 -92
  245. package/src/server/routes/experimental.ts +0 -208
  246. package/src/server/routes/file.ts +0 -197
  247. package/src/server/routes/global.ts +0 -183
  248. package/src/server/routes/mcp.ts +0 -225
  249. package/src/server/routes/permission.ts +0 -68
  250. package/src/server/routes/project.ts +0 -82
  251. package/src/server/routes/provider.ts +0 -165
  252. package/src/server/routes/pty.ts +0 -169
  253. package/src/server/routes/question.ts +0 -98
  254. package/src/server/routes/session.ts +0 -939
  255. package/src/server/routes/tui.ts +0 -379
  256. package/src/server/server.ts +0 -613
  257. package/src/session/compaction.ts +0 -226
  258. package/src/session/index.ts +0 -517
  259. package/src/session/instruction.ts +0 -197
  260. package/src/session/llm.ts +0 -289
  261. package/src/session/message-v2.ts +0 -802
  262. package/src/session/message.ts +0 -189
  263. package/src/session/processor.ts +0 -407
  264. package/src/session/prompt/anthropic-20250930.txt +0 -166
  265. package/src/session/prompt/anthropic.txt +0 -105
  266. package/src/session/prompt/beast.txt +0 -147
  267. package/src/session/prompt/build-switch.txt +0 -5
  268. package/src/session/prompt/codex_header.txt +0 -79
  269. package/src/session/prompt/copilot-gpt-5.txt +0 -143
  270. package/src/session/prompt/gemini.txt +0 -155
  271. package/src/session/prompt/max-steps.txt +0 -16
  272. package/src/session/prompt/plan-reminder-anthropic.txt +0 -67
  273. package/src/session/prompt/plan.txt +0 -26
  274. package/src/session/prompt/qwen.txt +0 -109
  275. package/src/session/prompt/trinity.txt +0 -97
  276. package/src/session/prompt.ts +0 -1866
  277. package/src/session/retry.ts +0 -97
  278. package/src/session/revert.ts +0 -121
  279. package/src/session/status.ts +0 -76
  280. package/src/session/summary.ts +0 -217
  281. package/src/session/system.ts +0 -54
  282. package/src/session/todo.ts +0 -37
  283. package/src/share/share-next.ts +0 -200
  284. package/src/share/share.ts +0 -92
  285. package/src/shell/shell.ts +0 -67
  286. package/src/skill/discovery.ts +0 -97
  287. package/src/skill/index.ts +0 -1
  288. package/src/skill/skill.ts +0 -188
  289. package/src/snapshot/index.ts +0 -255
  290. package/src/storage/storage.ts +0 -227
  291. package/src/tool/apply_patch.ts +0 -281
  292. package/src/tool/apply_patch.txt +0 -33
  293. package/src/tool/bash.ts +0 -269
  294. package/src/tool/bash.txt +0 -115
  295. package/src/tool/batch.ts +0 -175
  296. package/src/tool/batch.txt +0 -24
  297. package/src/tool/codesearch.ts +0 -132
  298. package/src/tool/codesearch.txt +0 -12
  299. package/src/tool/edit.ts +0 -655
  300. package/src/tool/edit.txt +0 -10
  301. package/src/tool/external-directory.ts +0 -32
  302. package/src/tool/glob.ts +0 -78
  303. package/src/tool/glob.txt +0 -6
  304. package/src/tool/grep.ts +0 -147
  305. package/src/tool/grep.txt +0 -8
  306. package/src/tool/invalid.ts +0 -17
  307. package/src/tool/ls.ts +0 -121
  308. package/src/tool/ls.txt +0 -1
  309. package/src/tool/lsp.ts +0 -96
  310. package/src/tool/lsp.txt +0 -19
  311. package/src/tool/multiedit.ts +0 -46
  312. package/src/tool/multiedit.txt +0 -41
  313. package/src/tool/plan-enter.txt +0 -14
  314. package/src/tool/plan-exit.txt +0 -13
  315. package/src/tool/plan.ts +0 -130
  316. package/src/tool/question.ts +0 -33
  317. package/src/tool/question.txt +0 -10
  318. package/src/tool/read.ts +0 -211
  319. package/src/tool/read.txt +0 -12
  320. package/src/tool/registry.ts +0 -160
  321. package/src/tool/skill.ts +0 -123
  322. package/src/tool/task.ts +0 -165
  323. package/src/tool/task.txt +0 -60
  324. package/src/tool/todo.ts +0 -53
  325. package/src/tool/todoread.txt +0 -14
  326. package/src/tool/todowrite.txt +0 -167
  327. package/src/tool/tool.ts +0 -89
  328. package/src/tool/truncation.ts +0 -106
  329. package/src/tool/webfetch.ts +0 -186
  330. package/src/tool/webfetch.txt +0 -13
  331. package/src/tool/websearch.ts +0 -150
  332. package/src/tool/websearch.txt +0 -14
  333. package/src/tool/write.ts +0 -85
  334. package/src/tool/write.txt +0 -8
  335. package/src/util/abort.ts +0 -35
  336. package/src/util/archive.ts +0 -16
  337. package/src/util/color.ts +0 -19
  338. package/src/util/context.ts +0 -25
  339. package/src/util/defer.ts +0 -12
  340. package/src/util/eventloop.ts +0 -20
  341. package/src/util/filesystem.ts +0 -93
  342. package/src/util/fn.ts +0 -11
  343. package/src/util/format.ts +0 -20
  344. package/src/util/iife.ts +0 -3
  345. package/src/util/keybind.ts +0 -103
  346. package/src/util/lazy.ts +0 -18
  347. package/src/util/locale.ts +0 -81
  348. package/src/util/lock.ts +0 -98
  349. package/src/util/log.ts +0 -180
  350. package/src/util/proxied.ts +0 -3
  351. package/src/util/queue.ts +0 -32
  352. package/src/util/rpc.ts +0 -66
  353. package/src/util/scrap.ts +0 -10
  354. package/src/util/signal.ts +0 -12
  355. package/src/util/timeout.ts +0 -14
  356. package/src/util/token.ts +0 -7
  357. package/src/util/wildcard.ts +0 -56
  358. package/src/worktree/index.ts +0 -574
  359. package/tsconfig.json +0 -10
@@ -1,624 +0,0 @@
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
- }