@stonerzju/opencode 1.2.16-offline.1

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 (496) hide show
  1. package/AGENTS.md +10 -0
  2. package/BUN_SHELL_MIGRATION_PLAN.md +136 -0
  3. package/Dockerfile +18 -0
  4. package/README.md +15 -0
  5. package/bin/opencode +179 -0
  6. package/bunfig.toml +7 -0
  7. package/drizzle.config.ts +10 -0
  8. package/migration/20260127222353_familiar_lady_ursula/migration.sql +90 -0
  9. package/migration/20260127222353_familiar_lady_ursula/snapshot.json +796 -0
  10. package/migration/20260211171708_add_project_commands/migration.sql +1 -0
  11. package/migration/20260211171708_add_project_commands/snapshot.json +806 -0
  12. package/migration/20260213144116_wakeful_the_professor/migration.sql +11 -0
  13. package/migration/20260213144116_wakeful_the_professor/snapshot.json +897 -0
  14. package/migration/20260225215848_workspace/migration.sql +7 -0
  15. package/migration/20260225215848_workspace/snapshot.json +959 -0
  16. package/package.json +140 -0
  17. package/package.json.bak +140 -0
  18. package/parsers-config.ts +254 -0
  19. package/script/build.ts +224 -0
  20. package/script/check-migrations.ts +16 -0
  21. package/script/postinstall.mjs +131 -0
  22. package/script/publish.ts +181 -0
  23. package/script/schema.ts +63 -0
  24. package/script/seed-e2e.ts +50 -0
  25. package/src/acp/README.md +174 -0
  26. package/src/acp/agent.ts +1741 -0
  27. package/src/acp/session.ts +116 -0
  28. package/src/acp/types.ts +23 -0
  29. package/src/agent/agent.ts +339 -0
  30. package/src/agent/generate.txt +75 -0
  31. package/src/agent/prompt/compaction.txt +14 -0
  32. package/src/agent/prompt/explore.txt +18 -0
  33. package/src/agent/prompt/summary.txt +11 -0
  34. package/src/agent/prompt/title.txt +44 -0
  35. package/src/auth/index.ts +68 -0
  36. package/src/bun/index.ts +131 -0
  37. package/src/bun/registry.ts +50 -0
  38. package/src/bus/bus-event.ts +43 -0
  39. package/src/bus/global.ts +10 -0
  40. package/src/bus/index.ts +105 -0
  41. package/src/cli/bootstrap.ts +17 -0
  42. package/src/cli/cmd/acp.ts +70 -0
  43. package/src/cli/cmd/agent.ts +257 -0
  44. package/src/cli/cmd/auth.ts +449 -0
  45. package/src/cli/cmd/cmd.ts +7 -0
  46. package/src/cli/cmd/db.ts +118 -0
  47. package/src/cli/cmd/debug/agent.ts +167 -0
  48. package/src/cli/cmd/debug/config.ts +16 -0
  49. package/src/cli/cmd/debug/file.ts +97 -0
  50. package/src/cli/cmd/debug/index.ts +48 -0
  51. package/src/cli/cmd/debug/lsp.ts +52 -0
  52. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  53. package/src/cli/cmd/debug/scrap.ts +16 -0
  54. package/src/cli/cmd/debug/skill.ts +16 -0
  55. package/src/cli/cmd/debug/snapshot.ts +52 -0
  56. package/src/cli/cmd/export.ts +88 -0
  57. package/src/cli/cmd/generate.ts +38 -0
  58. package/src/cli/cmd/github.ts +1631 -0
  59. package/src/cli/cmd/import.ts +170 -0
  60. package/src/cli/cmd/mcp.ts +754 -0
  61. package/src/cli/cmd/models.ts +77 -0
  62. package/src/cli/cmd/pr.ts +112 -0
  63. package/src/cli/cmd/run.ts +625 -0
  64. package/src/cli/cmd/serve.ts +31 -0
  65. package/src/cli/cmd/session.ts +156 -0
  66. package/src/cli/cmd/stats.ts +410 -0
  67. package/src/cli/cmd/tui/app.tsx +845 -0
  68. package/src/cli/cmd/tui/attach.ts +88 -0
  69. package/src/cli/cmd/tui/component/border.tsx +21 -0
  70. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  71. package/src/cli/cmd/tui/component/dialog-command.tsx +147 -0
  72. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  73. package/src/cli/cmd/tui/component/dialog-model.tsx +165 -0
  74. package/src/cli/cmd/tui/component/dialog-provider.tsx +259 -0
  75. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  76. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  77. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  78. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  79. package/src/cli/cmd/tui/component/dialog-status.tsx +167 -0
  80. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  81. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  82. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  83. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +667 -0
  84. package/src/cli/cmd/tui/component/prompt/frecency.tsx +90 -0
  85. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  86. package/src/cli/cmd/tui/component/prompt/index.tsx +1155 -0
  87. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  88. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  89. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  90. package/src/cli/cmd/tui/component/tips.tsx +152 -0
  91. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  92. package/src/cli/cmd/tui/context/args.tsx +15 -0
  93. package/src/cli/cmd/tui/context/directory.ts +13 -0
  94. package/src/cli/cmd/tui/context/exit.tsx +53 -0
  95. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  96. package/src/cli/cmd/tui/context/keybind.tsx +102 -0
  97. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  98. package/src/cli/cmd/tui/context/local.tsx +406 -0
  99. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  100. package/src/cli/cmd/tui/context/route.tsx +46 -0
  101. package/src/cli/cmd/tui/context/sdk.tsx +101 -0
  102. package/src/cli/cmd/tui/context/sync.tsx +488 -0
  103. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  104. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  105. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  106. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  107. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  108. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  109. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  110. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  111. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  112. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  113. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  114. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  115. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  116. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  117. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  118. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  119. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  120. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  121. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  122. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  123. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  124. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  125. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  126. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  127. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  128. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  129. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  130. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  131. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  132. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  133. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  134. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  135. package/src/cli/cmd/tui/context/theme.tsx +1152 -0
  136. package/src/cli/cmd/tui/context/tui-config.tsx +9 -0
  137. package/src/cli/cmd/tui/event.ts +48 -0
  138. package/src/cli/cmd/tui/routes/home.tsx +145 -0
  139. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  140. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  141. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  142. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  143. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  144. package/src/cli/cmd/tui/routes/session/header.tsx +135 -0
  145. package/src/cli/cmd/tui/routes/session/index.tsx +2219 -0
  146. package/src/cli/cmd/tui/routes/session/permission.tsx +685 -0
  147. package/src/cli/cmd/tui/routes/session/question.tsx +466 -0
  148. package/src/cli/cmd/tui/routes/session/sidebar.tsx +321 -0
  149. package/src/cli/cmd/tui/thread.ts +199 -0
  150. package/src/cli/cmd/tui/ui/dialog-alert.tsx +59 -0
  151. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +85 -0
  152. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +207 -0
  153. package/src/cli/cmd/tui/ui/dialog-help.tsx +40 -0
  154. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +80 -0
  155. package/src/cli/cmd/tui/ui/dialog-select.tsx +401 -0
  156. package/src/cli/cmd/tui/ui/dialog.tsx +182 -0
  157. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  158. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  159. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  160. package/src/cli/cmd/tui/util/clipboard.ts +164 -0
  161. package/src/cli/cmd/tui/util/editor.ts +33 -0
  162. package/src/cli/cmd/tui/util/selection.ts +25 -0
  163. package/src/cli/cmd/tui/util/signal.ts +7 -0
  164. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  165. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  166. package/src/cli/cmd/tui/win32.ts +129 -0
  167. package/src/cli/cmd/tui/worker.ts +157 -0
  168. package/src/cli/cmd/uninstall.ts +356 -0
  169. package/src/cli/cmd/upgrade.ts +73 -0
  170. package/src/cli/cmd/web.ts +81 -0
  171. package/src/cli/cmd/workspace-serve.ts +16 -0
  172. package/src/cli/error.ts +57 -0
  173. package/src/cli/logo.ts +6 -0
  174. package/src/cli/network.ts +60 -0
  175. package/src/cli/ui.ts +116 -0
  176. package/src/cli/upgrade.ts +25 -0
  177. package/src/command/index.ts +150 -0
  178. package/src/command/template/initialize.txt +10 -0
  179. package/src/command/template/review.txt +101 -0
  180. package/src/config/config.ts +1408 -0
  181. package/src/config/markdown.ts +99 -0
  182. package/src/config/migrate-tui-config.ts +155 -0
  183. package/src/config/paths.ts +174 -0
  184. package/src/config/tui-schema.ts +34 -0
  185. package/src/config/tui.ts +118 -0
  186. package/src/control/control.sql.ts +22 -0
  187. package/src/control/index.ts +67 -0
  188. package/src/control-plane/adaptors/index.ts +10 -0
  189. package/src/control-plane/adaptors/types.ts +7 -0
  190. package/src/control-plane/adaptors/worktree.ts +26 -0
  191. package/src/control-plane/config.ts +10 -0
  192. package/src/control-plane/session-proxy-middleware.ts +46 -0
  193. package/src/control-plane/sse.ts +66 -0
  194. package/src/control-plane/workspace-server/routes.ts +33 -0
  195. package/src/control-plane/workspace-server/server.ts +24 -0
  196. package/src/control-plane/workspace.sql.ts +12 -0
  197. package/src/control-plane/workspace.ts +160 -0
  198. package/src/env/index.ts +28 -0
  199. package/src/file/ignore.ts +82 -0
  200. package/src/file/index.ts +646 -0
  201. package/src/file/ripgrep.ts +372 -0
  202. package/src/file/time.ts +71 -0
  203. package/src/file/watcher.ts +128 -0
  204. package/src/flag/flag.ts +109 -0
  205. package/src/format/formatter.ts +395 -0
  206. package/src/format/index.ts +140 -0
  207. package/src/global/index.ts +54 -0
  208. package/src/id/id.ts +84 -0
  209. package/src/ide/index.ts +76 -0
  210. package/src/index.ts +210 -0
  211. package/src/installation/index.ts +266 -0
  212. package/src/lsp/client.ts +251 -0
  213. package/src/lsp/index.ts +485 -0
  214. package/src/lsp/language.ts +120 -0
  215. package/src/lsp/server.ts +2142 -0
  216. package/src/mcp/auth.ts +130 -0
  217. package/src/mcp/index.ts +937 -0
  218. package/src/mcp/oauth-callback.ts +200 -0
  219. package/src/mcp/oauth-provider.ts +176 -0
  220. package/src/patch/index.ts +680 -0
  221. package/src/permission/arity.ts +163 -0
  222. package/src/permission/index.ts +210 -0
  223. package/src/permission/next.ts +286 -0
  224. package/src/plugin/codex.ts +624 -0
  225. package/src/plugin/copilot.ts +327 -0
  226. package/src/plugin/index.ts +143 -0
  227. package/src/project/bootstrap.ts +33 -0
  228. package/src/project/instance.ts +114 -0
  229. package/src/project/project.sql.ts +15 -0
  230. package/src/project/project.ts +441 -0
  231. package/src/project/state.ts +70 -0
  232. package/src/project/vcs.ts +76 -0
  233. package/src/provider/auth.ts +147 -0
  234. package/src/provider/error.ts +189 -0
  235. package/src/provider/models.ts +146 -0
  236. package/src/provider/provider.ts +1338 -0
  237. package/src/provider/sdk/copilot/README.md +5 -0
  238. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +164 -0
  239. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  240. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +17 -0
  241. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  242. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +780 -0
  243. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  244. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  245. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +87 -0
  246. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  247. package/src/provider/sdk/copilot/index.ts +2 -0
  248. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  249. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +303 -0
  250. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  251. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  252. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  253. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +207 -0
  254. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1732 -0
  255. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +177 -0
  256. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  257. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +88 -0
  258. package/src/provider/sdk/copilot/responses/tool/file-search.ts +128 -0
  259. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +115 -0
  260. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +65 -0
  261. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +104 -0
  262. package/src/provider/sdk/copilot/responses/tool/web-search.ts +103 -0
  263. package/src/provider/transform.ts +955 -0
  264. package/src/pty/index.ts +324 -0
  265. package/src/question/index.ts +171 -0
  266. package/src/scheduler/index.ts +61 -0
  267. package/src/server/error.ts +36 -0
  268. package/src/server/event.ts +7 -0
  269. package/src/server/mdns.ts +60 -0
  270. package/src/server/routes/config.ts +92 -0
  271. package/src/server/routes/experimental.ts +270 -0
  272. package/src/server/routes/file.ts +197 -0
  273. package/src/server/routes/global.ts +185 -0
  274. package/src/server/routes/mcp.ts +225 -0
  275. package/src/server/routes/permission.ts +68 -0
  276. package/src/server/routes/project.ts +82 -0
  277. package/src/server/routes/provider.ts +165 -0
  278. package/src/server/routes/pty.ts +200 -0
  279. package/src/server/routes/question.ts +98 -0
  280. package/src/server/routes/session.ts +974 -0
  281. package/src/server/routes/tui.ts +379 -0
  282. package/src/server/routes/workspace.ts +104 -0
  283. package/src/server/server.ts +623 -0
  284. package/src/session/compaction.ts +261 -0
  285. package/src/session/index.ts +877 -0
  286. package/src/session/instruction.ts +192 -0
  287. package/src/session/llm.ts +279 -0
  288. package/src/session/message-v2.ts +899 -0
  289. package/src/session/message.ts +189 -0
  290. package/src/session/processor.ts +421 -0
  291. package/src/session/prompt/anthropic-20250930.txt +166 -0
  292. package/src/session/prompt/anthropic.txt +105 -0
  293. package/src/session/prompt/beast.txt +147 -0
  294. package/src/session/prompt/build-switch.txt +5 -0
  295. package/src/session/prompt/codex_header.txt +79 -0
  296. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  297. package/src/session/prompt/gemini.txt +155 -0
  298. package/src/session/prompt/max-steps.txt +16 -0
  299. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  300. package/src/session/prompt/plan.txt +26 -0
  301. package/src/session/prompt/qwen.txt +109 -0
  302. package/src/session/prompt/trinity.txt +97 -0
  303. package/src/session/prompt.ts +1959 -0
  304. package/src/session/retry.ts +101 -0
  305. package/src/session/revert.ts +138 -0
  306. package/src/session/session.sql.ts +88 -0
  307. package/src/session/status.ts +76 -0
  308. package/src/session/summary.ts +161 -0
  309. package/src/session/system.ts +54 -0
  310. package/src/session/todo.ts +56 -0
  311. package/src/share/share-next.ts +210 -0
  312. package/src/share/share.sql.ts +13 -0
  313. package/src/shell/shell.ts +68 -0
  314. package/src/skill/discovery.ts +98 -0
  315. package/src/skill/index.ts +1 -0
  316. package/src/skill/skill.ts +189 -0
  317. package/src/snapshot/index.ts +297 -0
  318. package/src/sql.d.ts +4 -0
  319. package/src/storage/db.ts +155 -0
  320. package/src/storage/json-migration.ts +425 -0
  321. package/src/storage/schema.sql.ts +10 -0
  322. package/src/storage/schema.ts +5 -0
  323. package/src/storage/storage.ts +220 -0
  324. package/src/tool/apply_patch.ts +281 -0
  325. package/src/tool/apply_patch.txt +33 -0
  326. package/src/tool/bash.ts +274 -0
  327. package/src/tool/bash.txt +115 -0
  328. package/src/tool/batch.ts +181 -0
  329. package/src/tool/batch.txt +24 -0
  330. package/src/tool/codesearch.ts +132 -0
  331. package/src/tool/codesearch.txt +12 -0
  332. package/src/tool/edit.ts +654 -0
  333. package/src/tool/edit.txt +10 -0
  334. package/src/tool/external-directory.ts +32 -0
  335. package/src/tool/glob.ts +78 -0
  336. package/src/tool/glob.txt +6 -0
  337. package/src/tool/grep.ts +156 -0
  338. package/src/tool/grep.txt +8 -0
  339. package/src/tool/invalid.ts +17 -0
  340. package/src/tool/ls.ts +121 -0
  341. package/src/tool/ls.txt +1 -0
  342. package/src/tool/lsp.ts +97 -0
  343. package/src/tool/lsp.txt +19 -0
  344. package/src/tool/multiedit.ts +46 -0
  345. package/src/tool/multiedit.txt +41 -0
  346. package/src/tool/plan-enter.txt +14 -0
  347. package/src/tool/plan-exit.txt +13 -0
  348. package/src/tool/plan.ts +131 -0
  349. package/src/tool/question.ts +33 -0
  350. package/src/tool/question.txt +10 -0
  351. package/src/tool/read.ts +293 -0
  352. package/src/tool/read.txt +14 -0
  353. package/src/tool/registry.ts +173 -0
  354. package/src/tool/skill.ts +123 -0
  355. package/src/tool/task.ts +165 -0
  356. package/src/tool/task.txt +60 -0
  357. package/src/tool/todo.ts +53 -0
  358. package/src/tool/todoread.txt +14 -0
  359. package/src/tool/todowrite.txt +167 -0
  360. package/src/tool/tool.ts +89 -0
  361. package/src/tool/truncation.ts +107 -0
  362. package/src/tool/webfetch.ts +206 -0
  363. package/src/tool/webfetch.txt +13 -0
  364. package/src/tool/websearch.ts +150 -0
  365. package/src/tool/websearch.txt +14 -0
  366. package/src/tool/write.ts +84 -0
  367. package/src/tool/write.txt +8 -0
  368. package/src/util/abort.ts +35 -0
  369. package/src/util/archive.ts +16 -0
  370. package/src/util/color.ts +19 -0
  371. package/src/util/context.ts +25 -0
  372. package/src/util/defer.ts +12 -0
  373. package/src/util/eventloop.ts +20 -0
  374. package/src/util/filesystem.ts +189 -0
  375. package/src/util/fn.ts +11 -0
  376. package/src/util/format.ts +20 -0
  377. package/src/util/git.ts +35 -0
  378. package/src/util/glob.ts +34 -0
  379. package/src/util/iife.ts +3 -0
  380. package/src/util/keybind.ts +103 -0
  381. package/src/util/lazy.ts +23 -0
  382. package/src/util/locale.ts +81 -0
  383. package/src/util/lock.ts +98 -0
  384. package/src/util/log.ts +182 -0
  385. package/src/util/process.ts +126 -0
  386. package/src/util/proxied.ts +3 -0
  387. package/src/util/queue.ts +32 -0
  388. package/src/util/rpc.ts +66 -0
  389. package/src/util/scrap.ts +10 -0
  390. package/src/util/signal.ts +12 -0
  391. package/src/util/timeout.ts +14 -0
  392. package/src/util/token.ts +7 -0
  393. package/src/util/wildcard.ts +59 -0
  394. package/src/worktree/index.ts +643 -0
  395. package/sst-env.d.ts +10 -0
  396. package/test/AGENTS.md +81 -0
  397. package/test/acp/agent-interface.test.ts +51 -0
  398. package/test/acp/event-subscription.test.ts +683 -0
  399. package/test/agent/agent.test.ts +689 -0
  400. package/test/bun.test.ts +53 -0
  401. package/test/cli/github-action.test.ts +197 -0
  402. package/test/cli/github-remote.test.ts +80 -0
  403. package/test/cli/import.test.ts +38 -0
  404. package/test/cli/plugin-auth-picker.test.ts +120 -0
  405. package/test/cli/tui/transcript.test.ts +322 -0
  406. package/test/config/agent-color.test.ts +71 -0
  407. package/test/config/config.test.ts +1886 -0
  408. package/test/config/fixtures/empty-frontmatter.md +4 -0
  409. package/test/config/fixtures/frontmatter.md +28 -0
  410. package/test/config/fixtures/markdown-header.md +11 -0
  411. package/test/config/fixtures/no-frontmatter.md +1 -0
  412. package/test/config/fixtures/weird-model-id.md +13 -0
  413. package/test/config/markdown.test.ts +228 -0
  414. package/test/config/tui.test.ts +510 -0
  415. package/test/control-plane/session-proxy-middleware.test.ts +147 -0
  416. package/test/control-plane/sse.test.ts +56 -0
  417. package/test/control-plane/workspace-server-sse.test.ts +65 -0
  418. package/test/control-plane/workspace-sync.test.ts +97 -0
  419. package/test/file/ignore.test.ts +10 -0
  420. package/test/file/index.test.ts +394 -0
  421. package/test/file/path-traversal.test.ts +198 -0
  422. package/test/file/ripgrep.test.ts +39 -0
  423. package/test/file/time.test.ts +361 -0
  424. package/test/fixture/db.ts +11 -0
  425. package/test/fixture/fixture.ts +45 -0
  426. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  427. package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
  428. package/test/fixture/skills/agents-sdk/references/callable.md +92 -0
  429. package/test/fixture/skills/cloudflare/SKILL.md +211 -0
  430. package/test/fixture/skills/index.json +6 -0
  431. package/test/ide/ide.test.ts +82 -0
  432. package/test/keybind.test.ts +421 -0
  433. package/test/lsp/client.test.ts +95 -0
  434. package/test/mcp/headers.test.ts +153 -0
  435. package/test/mcp/oauth-browser.test.ts +249 -0
  436. package/test/memory/abort-leak.test.ts +136 -0
  437. package/test/patch/patch.test.ts +348 -0
  438. package/test/permission/arity.test.ts +33 -0
  439. package/test/permission/next.test.ts +689 -0
  440. package/test/permission-task.test.ts +319 -0
  441. package/test/plugin/auth-override.test.ts +44 -0
  442. package/test/plugin/codex.test.ts +123 -0
  443. package/test/preload.ts +80 -0
  444. package/test/project/project.test.ts +348 -0
  445. package/test/project/worktree-remove.test.ts +65 -0
  446. package/test/provider/amazon-bedrock.test.ts +446 -0
  447. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  448. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  449. package/test/provider/gitlab-duo.test.ts +262 -0
  450. package/test/provider/provider.test.ts +2220 -0
  451. package/test/provider/transform.test.ts +2353 -0
  452. package/test/pty/pty-output-isolation.test.ts +140 -0
  453. package/test/question/question.test.ts +300 -0
  454. package/test/scheduler.test.ts +73 -0
  455. package/test/server/global-session-list.test.ts +89 -0
  456. package/test/server/session-list.test.ts +90 -0
  457. package/test/server/session-select.test.ts +78 -0
  458. package/test/session/compaction.test.ts +423 -0
  459. package/test/session/instruction.test.ts +170 -0
  460. package/test/session/llm.test.ts +667 -0
  461. package/test/session/message-v2.test.ts +924 -0
  462. package/test/session/prompt.test.ts +211 -0
  463. package/test/session/retry.test.ts +188 -0
  464. package/test/session/revert-compact.test.ts +285 -0
  465. package/test/session/session.test.ts +71 -0
  466. package/test/session/structured-output-integration.test.ts +233 -0
  467. package/test/session/structured-output.test.ts +385 -0
  468. package/test/skill/discovery.test.ts +110 -0
  469. package/test/skill/skill.test.ts +388 -0
  470. package/test/snapshot/snapshot.test.ts +1180 -0
  471. package/test/storage/json-migration.test.ts +846 -0
  472. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  473. package/test/tool/apply_patch.test.ts +566 -0
  474. package/test/tool/bash.test.ts +402 -0
  475. package/test/tool/edit.test.ts +496 -0
  476. package/test/tool/external-directory.test.ts +127 -0
  477. package/test/tool/fixtures/large-image.png +0 -0
  478. package/test/tool/fixtures/models-api.json +38413 -0
  479. package/test/tool/grep.test.ts +110 -0
  480. package/test/tool/question.test.ts +107 -0
  481. package/test/tool/read.test.ts +504 -0
  482. package/test/tool/registry.test.ts +122 -0
  483. package/test/tool/skill.test.ts +112 -0
  484. package/test/tool/truncation.test.ts +160 -0
  485. package/test/tool/webfetch.test.ts +100 -0
  486. package/test/tool/write.test.ts +348 -0
  487. package/test/util/filesystem.test.ts +443 -0
  488. package/test/util/format.test.ts +59 -0
  489. package/test/util/glob.test.ts +164 -0
  490. package/test/util/iife.test.ts +36 -0
  491. package/test/util/lazy.test.ts +50 -0
  492. package/test/util/lock.test.ts +72 -0
  493. package/test/util/process.test.ts +59 -0
  494. package/test/util/timeout.test.ts +21 -0
  495. package/test/util/wildcard.test.ts +90 -0
  496. package/tsconfig.json +16 -0
