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
package/src/index.ts ADDED
@@ -0,0 +1,158 @@
1
+ import yargs from "yargs"
2
+ import { hideBin } from "yargs/helpers"
3
+ import { RunCommand } from "./cli/cmd/run"
4
+ import { GenerateCommand } from "./cli/cmd/generate"
5
+ import { Log } from "./util/log"
6
+ import { AuthCommand } from "./cli/cmd/auth"
7
+ import { AgentCommand } from "./cli/cmd/agent"
8
+ import { UpgradeCommand } from "./cli/cmd/upgrade"
9
+ import { UninstallCommand } from "./cli/cmd/uninstall"
10
+ import { ModelsCommand } from "./cli/cmd/models"
11
+ import { UI } from "./cli/ui"
12
+ import { Installation } from "./installation"
13
+ import { NamedError } from "@opencode-ai/util/error"
14
+ import { FormatError } from "./cli/error"
15
+ import { ServeCommand } from "./cli/cmd/serve"
16
+ import { DebugCommand } from "./cli/cmd/debug"
17
+ import { StatsCommand } from "./cli/cmd/stats"
18
+ import { McpCommand } from "./cli/cmd/mcp"
19
+ import { GithubCommand } from "./cli/cmd/github"
20
+ import { ExportCommand } from "./cli/cmd/export"
21
+ import { ImportCommand } from "./cli/cmd/import"
22
+ import { AttachCommand } from "./cli/cmd/tui/attach"
23
+ import { TuiThreadCommand } from "./cli/cmd/tui/thread"
24
+ import { TuiSpawnCommand } from "./cli/cmd/tui/spawn"
25
+ import { AcpCommand } from "./cli/cmd/acp"
26
+ import { EOL } from "os"
27
+ import { WebCommand } from "./cli/cmd/web"
28
+ import { PrCommand } from "./cli/cmd/pr"
29
+ import { SessionCommand } from "./cli/cmd/session"
30
+
31
+ process.on("unhandledRejection", (e) => {
32
+ Log.Default.error("rejection", {
33
+ e: e instanceof Error ? e.message : e,
34
+ })
35
+ })
36
+
37
+ process.on("uncaughtException", (e) => {
38
+ Log.Default.error("exception", {
39
+ e: e instanceof Error ? e.message : e,
40
+ })
41
+ })
42
+
43
+ const cli = yargs(hideBin(process.argv))
44
+ .parserConfiguration({ "populate--": true })
45
+ .scriptName("opencode")
46
+ .wrap(100)
47
+ .help("help", "show help")
48
+ .alias("help", "h")
49
+ .version("version", "show version number", Installation.VERSION)
50
+ .alias("version", "v")
51
+ .option("print-logs", {
52
+ describe: "print logs to stderr",
53
+ type: "boolean",
54
+ })
55
+ .option("log-level", {
56
+ describe: "log level",
57
+ type: "string",
58
+ choices: ["DEBUG", "INFO", "WARN", "ERROR"],
59
+ })
60
+ .middleware(async (opts) => {
61
+ await Log.init({
62
+ print: process.argv.includes("--print-logs"),
63
+ dev: Installation.isLocal(),
64
+ level: (() => {
65
+ if (opts.logLevel) return opts.logLevel as Log.Level
66
+ if (Installation.isLocal()) return "DEBUG"
67
+ return "INFO"
68
+ })(),
69
+ })
70
+
71
+ process.env.AGENT = "1"
72
+ process.env.OPENCODE = "1"
73
+
74
+ Log.Default.info("opencode", {
75
+ version: Installation.VERSION,
76
+ args: process.argv.slice(2),
77
+ })
78
+ })
79
+ .usage("\n" + UI.logo())
80
+ .command(AcpCommand)
81
+ .command(McpCommand)
82
+ .command(TuiThreadCommand)
83
+ .command(TuiSpawnCommand)
84
+ .command(AttachCommand)
85
+ .command(RunCommand)
86
+ .command(GenerateCommand)
87
+ .command(DebugCommand)
88
+ .command(AuthCommand)
89
+ .command(AgentCommand)
90
+ .command(UpgradeCommand)
91
+ .command(UninstallCommand)
92
+ .command(ServeCommand)
93
+ .command(WebCommand)
94
+ .command(ModelsCommand)
95
+ .command(StatsCommand)
96
+ .command(ExportCommand)
97
+ .command(ImportCommand)
98
+ .command(GithubCommand)
99
+ .command(PrCommand)
100
+ .command(SessionCommand)
101
+ .fail((msg) => {
102
+ if (
103
+ msg.startsWith("Unknown argument") ||
104
+ msg.startsWith("Not enough non-option arguments") ||
105
+ msg.startsWith("Invalid values:")
106
+ ) {
107
+ cli.showHelp("log")
108
+ }
109
+ process.exit(1)
110
+ })
111
+ .strict()
112
+
113
+ try {
114
+ await cli.parse()
115
+ } catch (e) {
116
+ let data: Record<string, any> = {}
117
+ if (e instanceof NamedError) {
118
+ const obj = e.toObject()
119
+ Object.assign(data, {
120
+ ...obj.data,
121
+ })
122
+ }
123
+
124
+ if (e instanceof Error) {
125
+ Object.assign(data, {
126
+ name: e.name,
127
+ message: e.message,
128
+ cause: e.cause?.toString(),
129
+ stack: e.stack,
130
+ })
131
+ }
132
+
133
+ if (e instanceof ResolveMessage) {
134
+ Object.assign(data, {
135
+ name: e.name,
136
+ message: e.message,
137
+ code: e.code,
138
+ specifier: e.specifier,
139
+ referrer: e.referrer,
140
+ position: e.position,
141
+ importKind: e.importKind,
142
+ })
143
+ }
144
+ Log.Default.error("fatal", data)
145
+ const formatted = FormatError(e)
146
+ if (formatted) UI.error(formatted)
147
+ if (formatted === undefined) {
148
+ UI.error("Unexpected error, check log file at " + Log.file() + " for more details" + EOL)
149
+ console.error(e)
150
+ }
151
+ process.exitCode = 1
152
+ } finally {
153
+ // Some subprocesses don't react properly to SIGTERM and similar signals.
154
+ // Most notably, some docker-container-based MCP servers don't handle such signals unless
155
+ // run using `docker run --init`.
156
+ // Explicitly exit to avoid any hanging subprocesses.
157
+ process.exit()
158
+ }
@@ -0,0 +1,194 @@
1
+ import path from "path"
2
+ import { $ } from "bun"
3
+ import z from "zod"
4
+ import { NamedError } from "@opencode-ai/util/error"
5
+ import { Bus } from "../bus"
6
+ import { Log } from "../util/log"
7
+ import { iife } from "@/util/iife"
8
+
9
+ declare global {
10
+ const OPENCODE_VERSION: string
11
+ const OPENCODE_CHANNEL: string
12
+ }
13
+
14
+ export namespace Installation {
15
+ const log = Log.create({ service: "installation" })
16
+
17
+ export type Method = Awaited<ReturnType<typeof method>>
18
+
19
+ export const Event = {
20
+ Updated: Bus.event(
21
+ "installation.updated",
22
+ z.object({
23
+ version: z.string(),
24
+ }),
25
+ ),
26
+ UpdateAvailable: Bus.event(
27
+ "installation.update-available",
28
+ z.object({
29
+ version: z.string(),
30
+ }),
31
+ ),
32
+ }
33
+
34
+ export const Info = z
35
+ .object({
36
+ version: z.string(),
37
+ latest: z.string(),
38
+ })
39
+ .meta({
40
+ ref: "InstallationInfo",
41
+ })
42
+ export type Info = z.infer<typeof Info>
43
+
44
+ export async function info() {
45
+ return {
46
+ version: VERSION,
47
+ latest: await latest(),
48
+ }
49
+ }
50
+
51
+ export function isPreview() {
52
+ return CHANNEL !== "latest"
53
+ }
54
+
55
+ export function isLocal() {
56
+ return CHANNEL === "local"
57
+ }
58
+
59
+ export async function method() {
60
+ if (process.execPath.includes(path.join(".opencode", "bin"))) return "curl"
61
+ if (process.execPath.includes(path.join(".local", "bin"))) return "curl"
62
+ const exec = process.execPath.toLowerCase()
63
+
64
+ const checks = [
65
+ {
66
+ name: "npm" as const,
67
+ command: () => $`npm list -g --depth=0`.throws(false).text(),
68
+ },
69
+ {
70
+ name: "yarn" as const,
71
+ command: () => $`yarn global list`.throws(false).text(),
72
+ },
73
+ {
74
+ name: "pnpm" as const,
75
+ command: () => $`pnpm list -g --depth=0`.throws(false).text(),
76
+ },
77
+ {
78
+ name: "bun" as const,
79
+ command: () => $`bun pm ls -g`.throws(false).text(),
80
+ },
81
+ {
82
+ name: "brew" as const,
83
+ command: () => $`brew list --formula opencode`.throws(false).text(),
84
+ },
85
+ ]
86
+
87
+ checks.sort((a, b) => {
88
+ const aMatches = exec.includes(a.name)
89
+ const bMatches = exec.includes(b.name)
90
+ if (aMatches && !bMatches) return -1
91
+ if (!aMatches && bMatches) return 1
92
+ return 0
93
+ })
94
+
95
+ for (const check of checks) {
96
+ const output = await check.command()
97
+ if (output.includes(check.name === "brew" ? "opencode" : "opencode-ai")) {
98
+ return check.name
99
+ }
100
+ }
101
+
102
+ return "unknown"
103
+ }
104
+
105
+ export const UpgradeFailedError = NamedError.create(
106
+ "UpgradeFailedError",
107
+ z.object({
108
+ stderr: z.string(),
109
+ }),
110
+ )
111
+
112
+ async function getBrewFormula() {
113
+ const tapFormula = await $`brew list --formula sst/tap/opencode`.throws(false).text()
114
+ if (tapFormula.includes("opencode")) return "sst/tap/opencode"
115
+ const coreFormula = await $`brew list --formula opencode`.throws(false).text()
116
+ if (coreFormula.includes("opencode")) return "opencode"
117
+ return "opencode"
118
+ }
119
+
120
+ export async function upgrade(method: Method, target: string) {
121
+ let cmd
122
+ switch (method) {
123
+ case "curl":
124
+ cmd = $`curl -fsSL https://opencode.ai/install | bash`.env({
125
+ ...process.env,
126
+ VERSION: target,
127
+ })
128
+ break
129
+ case "npm":
130
+ cmd = $`npm install -g opencode-ai@${target}`
131
+ break
132
+ case "pnpm":
133
+ cmd = $`pnpm install -g opencode-ai@${target}`
134
+ break
135
+ case "bun":
136
+ cmd = $`bun install -g opencode-ai@${target}`
137
+ break
138
+ case "brew": {
139
+ const formula = await getBrewFormula()
140
+ cmd = $`brew install ${formula}`.env({
141
+ HOMEBREW_NO_AUTO_UPDATE: "1",
142
+ ...process.env,
143
+ })
144
+ break
145
+ }
146
+ default:
147
+ throw new Error(`Unknown method: ${method}`)
148
+ }
149
+ const result = await cmd.quiet().throws(false)
150
+ log.info("upgraded", {
151
+ method,
152
+ target,
153
+ stdout: result.stdout.toString(),
154
+ stderr: result.stderr.toString(),
155
+ })
156
+ if (result.exitCode !== 0)
157
+ throw new UpgradeFailedError({
158
+ stderr: result.stderr.toString("utf8"),
159
+ })
160
+ }
161
+
162
+ export const VERSION = typeof OPENCODE_VERSION === "string" ? OPENCODE_VERSION : "local"
163
+ export const CHANNEL = typeof OPENCODE_CHANNEL === "string" ? OPENCODE_CHANNEL : "local"
164
+ export const USER_AGENT = `opencode/${CHANNEL}/${VERSION}`
165
+
166
+ export async function latest(installMethod?: Method) {
167
+ const detectedMethod = installMethod || (await method())
168
+ if (detectedMethod === "brew") {
169
+ const formula = await getBrewFormula()
170
+ if (formula === "opencode") {
171
+ return fetch("https://formulae.brew.sh/api/formula/opencode.json")
172
+ .then((res) => {
173
+ if (!res.ok) throw new Error(res.statusText)
174
+ return res.json()
175
+ })
176
+ .then((data: any) => data.versions.stable)
177
+ }
178
+ }
179
+
180
+ const registry = await iife(async () => {
181
+ const r = (await $`npm config get registry`.quiet().nothrow().text()).trim()
182
+ const reg = r || "https://registry.npmjs.org"
183
+ return reg.endsWith("/") ? reg.slice(0, -1) : reg
184
+ })
185
+ const [major] = VERSION.split(".").map((x) => Number(x))
186
+ const channel = CHANNEL === "latest" ? `latest-${major}` : CHANNEL
187
+ return fetch(`${registry}/opencode-ai/${channel}`)
188
+ .then((res) => {
189
+ if (!res.ok) throw new Error(res.statusText)
190
+ return res.json()
191
+ })
192
+ .then((data: any) => data.version)
193
+ }
194
+ }
@@ -0,0 +1,215 @@
1
+ import path from "path"
2
+ import { createMessageConnection, StreamMessageReader, StreamMessageWriter } from "vscode-jsonrpc/node"
3
+ import type { Diagnostic as VSCodeDiagnostic } from "vscode-languageserver-types"
4
+ import { Log } from "../util/log"
5
+ import { LANGUAGE_EXTENSIONS } from "./language"
6
+ import { Bus } from "../bus"
7
+ import z from "zod"
8
+ import type { LSPServer } from "./server"
9
+ import { NamedError } from "@opencode-ai/util/error"
10
+ import { withTimeout } from "../util/timeout"
11
+ import { Instance } from "../project/instance"
12
+
13
+ export namespace LSPClient {
14
+ const log = Log.create({ service: "lsp.client" })
15
+
16
+ export type Info = NonNullable<Awaited<ReturnType<typeof create>>>
17
+
18
+ export type Diagnostic = VSCodeDiagnostic
19
+
20
+ export const InitializeError = NamedError.create(
21
+ "LSPInitializeError",
22
+ z.object({
23
+ serverID: z.string(),
24
+ }),
25
+ )
26
+
27
+ export const Event = {
28
+ Diagnostics: Bus.event(
29
+ "lsp.client.diagnostics",
30
+ z.object({
31
+ serverID: z.string(),
32
+ path: z.string(),
33
+ }),
34
+ ),
35
+ }
36
+
37
+ export async function create(input: { serverID: string; server: LSPServer.Handle; root: string }) {
38
+ const l = log.clone().tag("serverID", input.serverID)
39
+ l.info("starting client")
40
+
41
+ const connection = createMessageConnection(
42
+ new StreamMessageReader(input.server.process.stdout as any),
43
+ new StreamMessageWriter(input.server.process.stdin as any),
44
+ )
45
+
46
+ const diagnostics = new Map<string, Diagnostic[]>()
47
+ connection.onNotification("textDocument/publishDiagnostics", (params) => {
48
+ const path = new URL(params.uri).pathname
49
+ l.info("textDocument/publishDiagnostics", {
50
+ path,
51
+ })
52
+ const exists = diagnostics.has(path)
53
+ diagnostics.set(path, params.diagnostics)
54
+ if (!exists && input.serverID === "typescript") return
55
+ Bus.publish(Event.Diagnostics, { path, serverID: input.serverID })
56
+ })
57
+ connection.onRequest("window/workDoneProgress/create", (params) => {
58
+ l.info("window/workDoneProgress/create", params)
59
+ return null
60
+ })
61
+ connection.onRequest("workspace/configuration", async () => {
62
+ // Return server initialization options
63
+ return [input.server.initialization ?? {}]
64
+ })
65
+ connection.onRequest("client/registerCapability", async () => {})
66
+ connection.onRequest("client/unregisterCapability", async () => {})
67
+ connection.onRequest("workspace/workspaceFolders", async () => [
68
+ {
69
+ name: "workspace",
70
+ uri: "file://" + input.root,
71
+ },
72
+ ])
73
+ connection.listen()
74
+
75
+ l.info("sending initialize")
76
+ await withTimeout(
77
+ connection.sendRequest("initialize", {
78
+ rootUri: "file://" + input.root,
79
+ processId: input.server.process.pid,
80
+ workspaceFolders: [
81
+ {
82
+ name: "workspace",
83
+ uri: "file://" + input.root,
84
+ },
85
+ ],
86
+ initializationOptions: {
87
+ ...input.server.initialization,
88
+ },
89
+ capabilities: {
90
+ window: {
91
+ workDoneProgress: true,
92
+ },
93
+ workspace: {
94
+ configuration: true,
95
+ },
96
+ textDocument: {
97
+ synchronization: {
98
+ didOpen: true,
99
+ didChange: true,
100
+ },
101
+ publishDiagnostics: {
102
+ versionSupport: true,
103
+ },
104
+ },
105
+ },
106
+ }),
107
+ 45_000,
108
+ ).catch((err) => {
109
+ l.error("initialize error", { error: err })
110
+ throw new InitializeError(
111
+ { serverID: input.serverID },
112
+ {
113
+ cause: err,
114
+ },
115
+ )
116
+ })
117
+
118
+ await connection.sendNotification("initialized", {})
119
+
120
+ if (input.server.initialization) {
121
+ await connection.sendNotification("workspace/didChangeConfiguration", {
122
+ settings: input.server.initialization,
123
+ })
124
+ }
125
+
126
+ const files: {
127
+ [path: string]: number
128
+ } = {}
129
+
130
+ const result = {
131
+ root: input.root,
132
+ get serverID() {
133
+ return input.serverID
134
+ },
135
+ get connection() {
136
+ return connection
137
+ },
138
+ notify: {
139
+ async open(input: { path: string }) {
140
+ input.path = path.isAbsolute(input.path) ? input.path : path.resolve(Instance.directory, input.path)
141
+ const file = Bun.file(input.path)
142
+ const text = await file.text()
143
+ const extension = path.extname(input.path)
144
+ const languageId = LANGUAGE_EXTENSIONS[extension] ?? "plaintext"
145
+
146
+ const version = files[input.path]
147
+ if (version !== undefined) {
148
+ const next = version + 1
149
+ files[input.path] = next
150
+ log.info("textDocument/didChange", {
151
+ path: input.path,
152
+ version: next,
153
+ })
154
+ await connection.sendNotification("textDocument/didChange", {
155
+ textDocument: {
156
+ uri: `file://` + input.path,
157
+ version: next,
158
+ },
159
+ contentChanges: [{ text }],
160
+ })
161
+ return
162
+ }
163
+
164
+ log.info("textDocument/didOpen", input)
165
+ diagnostics.delete(input.path)
166
+ await connection.sendNotification("textDocument/didOpen", {
167
+ textDocument: {
168
+ uri: `file://` + input.path,
169
+ languageId,
170
+ version: 0,
171
+ text,
172
+ },
173
+ })
174
+ files[input.path] = 0
175
+ return
176
+ },
177
+ },
178
+ get diagnostics() {
179
+ return diagnostics
180
+ },
181
+ async waitForDiagnostics(input: { path: string }) {
182
+ input.path = path.isAbsolute(input.path) ? input.path : path.resolve(Instance.directory, input.path)
183
+ log.info("waiting for diagnostics", input)
184
+ let unsub: () => void
185
+ return await withTimeout(
186
+ new Promise<void>((resolve) => {
187
+ unsub = Bus.subscribe(Event.Diagnostics, (event) => {
188
+ if (event.properties.path === input.path && event.properties.serverID === result.serverID) {
189
+ log.info("got diagnostics", input)
190
+ unsub?.()
191
+ resolve()
192
+ }
193
+ })
194
+ }),
195
+ 3000,
196
+ )
197
+ .catch(() => {})
198
+ .finally(() => {
199
+ unsub?.()
200
+ })
201
+ },
202
+ async shutdown() {
203
+ l.info("shutting down")
204
+ connection.end()
205
+ connection.dispose()
206
+ input.server.process.kill()
207
+ l.info("shutdown")
208
+ },
209
+ }
210
+
211
+ l.info("initialized")
212
+
213
+ return result
214
+ }
215
+ }