cerebras-cli 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (314) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +10 -0
  3. package/README.md +15 -0
  4. package/bin/opencode +84 -0
  5. package/bunfig.toml +4 -0
  6. package/package.json +128 -0
  7. package/parsers-config.ts +239 -0
  8. package/script/build.ts +151 -0
  9. package/script/postinstall.mjs +122 -0
  10. package/script/publish.ts +256 -0
  11. package/script/schema.ts +47 -0
  12. package/src/acp/README.md +164 -0
  13. package/src/acp/agent.ts +812 -0
  14. package/src/acp/session.ts +70 -0
  15. package/src/acp/types.ts +22 -0
  16. package/src/agent/agent.ts +310 -0
  17. package/src/agent/generate.txt +75 -0
  18. package/src/auth/index.ts +70 -0
  19. package/src/bun/index.ts +152 -0
  20. package/src/bus/global.ts +10 -0
  21. package/src/bus/index.ts +142 -0
  22. package/src/cli/bootstrap.ts +17 -0
  23. package/src/cli/cmd/acp.ts +88 -0
  24. package/src/cli/cmd/agent.ts +165 -0
  25. package/src/cli/cmd/auth.ts +369 -0
  26. package/src/cli/cmd/cmd.ts +7 -0
  27. package/src/cli/cmd/debug/config.ts +15 -0
  28. package/src/cli/cmd/debug/file.ts +91 -0
  29. package/src/cli/cmd/debug/index.ts +41 -0
  30. package/src/cli/cmd/debug/lsp.ts +47 -0
  31. package/src/cli/cmd/debug/ripgrep.ts +83 -0
  32. package/src/cli/cmd/debug/scrap.ts +15 -0
  33. package/src/cli/cmd/debug/snapshot.ts +48 -0
  34. package/src/cli/cmd/export.ts +88 -0
  35. package/src/cli/cmd/generate.ts +38 -0
  36. package/src/cli/cmd/github.ts +1200 -0
  37. package/src/cli/cmd/import.ts +98 -0
  38. package/src/cli/cmd/mcp.ts +400 -0
  39. package/src/cli/cmd/models.ts +77 -0
  40. package/src/cli/cmd/pr.ts +112 -0
  41. package/src/cli/cmd/run.ts +342 -0
  42. package/src/cli/cmd/serve.ts +31 -0
  43. package/src/cli/cmd/session.ts +106 -0
  44. package/src/cli/cmd/stats.ts +298 -0
  45. package/src/cli/cmd/tui/app.tsx +732 -0
  46. package/src/cli/cmd/tui/attach.ts +25 -0
  47. package/src/cli/cmd/tui/component/border.tsx +21 -0
  48. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  49. package/src/cli/cmd/tui/component/dialog-command.tsx +124 -0
  50. package/src/cli/cmd/tui/component/dialog-feedback.tsx +160 -0
  51. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  52. package/src/cli/cmd/tui/component/dialog-model.tsx +223 -0
  53. package/src/cli/cmd/tui/component/dialog-notification.tsx +78 -0
  54. package/src/cli/cmd/tui/component/dialog-provider.tsx +222 -0
  55. package/src/cli/cmd/tui/component/dialog-session-list.tsx +97 -0
  56. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  57. package/src/cli/cmd/tui/component/dialog-status.tsx +114 -0
  58. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  59. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  60. package/src/cli/cmd/tui/component/logo.tsx +37 -0
  61. package/src/cli/cmd/tui/component/notification-banner.tsx +58 -0
  62. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +530 -0
  63. package/src/cli/cmd/tui/component/prompt/history.tsx +107 -0
  64. package/src/cli/cmd/tui/component/prompt/index.tsx +931 -0
  65. package/src/cli/cmd/tui/context/args.tsx +14 -0
  66. package/src/cli/cmd/tui/context/directory.ts +12 -0
  67. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  68. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  69. package/src/cli/cmd/tui/context/keybind.tsx +111 -0
  70. package/src/cli/cmd/tui/context/kv.tsx +49 -0
  71. package/src/cli/cmd/tui/context/local.tsx +339 -0
  72. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  73. package/src/cli/cmd/tui/context/route.tsx +45 -0
  74. package/src/cli/cmd/tui/context/sdk.tsx +75 -0
  75. package/src/cli/cmd/tui/context/sync.tsx +374 -0
  76. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  77. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  78. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  79. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  80. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  81. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  82. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  83. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  84. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  85. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  86. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  87. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  88. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  89. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  90. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  91. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  92. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  93. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  94. package/src/cli/cmd/tui/context/theme/orng.json +245 -0
  95. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  96. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  97. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  98. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  99. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  100. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  101. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  102. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  103. package/src/cli/cmd/tui/context/theme.tsx +1077 -0
  104. package/src/cli/cmd/tui/event.ts +39 -0
  105. package/src/cli/cmd/tui/routes/home.tsx +104 -0
  106. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +93 -0
  107. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +37 -0
  108. package/src/cli/cmd/tui/routes/session/footer.tsx +76 -0
  109. package/src/cli/cmd/tui/routes/session/header.tsx +183 -0
  110. package/src/cli/cmd/tui/routes/session/index.tsx +1703 -0
  111. package/src/cli/cmd/tui/routes/session/sidebar.tsx +586 -0
  112. package/src/cli/cmd/tui/spawn.ts +60 -0
  113. package/src/cli/cmd/tui/thread.ts +120 -0
  114. package/src/cli/cmd/tui/ui/dialog-alert.tsx +55 -0
  115. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +81 -0
  116. package/src/cli/cmd/tui/ui/dialog-help.tsx +36 -0
  117. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +75 -0
  118. package/src/cli/cmd/tui/ui/dialog-select.tsx +317 -0
  119. package/src/cli/cmd/tui/ui/dialog.tsx +170 -0
  120. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  121. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  122. package/src/cli/cmd/tui/util/clipboard.ts +127 -0
  123. package/src/cli/cmd/tui/util/editor.ts +32 -0
  124. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  125. package/src/cli/cmd/tui/worker.ts +63 -0
  126. package/src/cli/cmd/uninstall.ts +344 -0
  127. package/src/cli/cmd/upgrade.ts +67 -0
  128. package/src/cli/cmd/web.ts +84 -0
  129. package/src/cli/error.ts +55 -0
  130. package/src/cli/ui.ts +84 -0
  131. package/src/cli/upgrade.ts +25 -0
  132. package/src/command/index.ts +79 -0
  133. package/src/command/template/initialize.txt +10 -0
  134. package/src/command/template/review.txt +73 -0
  135. package/src/config/config.ts +886 -0
  136. package/src/config/markdown.ts +41 -0
  137. package/src/env/index.ts +26 -0
  138. package/src/file/fzf.ts +124 -0
  139. package/src/file/ignore.ts +83 -0
  140. package/src/file/index.ts +326 -0
  141. package/src/file/ripgrep.ts +391 -0
  142. package/src/file/time.ts +38 -0
  143. package/src/file/watcher.ts +89 -0
  144. package/src/flag/flag.ts +28 -0
  145. package/src/format/formatter.ts +277 -0
  146. package/src/format/index.ts +137 -0
  147. package/src/global/index.ts +52 -0
  148. package/src/id/id.ts +73 -0
  149. package/src/ide/index.ts +75 -0
  150. package/src/index.ts +158 -0
  151. package/src/installation/index.ts +194 -0
  152. package/src/lsp/client.ts +215 -0
  153. package/src/lsp/index.ts +370 -0
  154. package/src/lsp/language.ts +111 -0
  155. package/src/lsp/server.ts +1327 -0
  156. package/src/mcp/auth.ts +82 -0
  157. package/src/mcp/index.ts +576 -0
  158. package/src/mcp/oauth-callback.ts +203 -0
  159. package/src/mcp/oauth-provider.ts +132 -0
  160. package/src/notification/index.ts +101 -0
  161. package/src/patch/index.ts +622 -0
  162. package/src/permission/index.ts +198 -0
  163. package/src/plugin/index.ts +95 -0
  164. package/src/project/bootstrap.ts +31 -0
  165. package/src/project/instance.ts +68 -0
  166. package/src/project/project.ts +133 -0
  167. package/src/project/state.ts +65 -0
  168. package/src/project/vcs.ts +77 -0
  169. package/src/provider/auth.ts +143 -0
  170. package/src/provider/models-macro.ts +11 -0
  171. package/src/provider/models.ts +93 -0
  172. package/src/provider/provider.ts +996 -0
  173. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  174. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  175. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  176. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  177. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +27 -0
  178. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  179. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  180. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  181. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1713 -0
  182. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  183. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  184. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  185. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  186. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  187. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  188. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  189. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  190. package/src/provider/transform.ts +406 -0
  191. package/src/pty/index.ts +226 -0
  192. package/src/ratelimit/index.ts +185 -0
  193. package/src/server/error.ts +36 -0
  194. package/src/server/project.ts +50 -0
  195. package/src/server/server.ts +2463 -0
  196. package/src/server/tui.ts +71 -0
  197. package/src/session/compaction.ts +257 -0
  198. package/src/session/index.ts +470 -0
  199. package/src/session/message-v2.ts +641 -0
  200. package/src/session/message.ts +189 -0
  201. package/src/session/processor.ts +443 -0
  202. package/src/session/prompt/anthropic-20250930.txt +166 -0
  203. package/src/session/prompt/anthropic.txt +105 -0
  204. package/src/session/prompt/anthropic_spoof.txt +1 -0
  205. package/src/session/prompt/beast.txt +147 -0
  206. package/src/session/prompt/build-switch.txt +5 -0
  207. package/src/session/prompt/codex.txt +318 -0
  208. package/src/session/prompt/compaction.txt +12 -0
  209. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  210. package/src/session/prompt/gemini.txt +155 -0
  211. package/src/session/prompt/max-steps.txt +16 -0
  212. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  213. package/src/session/prompt/plan.txt +26 -0
  214. package/src/session/prompt/polaris.txt +107 -0
  215. package/src/session/prompt/qwen.txt +109 -0
  216. package/src/session/prompt/summarize.txt +4 -0
  217. package/src/session/prompt/title.txt +36 -0
  218. package/src/session/prompt.ts +1541 -0
  219. package/src/session/retry.ts +82 -0
  220. package/src/session/revert.ts +108 -0
  221. package/src/session/status.ts +75 -0
  222. package/src/session/summary.ts +203 -0
  223. package/src/session/system.ts +148 -0
  224. package/src/session/todo.ts +36 -0
  225. package/src/share/share-next.ts +195 -0
  226. package/src/share/share.ts +87 -0
  227. package/src/snapshot/index.ts +197 -0
  228. package/src/storage/storage.ts +226 -0
  229. package/src/telemetry/index.ts +232 -0
  230. package/src/tool/bash.ts +365 -0
  231. package/src/tool/bash.txt +128 -0
  232. package/src/tool/batch.ts +173 -0
  233. package/src/tool/batch.txt +28 -0
  234. package/src/tool/codesearch.ts +138 -0
  235. package/src/tool/codesearch.txt +12 -0
  236. package/src/tool/edit.ts +674 -0
  237. package/src/tool/edit.txt +10 -0
  238. package/src/tool/glob.ts +65 -0
  239. package/src/tool/glob.txt +6 -0
  240. package/src/tool/grep.ts +120 -0
  241. package/src/tool/grep.txt +8 -0
  242. package/src/tool/invalid.ts +17 -0
  243. package/src/tool/ls.ts +110 -0
  244. package/src/tool/ls.txt +1 -0
  245. package/src/tool/lsp-diagnostics.ts +26 -0
  246. package/src/tool/lsp-diagnostics.txt +1 -0
  247. package/src/tool/lsp-hover.ts +31 -0
  248. package/src/tool/lsp-hover.txt +1 -0
  249. package/src/tool/multiedit.ts +46 -0
  250. package/src/tool/multiedit.txt +41 -0
  251. package/src/tool/patch.ts +233 -0
  252. package/src/tool/patch.txt +1 -0
  253. package/src/tool/read.ts +217 -0
  254. package/src/tool/read.txt +12 -0
  255. package/src/tool/registry.ts +148 -0
  256. package/src/tool/task.ts +135 -0
  257. package/src/tool/task.txt +60 -0
  258. package/src/tool/todo.ts +39 -0
  259. package/src/tool/todoread.txt +14 -0
  260. package/src/tool/todowrite.txt +167 -0
  261. package/src/tool/tool.ts +66 -0
  262. package/src/tool/webfetch.ts +187 -0
  263. package/src/tool/webfetch.txt +14 -0
  264. package/src/tool/websearch.ts +150 -0
  265. package/src/tool/websearch.txt +11 -0
  266. package/src/tool/write.ts +99 -0
  267. package/src/tool/write.txt +8 -0
  268. package/src/types/shims.d.ts +3 -0
  269. package/src/util/color.ts +19 -0
  270. package/src/util/context.ts +25 -0
  271. package/src/util/defer.ts +12 -0
  272. package/src/util/eventloop.ts +20 -0
  273. package/src/util/filesystem.ts +69 -0
  274. package/src/util/fn.ts +11 -0
  275. package/src/util/iife.ts +3 -0
  276. package/src/util/keybind.ts +79 -0
  277. package/src/util/lazy.ts +11 -0
  278. package/src/util/locale.ts +81 -0
  279. package/src/util/lock.ts +98 -0
  280. package/src/util/log.ts +177 -0
  281. package/src/util/queue.ts +32 -0
  282. package/src/util/rpc.ts +42 -0
  283. package/src/util/scrap.ts +10 -0
  284. package/src/util/signal.ts +12 -0
  285. package/src/util/timeout.ts +14 -0
  286. package/src/util/token.ts +7 -0
  287. package/src/util/wildcard.ts +54 -0
  288. package/sst-env.d.ts +9 -0
  289. package/test/bun.test.ts +53 -0
  290. package/test/config/agent-color.test.ts +66 -0
  291. package/test/config/config.test.ts +503 -0
  292. package/test/config/markdown.test.ts +89 -0
  293. package/test/file/ignore.test.ts +10 -0
  294. package/test/fixture/fixture.ts +28 -0
  295. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  296. package/test/ide/ide.test.ts +82 -0
  297. package/test/keybind.test.ts +317 -0
  298. package/test/lsp/client.test.ts +95 -0
  299. package/test/patch/patch.test.ts +348 -0
  300. package/test/preload.ts +38 -0
  301. package/test/project/project.test.ts +42 -0
  302. package/test/provider/provider.test.ts +1809 -0
  303. package/test/provider/transform.test.ts +305 -0
  304. package/test/session/retry.test.ts +61 -0
  305. package/test/session/session.test.ts +71 -0
  306. package/test/snapshot/snapshot.test.ts +939 -0
  307. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  308. package/test/tool/bash.test.ts +55 -0
  309. package/test/tool/patch.test.ts +259 -0
  310. package/test/util/iife.test.ts +36 -0
  311. package/test/util/lazy.test.ts +50 -0
  312. package/test/util/timeout.test.ts +21 -0
  313. package/test/util/wildcard.test.ts +55 -0
  314. package/tsconfig.json +17 -0