@@ -0,0 +1,257 @@
1
+ import { cmd } from "./cmd"
2
+ import * as prompts from "@clack/prompts"
3
+ import { UI } from "../ui"
4
+ import { Global } from "../../global"
5
+ import { Agent } from "../../agent/agent"
6
+ import { Provider } from "../../provider/provider"
7
+ import path from "path"
8
+ import fs from "fs/promises"
9
+ import { Filesystem } from "../../util/filesystem"
10
+ import matter from "gray-matter"
11
+ import { Instance } from "../../project/instance"
12
+ import { EOL } from "os"
13
+ import type { Argv } from "yargs"
14
+
15
+ type AgentMode = "all" | "primary" | "subagent"
16
+
17
+ const AVAILABLE_TOOLS = [
18
+ "bash",
19
+ "read",
20
+ "write",
21
+ "edit",
22
+ "list",
23
+ "glob",
24
+ "grep",
25
+ "webfetch",
26
+ "task",
27
+ "todowrite",
28
+ "todoread",
29
+ ]
30
+
31
+ const AgentCreateCommand = cmd({
32
+ command: "create",
33
+ describe: "create a new agent",
34
+ builder: (yargs: Argv) =>
35
+ yargs
36
+ .option("path", {
37
+ type: "string",
38
+ describe: "directory path to generate the agent file",
39
+ })
40
+ .option("description", {
41
+ type: "string",
42
+ describe: "what the agent should do",
43
+ })
44
+ .option("mode", {
45
+ type: "string",
46
+ describe: "agent mode",
47
+ choices: ["all", "primary", "subagent"] as const,
48
+ })
49
+ .option("tools", {
50
+ type: "string",
51
+ describe: `comma-separated list of tools to enable (default: all). Available: "${AVAILABLE_TOOLS.join(", ")}"`,
52
+ })
53
+ .option("model", {
54
+ type: "string",
55
+ alias: ["m"],
56
+ describe: "model to use in the format of provider/model",
57
+ }),
58
+ async handler(args) {
59
+ await Instance.provide({
60
+ directory: process.cwd(),
61
+ async fn() {
62
+ const cliPath = args.path
63
+ const cliDescription = args.description
64
+ const cliMode = args.mode as AgentMode | undefined
65
+ const cliTools = args.tools
66
+
67
+ const isFullyNonInteractive = cliPath && cliDescription && cliMode && cliTools !== undefined
68
+
69
+ if (!isFullyNonInteractive) {
70
+ UI.empty()
71
+ prompts.intro("Create agent")
72
+ }
73
+
74
+ const project = Instance.project
75
+
76
+ // Determine scope/path
77
+ let targetPath: string
78
+ if (cliPath) {
79
+ targetPath = path.join(cliPath, "agent")
80
+ } else {
81
+ let scope: "global" | "project" = "global"
82
+ if (project.vcs === "git") {
83
+ const scopeResult = await prompts.select({
84
+ message: "Location",
85
+ options: [
86
+ {
87
+ label: "Current project",
88
+ value: "project" as const,
89
+ hint: Instance.worktree,
90
+ },
91
+ {
92
+ label: "Global",
93
+ value: "global" as const,
94
+ hint: Global.Path.config,
95
+ },
96
+ ],
97
+ })
98
+ if (prompts.isCancel(scopeResult)) throw new UI.CancelledError()
99
+ scope = scopeResult
100
+ }
101
+ targetPath = path.join(
102
+ scope === "global" ? Global.Path.config : path.join(Instance.worktree, ".opencode"),
103
+ "agent",
104
+ )
105
+ }
106
+
107
+ // Get description
108
+ let description: string
109
+ if (cliDescription) {
110
+ description = cliDescription
111
+ } else {
112
+ const query = await prompts.text({
113
+ message: "Description",
114
+ placeholder: "What should this agent do?",
115
+ validate: (x) => (x && x.length > 0 ? undefined : "Required"),
116
+ })
117
+ if (prompts.isCancel(query)) throw new UI.CancelledError()
118
+ description = query
119
+ }
120
+
121
+ // Generate agent
122
+ const spinner = prompts.spinner()
123
+ spinner.start("Generating agent configuration...")
124
+ const model = args.model ? Provider.parseModel(args.model) : undefined
125
+ const generated = await Agent.generate({ description, model }).catch((error) => {
126
+ spinner.stop(`LLM failed to generate agent: ${error.message}`, 1)
127
+ if (isFullyNonInteractive) process.exit(1)
128
+ throw new UI.CancelledError()
129
+ })
130
+ spinner.stop(`Agent ${generated.identifier} generated`)
131
+
132
+ // Select tools
133
+ let selectedTools: string[]
134
+ if (cliTools !== undefined) {
135
+ selectedTools = cliTools ? cliTools.split(",").map((t) => t.trim()) : AVAILABLE_TOOLS
136
+ } else {
137
+ const result = await prompts.multiselect({
138
+ message: "Select tools to enable (Space to toggle)",
139
+ options: AVAILABLE_TOOLS.map((tool) => ({
140
+ label: tool,
141
+ value: tool,
142
+ })),
143
+ initialValues: AVAILABLE_TOOLS,
144
+ })
145
+ if (prompts.isCancel(result)) throw new UI.CancelledError()
146
+ selectedTools = result
147
+ }
148
+
149
+ // Get mode
150
+ let mode: AgentMode
151
+ if (cliMode) {
152
+ mode = cliMode
153
+ } else {
154
+ const modeResult = await prompts.select({
155
+ message: "Agent mode",
156
+ options: [
157
+ {
158
+ label: "All",
159
+ value: "all" as const,
160
+ hint: "Can function in both primary and subagent roles",
161
+ },
162
+ {
163
+ label: "Primary",
164
+ value: "primary" as const,
165
+ hint: "Acts as a primary/main agent",
166
+ },
167
+ {
168
+ label: "Subagent",
169
+ value: "subagent" as const,
170
+ hint: "Can be used as a subagent by other agents",
171
+ },
172
+ ],
173
+ initialValue: "all" as const,
174
+ })
175
+ if (prompts.isCancel(modeResult)) throw new UI.CancelledError()
176
+ mode = modeResult
177
+ }
178
+
179
+ // Build tools config
180
+ const tools: Record<string, boolean> = {}
181
+ for (const tool of AVAILABLE_TOOLS) {
182
+ if (!selectedTools.includes(tool)) {
183
+ tools[tool] = false
184
+ }
185
+ }
186
+
187
+ // Build frontmatter
188
+ const frontmatter: {
189
+ description: string
190
+ mode: AgentMode
191
+ tools?: Record<string, boolean>
192
+ } = {
193
+ description: generated.whenToUse,
194
+ mode,
195
+ }
196
+ if (Object.keys(tools).length > 0) {
197
+ frontmatter.tools = tools
198
+ }
199
+
200
+ // Write file
201
+ const content = matter.stringify(generated.systemPrompt, frontmatter)
202
+ const filePath = path.join(targetPath, `${generated.identifier}.md`)
203
+
204
+ await fs.mkdir(targetPath, { recursive: true })
205
+
206
+ if (await Filesystem.exists(filePath)) {
207
+ if (isFullyNonInteractive) {
208
+ console.error(`Error: Agent file already exists: ${filePath}`)
209
+ process.exit(1)
210
+ }
211
+ prompts.log.error(`Agent file already exists: ${filePath}`)
212
+ throw new UI.CancelledError()
213
+ }
214
+
215
+ await Filesystem.write(filePath, content)
216
+
217
+ if (isFullyNonInteractive) {
218
+ console.log(filePath)
219
+ } else {
220
+ prompts.log.success(`Agent created: ${filePath}`)
221
+ prompts.outro("Done")
222
+ }
223
+ },
224
+ })
225
+ },
226
+ })
227
+
228
+ const AgentListCommand = cmd({
229
+ command: "list",
230
+ describe: "list all available agents",
231
+ async handler() {
232
+ await Instance.provide({
233
+ directory: process.cwd(),
234
+ async fn() {
235
+ const agents = await Agent.list()
236
+ const sortedAgents = agents.sort((a, b) => {
237
+ if (a.native !== b.native) {
238
+ return a.native ? -1 : 1
239
+ }
240
+ return a.name.localeCompare(b.name)
241
+ })
242
+
243
+ for (const agent of sortedAgents) {
244
+ process.stdout.write(`${agent.name} (${agent.mode})` + EOL)
245
+ process.stdout.write(` ${JSON.stringify(agent.permission, null, 2)}` + EOL)
246
+ }
247
+ },
248
+ })
249
+ },
250
+ })
251
+
252
+ export const AgentCommand = cmd({
253
+ command: "agent",
254
+ describe: "manage agents",
255
+ builder: (yargs) => yargs.command(AgentCreateCommand).command(AgentListCommand).demandCommand(),
256
+ async handler() {},
257
+ })
@@ -0,0 +1,449 @@
1
+ import { Auth } from "../../auth"
2
+ import { cmd } from "./cmd"
3
+ import * as prompts from "@clack/prompts"
4
+ import { UI } from "../ui"
5
+ import { ModelsDev } from "../../provider/models"
6
+ import { map, pipe, sortBy, values } from "remeda"
7
+ import path from "path"
8
+ import os from "os"
9
+ import { Config } from "../../config/config"
10
+ import { Global } from "../../global"
11
+ import { Plugin } from "../../plugin"
12
+ import { Instance } from "../../project/instance"
13
+ import type { Hooks } from "@opencode-ai/plugin"
14
+ import { Process } from "../../util/process"
15
+ import { text } from "node:stream/consumers"
16
+
17
+ type PluginAuth = NonNullable<Hooks["auth"]>
18
+
19
+ /**
20
+ * Handle plugin-based authentication flow.
21
+ * Returns true if auth was handled, false if it should fall through to default handling.
22
+ */
23
+ async function handlePluginAuth(plugin: { auth: PluginAuth }, provider: string): Promise<boolean> {
24
+ let index = 0
25
+ if (plugin.auth.methods.length > 1) {
26
+ const method = await prompts.select({
27
+ message: "Login method",
28
+ options: [
29
+ ...plugin.auth.methods.map((x, index) => ({
30
+ label: x.label,
31
+ value: index.toString(),
32
+ })),
33
+ ],
34
+ })
35
+ if (prompts.isCancel(method)) throw new UI.CancelledError()
36
+ index = parseInt(method)
37
+ }
38
+ const method = plugin.auth.methods[index]
39
+
40
+ // Handle prompts for all auth types
41
+ await Bun.sleep(10)
42
+ const inputs: Record<string, string> = {}
43
+ if (method.prompts) {
44
+ for (const prompt of method.prompts) {
45
+ if (prompt.condition && !prompt.condition(inputs)) {
46
+ continue
47
+ }
48
+ if (prompt.type === "select") {
49
+ const value = await prompts.select({
50
+ message: prompt.message,
51
+ options: prompt.options,
52
+ })
53
+ if (prompts.isCancel(value)) throw new UI.CancelledError()
54
+ inputs[prompt.key] = value
55
+ } else {
56
+ const value = await prompts.text({
57
+ message: prompt.message,
58
+ placeholder: prompt.placeholder,
59
+ validate: prompt.validate ? (v) => prompt.validate!(v ?? "") : undefined,
60
+ })
61
+ if (prompts.isCancel(value)) throw new UI.CancelledError()
62
+ inputs[prompt.key] = value
63
+ }
64
+ }
65
+ }
66
+
67
+ if (method.type === "oauth") {
68
+ const authorize = await method.authorize(inputs)
69
+
70
+ if (authorize.url) {
71
+ prompts.log.info("Go to: " + authorize.url)
72
+ }
73
+
74
+ if (authorize.method === "auto") {
75
+ if (authorize.instructions) {
76
+ prompts.log.info(authorize.instructions)
77
+ }
78
+ const spinner = prompts.spinner()
79
+ spinner.start("Waiting for authorization...")
80
+ const result = await authorize.callback()
81
+ if (result.type === "failed") {
82
+ spinner.stop("Failed to authorize", 1)
83
+ }
84
+ if (result.type === "success") {
85
+ const saveProvider = result.provider ?? provider
86
+ if ("refresh" in result) {
87
+ const { type: _, provider: __, refresh, access, expires, ...extraFields } = result
88
+ await Auth.set(saveProvider, {
89
+ type: "oauth",
90
+ refresh,
91
+ access,
92
+ expires,
93
+ ...extraFields,
94
+ })
95
+ }
96
+ if ("key" in result) {
97
+ await Auth.set(saveProvider, {
98
+ type: "api",
99
+ key: result.key,
100
+ })
101
+ }
102
+ spinner.stop("Login successful")
103
+ }
104
+ }
105
+
106
+ if (authorize.method === "code") {
107
+ const code = await prompts.text({
108
+ message: "Paste the authorization code here: ",
109
+ validate: (x) => (x && x.length > 0 ? undefined : "Required"),
110
+ })
111
+ if (prompts.isCancel(code)) throw new UI.CancelledError()
112
+ const result = await authorize.callback(code)
113
+ if (result.type === "failed") {
114
+ prompts.log.error("Failed to authorize")
115
+ }
116
+ if (result.type === "success") {
117
+ const saveProvider = result.provider ?? provider
118
+ if ("refresh" in result) {
119
+ const { type: _, provider: __, refresh, access, expires, ...extraFields } = result
120
+ await Auth.set(saveProvider, {
121
+ type: "oauth",
122
+ refresh,
123
+ access,
124
+ expires,
125
+ ...extraFields,
126
+ })
127
+ }
128
+ if ("key" in result) {
129
+ await Auth.set(saveProvider, {
130
+ type: "api",
131
+ key: result.key,
132
+ })
133
+ }
134
+ prompts.log.success("Login successful")
135
+ }
136
+ }
137
+
138
+ prompts.outro("Done")
139
+ return true
140
+ }
141
+
142
+ if (method.type === "api") {
143
+ if (method.authorize) {
144
+ const result = await method.authorize(inputs)
145
+ if (result.type === "failed") {
146
+ prompts.log.error("Failed to authorize")
147
+ }
148
+ if (result.type === "success") {
149
+ const saveProvider = result.provider ?? provider
150
+ await Auth.set(saveProvider, {
151
+ type: "api",
152
+ key: result.key,
153
+ })
154
+ prompts.log.success("Login successful")
155
+ }
156
+ prompts.outro("Done")
157
+ return true
158
+ }
159
+ }
160
+
161
+ return false
162
+ }
163
+
164
+ /**
165
+ * Build a deduplicated list of plugin-registered auth providers that are not
166
+ * already present in models.dev, respecting enabled/disabled provider lists.
167
+ * Pure function with no side effects; safe to test without mocking.
168
+ */
169
+ export function resolvePluginProviders(input: {
170
+ hooks: Hooks[]
171
+ existingProviders: Record<string, unknown>
172
+ disabled: Set<string>
173
+ enabled?: Set<string>
174
+ providerNames: Record<string, string | undefined>
175
+ }): Array<{ id: string; name: string }> {
176
+ const seen = new Set<string>()
177
+ const result: Array<{ id: string; name: string }> = []
178
+
179
+ for (const hook of input.hooks) {
180
+ if (!hook.auth) continue
181
+ const id = hook.auth.provider
182
+ if (seen.has(id)) continue
183
+ seen.add(id)
184
+ if (Object.hasOwn(input.existingProviders, id)) continue
185
+ if (input.disabled.has(id)) continue
186
+ if (input.enabled && !input.enabled.has(id)) continue
187
+ result.push({
188
+ id,
189
+ name: input.providerNames[id] ?? id,
190
+ })
191
+ }
192
+
193
+ return result
194
+ }
195
+
196
+ export const AuthCommand = cmd({
197
+ command: "auth",
198
+ describe: "manage credentials",
199
+ builder: (yargs) =>
200
+ yargs.command(AuthLoginCommand).command(AuthLogoutCommand).command(AuthListCommand).demandCommand(),
201
+ async handler() {},
202
+ })
203
+
204
+ export const AuthListCommand = cmd({
205
+ command: "list",
206
+ aliases: ["ls"],
207
+ describe: "list providers",
208
+ async handler() {
209
+ UI.empty()
210
+ const authPath = path.join(Global.Path.data, "auth.json")
211
+ const homedir = os.homedir()
212
+ const displayPath = authPath.startsWith(homedir) ? authPath.replace(homedir, "~") : authPath
213
+ prompts.intro(`Credentials ${UI.Style.TEXT_DIM}${displayPath}`)
214
+ const results = Object.entries(await Auth.all())
215
+ const database = await ModelsDev.get()
216
+
217
+ for (const [providerID, result] of results) {
218
+ const name = database[providerID]?.name || providerID
219
+ prompts.log.info(`${name} ${UI.Style.TEXT_DIM}${result.type}`)
220
+ }
221
+
222
+ prompts.outro(`${results.length} credentials`)
223
+
224
+ // Environment variables section
225
+ const activeEnvVars: Array<{ provider: string; envVar: string }> = []
226
+
227
+ for (const [providerID, provider] of Object.entries(database)) {
228
+ for (const envVar of provider.env) {
229
+ if (process.env[envVar]) {
230
+ activeEnvVars.push({
231
+ provider: provider.name || providerID,
232
+ envVar,
233
+ })
234
+ }
235
+ }
236
+ }
237
+
238
+ if (activeEnvVars.length > 0) {
239
+ UI.empty()
240
+ prompts.intro("Environment")
241
+
242
+ for (const { provider, envVar } of activeEnvVars) {
243
+ prompts.log.info(`${provider} ${UI.Style.TEXT_DIM}${envVar}`)
244
+ }
245
+
246
+ prompts.outro(`${activeEnvVars.length} environment variable` + (activeEnvVars.length === 1 ? "" : "s"))
247
+ }
248
+ },
249
+ })
250
+
251
+ export const AuthLoginCommand = cmd({
252
+ command: "login [url]",
253
+ describe: "log in to a provider",
254
+ builder: (yargs) =>
255
+ yargs.positional("url", {
256
+ describe: "opencode auth provider",
257
+ type: "string",
258
+ }),
259
+ async handler(args) {
260
+ await Instance.provide({
261
+ directory: process.cwd(),
262
+ async fn() {
263
+ UI.empty()
264
+ prompts.intro("Add credential")
265
+ if (args.url) {
266
+ const wellknown = await fetch(`${args.url}/.well-known/opencode`).then((x) => x.json() as any)
267
+ prompts.log.info(`Running \`${wellknown.auth.command.join(" ")}\``)
268
+ const proc = Process.spawn(wellknown.auth.command, {
269
+ stdout: "pipe",
270
+ })
271
+ if (!proc.stdout) {
272
+ prompts.log.error("Failed")
273
+ prompts.outro("Done")
274
+ return
275
+ }
276
+ const [exit, token] = await Promise.all([proc.exited, text(proc.stdout)])
277
+ if (exit !== 0) {
278
+ prompts.log.error("Failed")
279
+ prompts.outro("Done")
280
+ return
281
+ }
282
+ await Auth.set(args.url, {
283
+ type: "wellknown",
284
+ key: wellknown.auth.env,
285
+ token: token.trim(),
286
+ })
287
+ prompts.log.success("Logged into " + args.url)
288
+ prompts.outro("Done")
289
+ return
290
+ }
291
+ await ModelsDev.refresh().catch(() => {})
292
+
293
+ const config = await Config.get()
294
+
295
+ const disabled = new Set(config.disabled_providers ?? [])
296
+ const enabled = config.enabled_providers ? new Set(config.enabled_providers) : undefined
297
+
298
+ const providers = await ModelsDev.get().then((x) => {
299
+ const filtered: Record<string, (typeof x)[string]> = {}
300
+ for (const [key, value] of Object.entries(x)) {
301
+ if ((enabled ? enabled.has(key) : true) && !disabled.has(key)) {
302
+ filtered[key] = value
303
+ }
304
+ }
305
+ return filtered
306
+ })
307
+
308
+ const priority: Record<string, number> = {
309
+ opencode: 0,
310
+ anthropic: 1,
311
+ "github-copilot": 2,
312
+ openai: 3,
313
+ google: 4,
314
+ openrouter: 5,
315
+ vercel: 6,
316
+ }
317
+ const pluginProviders = resolvePluginProviders({
318
+ hooks: await Plugin.list(),
319
+ existingProviders: providers,
320
+ disabled,
321
+ enabled,
322
+ providerNames: Object.fromEntries(Object.entries(config.provider ?? {}).map(([id, p]) => [id, p.name])),
323
+ })
324
+ let provider = await prompts.autocomplete({
325
+ message: "Select provider",
326
+ maxItems: 8,
327
+ options: [
328
+ ...pipe(
329
+ providers,
330
+ values(),
331
+ sortBy(
332
+ (x) => priority[x.id] ?? 99,
333
+ (x) => x.name ?? x.id,
334
+ ),
335
+ map((x) => ({
336
+ label: x.name,
337
+ value: x.id,
338
+ hint: {
339
+ opencode: "recommended",
340
+ anthropic: "Claude Max or API key",
341
+ openai: "ChatGPT Plus/Pro or API key",
342
+ }[x.id],
343
+ })),
344
+ ),
345
+ ...pluginProviders.map((x) => ({
346
+ label: x.name,
347
+ value: x.id,
348
+ hint: "plugin",
349
+ })),
350
+ {
351
+ value: "other",
352
+ label: "Other",
353
+ },
354
+ ],
355
+ })
356
+
357
+ if (prompts.isCancel(provider)) throw new UI.CancelledError()
358
+
359
+ const plugin = await Plugin.list().then((x) => x.findLast((x) => x.auth?.provider === provider))
360
+ if (plugin && plugin.auth) {
361
+ const handled = await handlePluginAuth({ auth: plugin.auth }, provider)
362
+ if (handled) return
363
+ }
364
+
365
+ if (provider === "other") {
366
+ provider = await prompts.text({
367
+ message: "Enter provider id",
368
+ validate: (x) => (x && x.match(/^[0-9a-z-]+$/) ? undefined : "a-z, 0-9 and hyphens only"),
369
+ })
370
+ if (prompts.isCancel(provider)) throw new UI.CancelledError()
371
+ provider = provider.replace(/^@ai-sdk\//, "")
372
+ if (prompts.isCancel(provider)) throw new UI.CancelledError()
373
+
374
+ // Check if a plugin provides auth for this custom provider
375
+ const customPlugin = await Plugin.list().then((x) => x.findLast((x) => x.auth?.provider === provider))
376
+ if (customPlugin && customPlugin.auth) {
377
+ const handled = await handlePluginAuth({ auth: customPlugin.auth }, provider)
378
+ if (handled) return
379
+ }
380
+
381
+ prompts.log.warn(
382
+ `This only stores a credential for ${provider} - you will need configure it in opencode.json, check the docs for examples.`,
383
+ )
384
+ }
385
+
386
+ if (provider === "amazon-bedrock") {
387
+ prompts.log.info(
388
+ "Amazon Bedrock authentication priority:\n" +
389
+ " 1. Bearer token (AWS_BEARER_TOKEN_BEDROCK or /connect)\n" +
390
+ " 2. AWS credential chain (profile, access keys, IAM roles, EKS IRSA)\n\n" +
391
+ "Configure via opencode.json options (profile, region, endpoint) or\n" +
392
+ "AWS environment variables (AWS_PROFILE, AWS_REGION, AWS_ACCESS_KEY_ID, AWS_WEB_IDENTITY_TOKEN_FILE).",
393
+ )
394
+ }
395
+
396
+ if (provider === "opencode") {
397
+ prompts.log.info("Create an api key at https://opencode.ai/auth")
398
+ }
399
+
400
+ if (provider === "vercel") {
401
+ prompts.log.info("You can create an api key at https://vercel.link/ai-gateway-token")
402
+ }
403
+
404
+ if (["cloudflare", "cloudflare-ai-gateway"].includes(provider)) {
405
+ prompts.log.info(
406
+ "Cloudflare AI Gateway can be configured with CLOUDFLARE_GATEWAY_ID, CLOUDFLARE_ACCOUNT_ID, and CLOUDFLARE_API_TOKEN environment variables. Read more: https://opencode.ai/docs/providers/#cloudflare-ai-gateway",
407
+ )
408
+ }
409
+
410
+ const key = await prompts.password({
411
+ message: "Enter your API key",
412
+ validate: (x) => (x && x.length > 0 ? undefined : "Required"),
413
+ })
414
+ if (prompts.isCancel(key)) throw new UI.CancelledError()
415
+ await Auth.set(provider, {
416
+ type: "api",
417
+ key,
418
+ })
419
+
420
+ prompts.outro("Done")
421
+ },
422
+ })
423
+ },
424
+ })
425
+
426
+ export const AuthLogoutCommand = cmd({
427
+ command: "logout",
428
+ describe: "log out from a configured provider",
429
+ async handler() {
430
+ UI.empty()
431
+ const credentials = await Auth.all().then((x) => Object.entries(x))
432
+ prompts.intro("Remove credential")
433
+ if (credentials.length === 0) {
434
+ prompts.log.error("No credentials found")
435
+ return
436
+ }
437
+ const database = await ModelsDev.get()
438
+ const providerID = await prompts.select({
439
+ message: "Select provider",
440
+ options: credentials.map(([key, value]) => ({
441
+ label: (database[key]?.name || key) + UI.Style.TEXT_DIM + " (" + value.type + ")",
442
+ value: key,
443
+ })),
444
+ })
445
+ if (prompts.isCancel(providerID)) throw new UI.CancelledError()
446
+ await Auth.remove(providerID)
447
+ prompts.outro("Logout successful")
448
+ },
449
+ })
@@ -0,0 +1,7 @@
1
+ import type { CommandModule } from "yargs"
2
+
3
+ type WithDoubleDash<T> = T & { "--"?: string[] }
4
+
5
+ export function cmd<T, U>(input: CommandModule<T, WithDoubleDash<U>>) {
6
+ return input
7
+ }