@@ -0,0 +1,369 @@
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
+
14
+ export const AuthCommand = cmd({
15
+ command: "auth",
16
+ describe: "manage credentials",
17
+ builder: (yargs) =>
18
+ yargs.command(AuthLoginCommand).command(AuthLogoutCommand).command(AuthListCommand).demandCommand(),
19
+ async handler() {},
20
+ })
21
+
22
+ export const AuthListCommand = cmd({
23
+ command: "list",
24
+ aliases: ["ls"],
25
+ describe: "list providers",
26
+ async handler() {
27
+ UI.empty()
28
+ const authPath = path.join(Global.Path.data, "auth.json")
29
+ const homedir = os.homedir()
30
+ const displayPath = authPath.startsWith(homedir) ? authPath.replace(homedir, "~") : authPath
31
+ prompts.intro(`Credentials ${UI.Style.TEXT_DIM}${displayPath}`)
32
+ const results = Object.entries(await Auth.all())
33
+ const database = await ModelsDev.get()
34
+
35
+ for (const [providerID, result] of results) {
36
+ const name = database[providerID]?.name || providerID
37
+ prompts.log.info(`${name} ${UI.Style.TEXT_DIM}${result.type}`)
38
+ }
39
+
40
+ prompts.outro(`${results.length} credentials`)
41
+
42
+ // Environment variables section
43
+ const activeEnvVars: Array<{ provider: string; envVar: string }> = []
44
+
45
+ for (const [providerID, provider] of Object.entries(database)) {
46
+ for (const envVar of provider.env) {
47
+ if (process.env[envVar]) {
48
+ activeEnvVars.push({
49
+ provider: provider.name || providerID,
50
+ envVar,
51
+ })
52
+ }
53
+ }
54
+ }
55
+
56
+ if (activeEnvVars.length > 0) {
57
+ UI.empty()
58
+ prompts.intro("Environment")
59
+
60
+ for (const { provider, envVar } of activeEnvVars) {
61
+ prompts.log.info(`${provider} ${UI.Style.TEXT_DIM}${envVar}`)
62
+ }
63
+
64
+ prompts.outro(`${activeEnvVars.length} environment variable` + (activeEnvVars.length === 1 ? "" : "s"))
65
+ }
66
+ },
67
+ })
68
+
69
+ export const AuthLoginCommand = cmd({
70
+ command: "login [url]",
71
+ describe: "log in to a provider",
72
+ builder: (yargs) =>
73
+ yargs.positional("url", {
74
+ describe: "opencode auth provider",
75
+ type: "string",
76
+ }),
77
+ async handler(args) {
78
+ await Instance.provide({
79
+ directory: process.cwd(),
80
+ async fn() {
81
+ UI.empty()
82
+ prompts.intro("Add credential")
83
+ if (args.url) {
84
+ const wellknown = await fetch(`${args.url}/.well-known/opencode`).then((x) => x.json() as any)
85
+ prompts.log.info(`Running \`${wellknown.auth.command.join(" ")}\``)
86
+ const proc = Bun.spawn({
87
+ cmd: wellknown.auth.command,
88
+ stdout: "pipe",
89
+ })
90
+ const exit = await proc.exited
91
+ if (exit !== 0) {
92
+ prompts.log.error("Failed")
93
+ prompts.outro("Done")
94
+ return
95
+ }
96
+ const token = await new Response(proc.stdout).text()
97
+ await Auth.set(args.url, {
98
+ type: "wellknown",
99
+ key: wellknown.auth.env,
100
+ token: token.trim(),
101
+ })
102
+ prompts.log.success("Logged into " + args.url)
103
+ prompts.outro("Done")
104
+ return
105
+ }
106
+ await ModelsDev.refresh().catch(() => {})
107
+
108
+ const config = await Config.get()
109
+
110
+ const disabled = new Set(config.disabled_providers ?? [])
111
+ const enabled = config.enabled_providers ? new Set(config.enabled_providers) : undefined
112
+
113
+ const providers = await ModelsDev.get().then((x) => {
114
+ const filtered: Record<string, (typeof x)[string]> = {}
115
+ for (const [key, value] of Object.entries(x)) {
116
+ if ((enabled ? enabled.has(key) : true) && !disabled.has(key)) {
117
+ filtered[key] = value
118
+ }
119
+ }
120
+ return filtered
121
+ })
122
+
123
+ const priority: Record<string, number> = {
124
+ opencode: 0,
125
+ anthropic: 1,
126
+ "github-copilot": 2,
127
+ openai: 3,
128
+ google: 4,
129
+ openrouter: 5,
130
+ vercel: 6,
131
+ }
132
+ let provider = await prompts.autocomplete({
133
+ message: "Select provider",
134
+ maxItems: 8,
135
+ options: [
136
+ ...pipe(
137
+ providers,
138
+ values(),
139
+ sortBy(
140
+ (x) => priority[x.id] ?? 99,
141
+ (x) => x.name ?? x.id,
142
+ ),
143
+ map((x) => ({
144
+ label: x.name,
145
+ value: x.id,
146
+ hint: {
147
+ opencode: "recommended",
148
+ anthropic: "Claude Max or API key",
149
+ }[x.id],
150
+ })),
151
+ ),
152
+ {
153
+ value: "other",
154
+ label: "Other",
155
+ },
156
+ ],
157
+ })
158
+
159
+ if (prompts.isCancel(provider)) throw new UI.CancelledError()
160
+
161
+ const plugin = await Plugin.list().then((x) => x.find((x) => x.auth?.provider === provider))
162
+ if (plugin && plugin.auth) {
163
+ let index = 0
164
+ if (plugin.auth.methods.length > 1) {
165
+ const method = await prompts.select({
166
+ message: "Login method",
167
+ options: [
168
+ ...plugin.auth.methods.map((x, index) => ({
169
+ label: x.label,
170
+ value: index.toString(),
171
+ })),
172
+ ],
173
+ })
174
+ if (prompts.isCancel(method)) throw new UI.CancelledError()
175
+ index = parseInt(method)
176
+ }
177
+ const method = plugin.auth.methods[index]
178
+
179
+ // Handle prompts for all auth types
180
+ await new Promise((resolve) => setTimeout(resolve, 10))
181
+ const inputs: Record<string, string> = {}
182
+ if (method.prompts) {
183
+ for (const prompt of method.prompts) {
184
+ if (prompt.condition && !prompt.condition(inputs)) {
185
+ continue
186
+ }
187
+ if (prompt.type === "select") {
188
+ const value = await prompts.select({
189
+ message: prompt.message,
190
+ options: prompt.options,
191
+ })
192
+ if (prompts.isCancel(value)) throw new UI.CancelledError()
193
+ inputs[prompt.key] = value
194
+ } else {
195
+ const value = await prompts.text({
196
+ message: prompt.message,
197
+ placeholder: prompt.placeholder,
198
+ validate: prompt.validate ? (v) => prompt.validate!(v ?? "") : undefined,
199
+ })
200
+ if (prompts.isCancel(value)) throw new UI.CancelledError()
201
+ inputs[prompt.key] = value
202
+ }
203
+ }
204
+ }
205
+
206
+ if (method.type === "oauth") {
207
+ const authorize = await method.authorize(inputs)
208
+
209
+ if (authorize.url) {
210
+ prompts.log.info("Go to: " + authorize.url)
211
+ }
212
+
213
+ if (authorize.method === "auto") {
214
+ if (authorize.instructions) {
215
+ prompts.log.info(authorize.instructions)
216
+ }
217
+ const spinner = prompts.spinner()
218
+ spinner.start("Waiting for authorization...")
219
+ const result = await authorize.callback()
220
+ if (result.type === "failed") {
221
+ spinner.stop("Failed to authorize", 1)
222
+ }
223
+ if (result.type === "success") {
224
+ const saveProvider = result.provider ?? provider
225
+ if ("refresh" in result) {
226
+ const { type: _, provider: __, refresh, access, expires, ...extraFields } = result
227
+ await Auth.set(saveProvider, {
228
+ type: "oauth",
229
+ refresh,
230
+ access,
231
+ expires,
232
+ ...extraFields,
233
+ })
234
+ }
235
+ if ("key" in result) {
236
+ await Auth.set(saveProvider, {
237
+ type: "api",
238
+ key: result.key,
239
+ })
240
+ }
241
+ spinner.stop("Login successful")
242
+ }
243
+ }
244
+
245
+ if (authorize.method === "code") {
246
+ const code = await prompts.text({
247
+ message: "Paste the authorization code here: ",
248
+ validate: (x) => (x && x.length > 0 ? undefined : "Required"),
249
+ })
250
+ if (prompts.isCancel(code)) throw new UI.CancelledError()
251
+ const result = await authorize.callback(code)
252
+ if (result.type === "failed") {
253
+ prompts.log.error("Failed to authorize")
254
+ }
255
+ if (result.type === "success") {
256
+ const saveProvider = result.provider ?? provider
257
+ if ("refresh" in result) {
258
+ const { type: _, provider: __, refresh, access, expires, ...extraFields } = result
259
+ await Auth.set(saveProvider, {
260
+ type: "oauth",
261
+ refresh,
262
+ access,
263
+ expires,
264
+ ...extraFields,
265
+ })
266
+ }
267
+ if ("key" in result) {
268
+ await Auth.set(saveProvider, {
269
+ type: "api",
270
+ key: result.key,
271
+ })
272
+ }
273
+ prompts.log.success("Login successful")
274
+ }
275
+ }
276
+
277
+ prompts.outro("Done")
278
+ return
279
+ }
280
+
281
+ if (method.type === "api") {
282
+ if (method.authorize) {
283
+ const result = await method.authorize(inputs)
284
+ if (result.type === "failed") {
285
+ prompts.log.error("Failed to authorize")
286
+ }
287
+ if (result.type === "success") {
288
+ const saveProvider = result.provider ?? provider
289
+ await Auth.set(saveProvider, {
290
+ type: "api",
291
+ key: result.key,
292
+ })
293
+ prompts.log.success("Login successful")
294
+ }
295
+ prompts.outro("Done")
296
+ return
297
+ }
298
+ }
299
+ }
300
+
301
+ if (provider === "other") {
302
+ provider = await prompts.text({
303
+ message: "Enter provider id",
304
+ validate: (x) => (x && x.match(/^[0-9a-z-]+$/) ? undefined : "a-z, 0-9 and hyphens only"),
305
+ })
306
+ if (prompts.isCancel(provider)) throw new UI.CancelledError()
307
+ provider = provider.replace(/^@ai-sdk\//, "")
308
+ if (prompts.isCancel(provider)) throw new UI.CancelledError()
309
+ prompts.log.warn(
310
+ `This only stores a credential for ${provider} - you will need configure it in opencode.json, check the docs for examples.`,
311
+ )
312
+ }
313
+
314
+ if (provider === "amazon-bedrock") {
315
+ prompts.log.info(
316
+ "Amazon bedrock can be configured with standard AWS environment variables like AWS_BEARER_TOKEN_BEDROCK, AWS_PROFILE or AWS_ACCESS_KEY_ID",
317
+ )
318
+ prompts.outro("Done")
319
+ return
320
+ }
321
+
322
+ if (provider === "opencode") {
323
+ prompts.log.info("Create an api key at https://opencode.ai/auth")
324
+ }
325
+
326
+ if (provider === "vercel") {
327
+ prompts.log.info("You can create an api key at https://vercel.link/ai-gateway-token")
328
+ }
329
+
330
+ const key = await prompts.password({
331
+ message: "Enter your API key",
332
+ validate: (x) => (x && x.length > 0 ? undefined : "Required"),
333
+ })
334
+ if (prompts.isCancel(key)) throw new UI.CancelledError()
335
+ await Auth.set(provider, {
336
+ type: "api",
337
+ key,
338
+ })
339
+
340
+ prompts.outro("Done")
341
+ },
342
+ })
343
+ },
344
+ })
345
+
346
+ export const AuthLogoutCommand = cmd({
347
+ command: "logout",
348
+ describe: "log out from a configured provider",
349
+ async handler() {
350
+ UI.empty()
351
+ const credentials = await Auth.all().then((x) => Object.entries(x))
352
+ prompts.intro("Remove credential")
353
+ if (credentials.length === 0) {
354
+ prompts.log.error("No credentials found")
355
+ return
356
+ }
357
+ const database = await ModelsDev.get()
358
+ const providerID = await prompts.select({
359
+ message: "Select provider",
360
+ options: credentials.map(([key, value]) => ({
361
+ label: (database[key]?.name || key) + UI.Style.TEXT_DIM + " (" + value.type + ")",
362
+ value: key,
363
+ })),
364
+ })
365
+ if (prompts.isCancel(providerID)) throw new UI.CancelledError()
366
+ await Auth.remove(providerID)
367
+ prompts.outro("Logout successful")
368
+ },
369
+ })
@@ -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
+ }
@@ -0,0 +1,15 @@
1
+ import { EOL } from "os"
2
+ import { Config } from "../../../config/config"
3
+ import { bootstrap } from "../../bootstrap"
4
+ import { cmd } from "../cmd"
5
+
6
+ export const ConfigCommand = cmd({
7
+ command: "config",
8
+ builder: (yargs) => yargs,
9
+ async handler() {
10
+ await bootstrap(process.cwd(), async () => {
11
+ const config = await Config.get()
12
+ process.stdout.write(JSON.stringify(config, null, 2) + EOL)
13
+ })
14
+ },
15
+ })
@@ -0,0 +1,91 @@
1
+ import { EOL } from "os"
2
+ import { File } from "../../../file"
3
+ import { bootstrap } from "../../bootstrap"
4
+ import { cmd } from "../cmd"
5
+ import { Ripgrep } from "@/file/ripgrep"
6
+
7
+ const FileSearchCommand = cmd({
8
+ command: "search <query>",
9
+ builder: (yargs) =>
10
+ yargs.positional("query", {
11
+ type: "string",
12
+ demandOption: true,
13
+ description: "Search query",
14
+ }),
15
+ async handler(args) {
16
+ await bootstrap(process.cwd(), async () => {
17
+ const results = await File.search({ query: args.query })
18
+ process.stdout.write(results.join(EOL) + EOL)
19
+ })
20
+ },
21
+ })
22
+
23
+ const FileReadCommand = cmd({
24
+ command: "read <path>",
25
+ builder: (yargs) =>
26
+ yargs.positional("path", {
27
+ type: "string",
28
+ demandOption: true,
29
+ description: "File path to read",
30
+ }),
31
+ async handler(args) {
32
+ await bootstrap(process.cwd(), async () => {
33
+ const content = await File.read(args.path)
34
+ process.stdout.write(JSON.stringify(content, null, 2) + EOL)
35
+ })
36
+ },
37
+ })
38
+
39
+ const FileStatusCommand = cmd({
40
+ command: "status",
41
+ builder: (yargs) => yargs,
42
+ async handler() {
43
+ await bootstrap(process.cwd(), async () => {
44
+ const status = await File.status()
45
+ process.stdout.write(JSON.stringify(status, null, 2) + EOL)
46
+ })
47
+ },
48
+ })
49
+
50
+ const FileListCommand = cmd({
51
+ command: "list <path>",
52
+ builder: (yargs) =>
53
+ yargs.positional("path", {
54
+ type: "string",
55
+ demandOption: true,
56
+ description: "File path to list",
57
+ }),
58
+ async handler(args) {
59
+ await bootstrap(process.cwd(), async () => {
60
+ const files = await File.list(args.path)
61
+ process.stdout.write(JSON.stringify(files, null, 2) + EOL)
62
+ })
63
+ },
64
+ })
65
+
66
+ const FileTreeCommand = cmd({
67
+ command: "tree [dir]",
68
+ builder: (yargs) =>
69
+ yargs.positional("dir", {
70
+ type: "string",
71
+ description: "Directory to tree",
72
+ default: process.cwd(),
73
+ }),
74
+ async handler(args) {
75
+ const files = await Ripgrep.tree({ cwd: args.dir, limit: 200 })
76
+ console.log(files)
77
+ },
78
+ })
79
+
80
+ export const FileCommand = cmd({
81
+ command: "file",
82
+ builder: (yargs) =>
83
+ yargs
84
+ .command(FileReadCommand)
85
+ .command(FileStatusCommand)
86
+ .command(FileListCommand)
87
+ .command(FileSearchCommand)
88
+ .command(FileTreeCommand)
89
+ .demandCommand(),
90
+ async handler() {},
91
+ })
@@ -0,0 +1,41 @@
1
+ import { Global } from "../../../global"
2
+ import { bootstrap } from "../../bootstrap"
3
+ import { cmd } from "../cmd"
4
+ import { ConfigCommand } from "./config"
5
+ import { FileCommand } from "./file"
6
+ import { LSPCommand } from "./lsp"
7
+ import { RipgrepCommand } from "./ripgrep"
8
+ import { ScrapCommand } from "./scrap"
9
+ import { SnapshotCommand } from "./snapshot"
10
+
11
+ export const DebugCommand = cmd({
12
+ command: "debug",
13
+ builder: (yargs) =>
14
+ yargs
15
+ .command(ConfigCommand)
16
+ .command(LSPCommand)
17
+ .command(RipgrepCommand)
18
+ .command(FileCommand)
19
+ .command(ScrapCommand)
20
+ .command(SnapshotCommand)
21
+ .command(PathsCommand)
22
+ .command({
23
+ command: "wait",
24
+ async handler() {
25
+ await bootstrap(process.cwd(), async () => {
26
+ await new Promise((resolve) => setTimeout(resolve, 1_000 * 60 * 60 * 24))
27
+ })
28
+ },
29
+ })
30
+ .demandCommand(),
31
+ async handler() {},
32
+ })
33
+
34
+ const PathsCommand = cmd({
35
+ command: "paths",
36
+ handler() {
37
+ for (const [key, value] of Object.entries(Global.Path)) {
38
+ console.log(key.padEnd(10), value)
39
+ }
40
+ },
41
+ })
@@ -0,0 +1,47 @@
1
+ import { LSP } from "../../../lsp"
2
+ import { bootstrap } from "../../bootstrap"
3
+ import { cmd } from "../cmd"
4
+ import { Log } from "../../../util/log"
5
+ import { EOL } from "os"
6
+
7
+ export const LSPCommand = cmd({
8
+ command: "lsp",
9
+ builder: (yargs) =>
10
+ yargs.command(DiagnosticsCommand).command(SymbolsCommand).command(DocumentSymbolsCommand).demandCommand(),
11
+ async handler() {},
12
+ })
13
+
14
+ const DiagnosticsCommand = cmd({
15
+ command: "diagnostics <file>",
16
+ builder: (yargs) => yargs.positional("file", { type: "string", demandOption: true }),
17
+ async handler(args) {
18
+ await bootstrap(process.cwd(), async () => {
19
+ await LSP.touchFile(args.file, true)
20
+ process.stdout.write(JSON.stringify(await LSP.diagnostics(), null, 2) + EOL)
21
+ })
22
+ },
23
+ })
24
+
25
+ export const SymbolsCommand = cmd({
26
+ command: "symbols <query>",
27
+ builder: (yargs) => yargs.positional("query", { type: "string", demandOption: true }),
28
+ async handler(args) {
29
+ await bootstrap(process.cwd(), async () => {
30
+ using _ = Log.Default.time("symbols")
31
+ const results = await LSP.workspaceSymbol(args.query)
32
+ process.stdout.write(JSON.stringify(results, null, 2) + EOL)
33
+ })
34
+ },
35
+ })
36
+
37
+ export const DocumentSymbolsCommand = cmd({
38
+ command: "document-symbols <uri>",
39
+ builder: (yargs) => yargs.positional("uri", { type: "string", demandOption: true }),
40
+ async handler(args) {
41
+ await bootstrap(process.cwd(), async () => {
42
+ using _ = Log.Default.time("document-symbols")
43
+ const results = await LSP.documentSymbol(args.uri)
44
+ process.stdout.write(JSON.stringify(results, null, 2) + EOL)
45
+ })
46
+ },
47
+ })
@@ -0,0 +1,83 @@
1
+ import { EOL } from "os"
2
+ import { Ripgrep } from "../../../file/ripgrep"
3
+ import { Instance } from "../../../project/instance"
4
+ import { bootstrap } from "../../bootstrap"
5
+ import { cmd } from "../cmd"
6
+
7
+ export const RipgrepCommand = cmd({
8
+ command: "rg",
9
+ builder: (yargs) => yargs.command(TreeCommand).command(FilesCommand).command(SearchCommand).demandCommand(),
10
+ async handler() {},
11
+ })
12
+
13
+ const TreeCommand = cmd({
14
+ command: "tree",
15
+ builder: (yargs) =>
16
+ yargs.option("limit", {
17
+ type: "number",
18
+ }),
19
+ async handler(args) {
20
+ await bootstrap(process.cwd(), async () => {
21
+ process.stdout.write((await Ripgrep.tree({ cwd: Instance.directory, limit: args.limit })) + EOL)
22
+ })
23
+ },
24
+ })
25
+
26
+ const FilesCommand = cmd({
27
+ command: "files",
28
+ builder: (yargs) =>
29
+ yargs
30
+ .option("query", {
31
+ type: "string",
32
+ description: "Filter files by query",
33
+ })
34
+ .option("glob", {
35
+ type: "string",
36
+ description: "Glob pattern to match files",
37
+ })
38
+ .option("limit", {
39
+ type: "number",
40
+ description: "Limit number of results",
41
+ }),
42
+ async handler(args) {
43
+ await bootstrap(process.cwd(), async () => {
44
+ const files: string[] = []
45
+ for await (const file of Ripgrep.files({
46
+ cwd: Instance.directory,
47
+ glob: args.glob ? [args.glob] : undefined,
48
+ })) {
49
+ files.push(file)
50
+ if (args.limit && files.length >= args.limit) break
51
+ }
52
+ process.stdout.write(files.join(EOL) + EOL)
53
+ })
54
+ },
55
+ })
56
+
57
+ const SearchCommand = cmd({
58
+ command: "search <pattern>",
59
+ builder: (yargs) =>
60
+ yargs
61
+ .positional("pattern", {
62
+ type: "string",
63
+ demandOption: true,
64
+ description: "Search pattern",
65
+ })
66
+ .option("glob", {
67
+ type: "array",
68
+ description: "File glob patterns",
69
+ })
70
+ .option("limit", {
71
+ type: "number",
72
+ description: "Limit number of results",
73
+ }),
74
+ async handler(args) {
75
+ const results = await Ripgrep.search({
76
+ cwd: process.cwd(),
77
+ pattern: args.pattern,
78
+ glob: args.glob as string[] | undefined,
79
+ limit: args.limit,
80
+ })
81
+ process.stdout.write(JSON.stringify(results, null, 2) + EOL)
82
+ },
83
+ })
@@ -0,0 +1,15 @@
1
+ import { EOL } from "os"
2
+ import { Project } from "../../../project/project"
3
+ import { Log } from "../../../util/log"
4
+ import { cmd } from "../cmd"
5
+
6
+ export const ScrapCommand = cmd({
7
+ command: "scrap",
8
+ builder: (yargs) => yargs,
9
+ async handler() {
10
+ const timer = Log.Default.time("scrap")
11
+ const list = await Project.list()
12
+ process.stdout.write(JSON.stringify(list, null, 2) + EOL)
13
+ timer.stop()
14
+ },
15
+ })