opencode-v2 1.1.53

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (439) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/opencode +84 -0
  5. package/bunfig.toml +5 -0
  6. package/package.json +126 -0
  7. package/parsers-config.ts +253 -0
  8. package/script/build.ts +193 -0
  9. package/script/postinstall.mjs +125 -0
  10. package/script/publish.ts +181 -0
  11. package/script/schema.ts +47 -0
  12. package/script/seed-e2e.ts +50 -0
  13. package/src/acp/README.md +164 -0
  14. package/src/acp/agent.ts +1676 -0
  15. package/src/acp/session.ts +117 -0
  16. package/src/acp/types.ts +23 -0
  17. package/src/agent/agent.ts +414 -0
  18. package/src/agent/generate.txt +75 -0
  19. package/src/agent/prompt/compaction.txt +12 -0
  20. package/src/agent/prompt/explore.txt +18 -0
  21. package/src/agent/prompt/summary.txt +11 -0
  22. package/src/agent/prompt/title.txt +44 -0
  23. package/src/auth/index.ts +70 -0
  24. package/src/bun/index.ts +137 -0
  25. package/src/bun/registry.ts +48 -0
  26. package/src/bus/bus-event.ts +43 -0
  27. package/src/bus/global.ts +10 -0
  28. package/src/bus/index.ts +105 -0
  29. package/src/cli/bootstrap.ts +17 -0
  30. package/src/cli/cmd/acp.ts +70 -0
  31. package/src/cli/cmd/agent.ts +257 -0
  32. package/src/cli/cmd/auth.ts +400 -0
  33. package/src/cli/cmd/cmd.ts +7 -0
  34. package/src/cli/cmd/debug/agent.ts +167 -0
  35. package/src/cli/cmd/debug/config.ts +16 -0
  36. package/src/cli/cmd/debug/file.ts +97 -0
  37. package/src/cli/cmd/debug/index.ts +48 -0
  38. package/src/cli/cmd/debug/lsp.ts +52 -0
  39. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  40. package/src/cli/cmd/debug/scrap.ts +16 -0
  41. package/src/cli/cmd/debug/skill.ts +16 -0
  42. package/src/cli/cmd/debug/snapshot.ts +52 -0
  43. package/src/cli/cmd/export.ts +88 -0
  44. package/src/cli/cmd/generate.ts +38 -0
  45. package/src/cli/cmd/github.ts +1540 -0
  46. package/src/cli/cmd/import.ts +147 -0
  47. package/src/cli/cmd/mcp.ts +755 -0
  48. package/src/cli/cmd/models.ts +77 -0
  49. package/src/cli/cmd/pr.ts +112 -0
  50. package/src/cli/cmd/run.ts +617 -0
  51. package/src/cli/cmd/serve.ts +20 -0
  52. package/src/cli/cmd/session.ts +135 -0
  53. package/src/cli/cmd/stats.ts +426 -0
  54. package/src/cli/cmd/tui/app.tsx +801 -0
  55. package/src/cli/cmd/tui/attach.ts +52 -0
  56. package/src/cli/cmd/tui/component/border.tsx +21 -0
  57. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  58. package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
  59. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  60. package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
  61. package/src/cli/cmd/tui/component/dialog-provider.tsx +266 -0
  62. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  63. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  64. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  65. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  66. package/src/cli/cmd/tui/component/dialog-status.tsx +177 -0
  67. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  68. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  69. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  70. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +666 -0
  71. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  72. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  73. package/src/cli/cmd/tui/component/prompt/index.tsx +1132 -0
  74. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  75. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  76. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  77. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  78. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  79. package/src/cli/cmd/tui/context/args.tsx +15 -0
  80. package/src/cli/cmd/tui/context/directory.ts +13 -0
  81. package/src/cli/cmd/tui/context/exit.tsx +52 -0
  82. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  83. package/src/cli/cmd/tui/context/keybind.tsx +100 -0
  84. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  85. package/src/cli/cmd/tui/context/local.tsx +409 -0
  86. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  87. package/src/cli/cmd/tui/context/route.tsx +46 -0
  88. package/src/cli/cmd/tui/context/sdk.tsx +101 -0
  89. package/src/cli/cmd/tui/context/sync.tsx +470 -0
  90. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  91. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  92. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  93. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  94. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  95. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  96. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  97. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  98. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  99. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  100. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  101. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  102. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  103. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  104. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  105. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  106. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  107. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  108. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  109. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  110. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  111. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  112. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  113. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  114. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  115. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  116. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  117. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  118. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  119. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  120. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  121. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  122. package/src/cli/cmd/tui/context/theme.tsx +1152 -0
  123. package/src/cli/cmd/tui/event.ts +48 -0
  124. package/src/cli/cmd/tui/routes/home.tsx +140 -0
  125. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  126. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  128. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  129. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  130. package/src/cli/cmd/tui/routes/session/header.tsx +142 -0
  131. package/src/cli/cmd/tui/routes/session/index.tsx +2126 -0
  132. package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
  133. package/src/cli/cmd/tui/routes/session/question.tsx +466 -0
  134. package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
  135. package/src/cli/cmd/tui/thread.ts +175 -0
  136. package/src/cli/cmd/tui/ui/dialog-alert.tsx +68 -0
  137. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +93 -0
  138. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +215 -0
  139. package/src/cli/cmd/tui/ui/dialog-help.tsx +49 -0
  140. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +88 -0
  141. package/src/cli/cmd/tui/ui/dialog-select.tsx +399 -0
  142. package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
  143. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  144. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  145. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  146. package/src/cli/cmd/tui/util/clipboard.ts +159 -0
  147. package/src/cli/cmd/tui/util/editor.ts +32 -0
  148. package/src/cli/cmd/tui/util/signal.ts +7 -0
  149. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  150. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  151. package/src/cli/cmd/tui/worker.ts +152 -0
  152. package/src/cli/cmd/uninstall.ts +357 -0
  153. package/src/cli/cmd/upgrade.ts +73 -0
  154. package/src/cli/cmd/web.ts +81 -0
  155. package/src/cli/error.ts +57 -0
  156. package/src/cli/logo.ts +6 -0
  157. package/src/cli/network.ts +60 -0
  158. package/src/cli/ui.ts +113 -0
  159. package/src/cli/upgrade.ts +25 -0
  160. package/src/command/index.ts +150 -0
  161. package/src/command/template/initialize.txt +10 -0
  162. package/src/command/template/review.txt +99 -0
  163. package/src/config/config.ts +1477 -0
  164. package/src/config/markdown.ts +98 -0
  165. package/src/env/index.ts +28 -0
  166. package/src/file/ignore.ts +83 -0
  167. package/src/file/index.ts +583 -0
  168. package/src/file/ripgrep.ts +375 -0
  169. package/src/file/time.ts +69 -0
  170. package/src/file/watcher.ts +127 -0
  171. package/src/flag/flag.ts +97 -0
  172. package/src/format/formatter.ts +366 -0
  173. package/src/format/index.ts +137 -0
  174. package/src/global/index.ts +55 -0
  175. package/src/id/id.ts +83 -0
  176. package/src/ide/index.ts +76 -0
  177. package/src/index.ts +159 -0
  178. package/src/installation/index.ts +246 -0
  179. package/src/lsp/client.ts +252 -0
  180. package/src/lsp/index.ts +485 -0
  181. package/src/lsp/language.ts +119 -0
  182. package/src/lsp/server.ts +2046 -0
  183. package/src/mcp/auth.ts +132 -0
  184. package/src/mcp/index.ts +934 -0
  185. package/src/mcp/oauth-callback.ts +200 -0
  186. package/src/mcp/oauth-provider.ts +154 -0
  187. package/src/patch/index.ts +680 -0
  188. package/src/permission/arity.ts +163 -0
  189. package/src/permission/index.ts +210 -0
  190. package/src/permission/next.ts +280 -0
  191. package/src/plugin/codex.ts +624 -0
  192. package/src/plugin/copilot.ts +327 -0
  193. package/src/plugin/index.ts +138 -0
  194. package/src/project/bootstrap.ts +35 -0
  195. package/src/project/instance.ts +114 -0
  196. package/src/project/project.ts +371 -0
  197. package/src/project/state.ts +70 -0
  198. package/src/project/vcs.ts +76 -0
  199. package/src/provider/auth.ts +147 -0
  200. package/src/provider/models.ts +133 -0
  201. package/src/provider/provider.ts +1262 -0
  202. package/src/provider/sdk/copilot/README.md +5 -0
  203. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +164 -0
  204. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  205. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +17 -0
  206. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  207. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +780 -0
  208. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  209. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  210. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +87 -0
  211. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  212. package/src/provider/sdk/copilot/index.ts +2 -0
  213. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  214. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +303 -0
  215. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  216. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  217. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  218. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +207 -0
  219. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1732 -0
  220. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +177 -0
  221. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  222. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +88 -0
  223. package/src/provider/sdk/copilot/responses/tool/file-search.ts +128 -0
  224. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +115 -0
  225. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +65 -0
  226. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +104 -0
  227. package/src/provider/sdk/copilot/responses/tool/web-search.ts +103 -0
  228. package/src/provider/transform.ts +828 -0
  229. package/src/pty/index.ts +250 -0
  230. package/src/question/index.ts +171 -0
  231. package/src/scheduler/index.ts +61 -0
  232. package/src/server/error.ts +36 -0
  233. package/src/server/event.ts +7 -0
  234. package/src/server/mdns.ts +60 -0
  235. package/src/server/routes/config.ts +92 -0
  236. package/src/server/routes/experimental.ts +208 -0
  237. package/src/server/routes/file.ts +197 -0
  238. package/src/server/routes/global.ts +183 -0
  239. package/src/server/routes/mcp.ts +225 -0
  240. package/src/server/routes/permission.ts +68 -0
  241. package/src/server/routes/project.ts +82 -0
  242. package/src/server/routes/provider.ts +165 -0
  243. package/src/server/routes/pty.ts +169 -0
  244. package/src/server/routes/question.ts +98 -0
  245. package/src/server/routes/session.ts +939 -0
  246. package/src/server/routes/tui.ts +379 -0
  247. package/src/server/server.ts +613 -0
  248. package/src/session/compaction.ts +226 -0
  249. package/src/session/index.ts +524 -0
  250. package/src/session/instruction.ts +197 -0
  251. package/src/session/llm.ts +289 -0
  252. package/src/session/message-v2.ts +802 -0
  253. package/src/session/message.ts +189 -0
  254. package/src/session/processor.ts +407 -0
  255. package/src/session/prompt/agent.txt +43 -0
  256. package/src/session/prompt/anthropic-20250930.txt +166 -0
  257. package/src/session/prompt/anthropic.txt +105 -0
  258. package/src/session/prompt/beast.txt +147 -0
  259. package/src/session/prompt/build-switch.txt +5 -0
  260. package/src/session/prompt/codex_header.txt +79 -0
  261. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  262. package/src/session/prompt/gemini.txt +155 -0
  263. package/src/session/prompt/max-steps.txt +16 -0
  264. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  265. package/src/session/prompt/plan.txt +26 -0
  266. package/src/session/prompt/qwen.txt +109 -0
  267. package/src/session/prompt/research.txt +81 -0
  268. package/src/session/prompt/trinity.txt +97 -0
  269. package/src/session/prompt.ts +1952 -0
  270. package/src/session/retry.ts +97 -0
  271. package/src/session/revert.ts +121 -0
  272. package/src/session/status.ts +76 -0
  273. package/src/session/summary.ts +217 -0
  274. package/src/session/system.ts +54 -0
  275. package/src/session/todo.ts +37 -0
  276. package/src/share/share-next.ts +200 -0
  277. package/src/share/share.ts +92 -0
  278. package/src/shell/shell.ts +67 -0
  279. package/src/skill/discovery.ts +97 -0
  280. package/src/skill/index.ts +1 -0
  281. package/src/skill/skill.ts +188 -0
  282. package/src/snapshot/index.ts +255 -0
  283. package/src/storage/storage.ts +227 -0
  284. package/src/tool/agent-enter.txt +1 -0
  285. package/src/tool/agent-exit.txt +1 -0
  286. package/src/tool/agent.ts +237 -0
  287. package/src/tool/apply_patch.ts +281 -0
  288. package/src/tool/apply_patch.txt +33 -0
  289. package/src/tool/bash.ts +269 -0
  290. package/src/tool/bash.txt +115 -0
  291. package/src/tool/batch.ts +175 -0
  292. package/src/tool/batch.txt +24 -0
  293. package/src/tool/chat-enter.txt +15 -0
  294. package/src/tool/chat-exit.txt +7 -0
  295. package/src/tool/chat.ts +217 -0
  296. package/src/tool/codesearch.ts +132 -0
  297. package/src/tool/codesearch.txt +12 -0
  298. package/src/tool/edit.ts +655 -0
  299. package/src/tool/edit.txt +10 -0
  300. package/src/tool/external-directory.ts +32 -0
  301. package/src/tool/glob.ts +78 -0
  302. package/src/tool/glob.txt +6 -0
  303. package/src/tool/grep.ts +147 -0
  304. package/src/tool/grep.txt +8 -0
  305. package/src/tool/invalid.ts +17 -0
  306. package/src/tool/ls.ts +121 -0
  307. package/src/tool/ls.txt +1 -0
  308. package/src/tool/lsp.ts +96 -0
  309. package/src/tool/lsp.txt +19 -0
  310. package/src/tool/multiedit.ts +46 -0
  311. package/src/tool/multiedit.txt +41 -0
  312. package/src/tool/plan-enter.txt +14 -0
  313. package/src/tool/plan-exit.txt +13 -0
  314. package/src/tool/plan.ts +130 -0
  315. package/src/tool/question.ts +33 -0
  316. package/src/tool/question.txt +10 -0
  317. package/src/tool/read.ts +211 -0
  318. package/src/tool/read.txt +12 -0
  319. package/src/tool/registry.ts +167 -0
  320. package/src/tool/research-enter.txt +1 -0
  321. package/src/tool/research-exit.txt +1 -0
  322. package/src/tool/research.ts +134 -0
  323. package/src/tool/skill.ts +123 -0
  324. package/src/tool/task.ts +165 -0
  325. package/src/tool/task.txt +60 -0
  326. package/src/tool/todo.ts +53 -0
  327. package/src/tool/todoread.txt +14 -0
  328. package/src/tool/todowrite.txt +167 -0
  329. package/src/tool/tool.ts +89 -0
  330. package/src/tool/truncation.ts +106 -0
  331. package/src/tool/webfetch.ts +186 -0
  332. package/src/tool/webfetch.txt +13 -0
  333. package/src/tool/websearch.ts +150 -0
  334. package/src/tool/websearch.txt +14 -0
  335. package/src/tool/write.ts +85 -0
  336. package/src/tool/write.txt +8 -0
  337. package/src/util/abort.ts +35 -0
  338. package/src/util/archive.ts +16 -0
  339. package/src/util/color.ts +19 -0
  340. package/src/util/context.ts +25 -0
  341. package/src/util/defer.ts +12 -0
  342. package/src/util/eventloop.ts +20 -0
  343. package/src/util/filesystem.ts +93 -0
  344. package/src/util/fn.ts +11 -0
  345. package/src/util/format.ts +20 -0
  346. package/src/util/iife.ts +3 -0
  347. package/src/util/keybind.ts +103 -0
  348. package/src/util/lazy.ts +18 -0
  349. package/src/util/locale.ts +81 -0
  350. package/src/util/lock.ts +98 -0
  351. package/src/util/log.ts +180 -0
  352. package/src/util/proxied.ts +3 -0
  353. package/src/util/queue.ts +32 -0
  354. package/src/util/rpc.ts +66 -0
  355. package/src/util/scrap.ts +10 -0
  356. package/src/util/signal.ts +12 -0
  357. package/src/util/timeout.ts +14 -0
  358. package/src/util/token.ts +7 -0
  359. package/src/util/wildcard.ts +56 -0
  360. package/src/worktree/index.ts +574 -0
  361. package/sst-env.d.ts +9 -0
  362. package/test/acp/agent-interface.test.ts +51 -0
  363. package/test/acp/event-subscription.test.ts +436 -0
  364. package/test/agent/agent.test.ts +675 -0
  365. package/test/bun.test.ts +53 -0
  366. package/test/cli/github-action.test.ts +161 -0
  367. package/test/cli/github-remote.test.ts +80 -0
  368. package/test/cli/import.test.ts +38 -0
  369. package/test/cli/tui/transcript.test.ts +322 -0
  370. package/test/config/agent-color.test.ts +71 -0
  371. package/test/config/config.test.ts +1802 -0
  372. package/test/config/fixtures/empty-frontmatter.md +4 -0
  373. package/test/config/fixtures/frontmatter.md +28 -0
  374. package/test/config/fixtures/markdown-header.md +11 -0
  375. package/test/config/fixtures/no-frontmatter.md +1 -0
  376. package/test/config/fixtures/weird-model-id.md +13 -0
  377. package/test/config/markdown.test.ts +228 -0
  378. package/test/file/ignore.test.ts +10 -0
  379. package/test/file/path-traversal.test.ts +198 -0
  380. package/test/file/ripgrep.test.ts +39 -0
  381. package/test/fixture/fixture.ts +45 -0
  382. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  383. package/test/ide/ide.test.ts +82 -0
  384. package/test/keybind.test.ts +421 -0
  385. package/test/lsp/client.test.ts +95 -0
  386. package/test/mcp/headers.test.ts +153 -0
  387. package/test/mcp/oauth-browser.test.ts +249 -0
  388. package/test/memory/abort-leak.test.ts +136 -0
  389. package/test/patch/patch.test.ts +348 -0
  390. package/test/permission/arity.test.ts +33 -0
  391. package/test/permission/next.test.ts +690 -0
  392. package/test/permission-task.test.ts +319 -0
  393. package/test/plugin/auth-override.test.ts +44 -0
  394. package/test/plugin/codex.test.ts +123 -0
  395. package/test/preload.ts +63 -0
  396. package/test/project/project.test.ts +120 -0
  397. package/test/provider/amazon-bedrock.test.ts +445 -0
  398. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  399. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  400. package/test/provider/gitlab-duo.test.ts +262 -0
  401. package/test/provider/provider.test.ts +2129 -0
  402. package/test/provider/transform.test.ts +2022 -0
  403. package/test/question/question.test.ts +300 -0
  404. package/test/scheduler.test.ts +73 -0
  405. package/test/server/session-list.test.ts +39 -0
  406. package/test/server/session-select.test.ts +78 -0
  407. package/test/session/compaction.test.ts +293 -0
  408. package/test/session/instruction.test.ts +170 -0
  409. package/test/session/llm.test.ts +691 -0
  410. package/test/session/message-v2.test.ts +786 -0
  411. package/test/session/prompt-missing-file.test.ts +53 -0
  412. package/test/session/prompt-special-chars.test.ts +56 -0
  413. package/test/session/prompt-variant.test.ts +60 -0
  414. package/test/session/retry.test.ts +179 -0
  415. package/test/session/revert-compact.test.ts +285 -0
  416. package/test/session/session.test.ts +71 -0
  417. package/test/skill/discovery.test.ts +60 -0
  418. package/test/skill/skill.test.ts +388 -0
  419. package/test/snapshot/snapshot.test.ts +1040 -0
  420. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  421. package/test/tool/apply_patch.test.ts +559 -0
  422. package/test/tool/bash.test.ts +399 -0
  423. package/test/tool/external-directory.test.ts +127 -0
  424. package/test/tool/fixtures/large-image.png +0 -0
  425. package/test/tool/fixtures/models-api.json +38413 -0
  426. package/test/tool/grep.test.ts +110 -0
  427. package/test/tool/question.test.ts +107 -0
  428. package/test/tool/read.test.ts +358 -0
  429. package/test/tool/registry.test.ts +122 -0
  430. package/test/tool/skill.test.ts +112 -0
  431. package/test/tool/truncation.test.ts +159 -0
  432. package/test/util/filesystem.test.ts +39 -0
  433. package/test/util/format.test.ts +59 -0
  434. package/test/util/iife.test.ts +36 -0
  435. package/test/util/lazy.test.ts +50 -0
  436. package/test/util/lock.test.ts +72 -0
  437. package/test/util/timeout.test.ts +21 -0
  438. package/test/util/wildcard.test.ts +75 -0
  439. package/tsconfig.json +16 -0
@@ -0,0 +1,617 @@
1
+ import type { Argv } from "yargs"
2
+ import path from "path"
3
+ import { pathToFileURL } from "bun"
4
+ import { UI } from "../ui"
5
+ import { cmd } from "./cmd"
6
+ import { Flag } from "../../flag/flag"
7
+ import { bootstrap } from "../bootstrap"
8
+ import { EOL } from "os"
9
+ import { createOpencodeClient, type Message, type OpencodeClient, type ToolPart } from "@opencode-ai/sdk/v2"
10
+ import { Server } from "../../server/server"
11
+ import { Provider } from "../../provider/provider"
12
+ import { Agent } from "../../agent/agent"
13
+ import { PermissionNext } from "../../permission/next"
14
+ import { Tool } from "../../tool/tool"
15
+ import { GlobTool } from "../../tool/glob"
16
+ import { GrepTool } from "../../tool/grep"
17
+ import { ListTool } from "../../tool/ls"
18
+ import { ReadTool } from "../../tool/read"
19
+ import { WebFetchTool } from "../../tool/webfetch"
20
+ import { EditTool } from "../../tool/edit"
21
+ import { WriteTool } from "../../tool/write"
22
+ import { CodeSearchTool } from "../../tool/codesearch"
23
+ import { WebSearchTool } from "../../tool/websearch"
24
+ import { TaskTool } from "../../tool/task"
25
+ import { SkillTool } from "../../tool/skill"
26
+ import { BashTool } from "../../tool/bash"
27
+ import { TodoWriteTool } from "../../tool/todo"
28
+ import { Locale } from "../../util/locale"
29
+
30
+ type ToolProps<T extends Tool.Info> = {
31
+ input: Tool.InferParameters<T>
32
+ metadata: Tool.InferMetadata<T>
33
+ part: ToolPart
34
+ }
35
+
36
+ function props<T extends Tool.Info>(part: ToolPart): ToolProps<T> {
37
+ const state = part.state
38
+ return {
39
+ input: state.input as Tool.InferParameters<T>,
40
+ metadata: ("metadata" in state ? state.metadata : {}) as Tool.InferMetadata<T>,
41
+ part,
42
+ }
43
+ }
44
+
45
+ type Inline = {
46
+ icon: string
47
+ title: string
48
+ description?: string
49
+ }
50
+
51
+ function inline(info: Inline) {
52
+ const suffix = info.description ? UI.Style.TEXT_DIM + ` ${info.description}` + UI.Style.TEXT_NORMAL : ""
53
+ UI.println(UI.Style.TEXT_NORMAL + info.icon, UI.Style.TEXT_NORMAL + info.title + suffix)
54
+ }
55
+
56
+ function block(info: Inline, output?: string) {
57
+ UI.empty()
58
+ inline(info)
59
+ if (!output?.trim()) return
60
+ UI.println(output)
61
+ UI.empty()
62
+ }
63
+
64
+ function fallback(part: ToolPart) {
65
+ const state = part.state
66
+ const input = "input" in state ? state.input : undefined
67
+ const title =
68
+ ("title" in state && state.title ? state.title : undefined) ||
69
+ (input && typeof input === "object" && Object.keys(input).length > 0 ? JSON.stringify(input) : "Unknown")
70
+ inline({
71
+ icon: "⚙",
72
+ title: `${part.tool} ${title}`,
73
+ })
74
+ }
75
+
76
+ function glob(info: ToolProps<typeof GlobTool>) {
77
+ const root = info.input.path ?? ""
78
+ const title = `Glob "${info.input.pattern}"`
79
+ const suffix = root ? `in ${normalizePath(root)}` : ""
80
+ const num = info.metadata.count
81
+ const description =
82
+ num === undefined ? suffix : `${suffix}${suffix ? " · " : ""}${num} ${num === 1 ? "match" : "matches"}`
83
+ inline({
84
+ icon: "✱",
85
+ title,
86
+ ...(description && { description }),
87
+ })
88
+ }
89
+
90
+ function grep(info: ToolProps<typeof GrepTool>) {
91
+ const root = info.input.path ?? ""
92
+ const title = `Grep "${info.input.pattern}"`
93
+ const suffix = root ? `in ${normalizePath(root)}` : ""
94
+ const num = info.metadata.matches
95
+ const description =
96
+ num === undefined ? suffix : `${suffix}${suffix ? " · " : ""}${num} ${num === 1 ? "match" : "matches"}`
97
+ inline({
98
+ icon: "✱",
99
+ title,
100
+ ...(description && { description }),
101
+ })
102
+ }
103
+
104
+ function list(info: ToolProps<typeof ListTool>) {
105
+ const dir = info.input.path ? normalizePath(info.input.path) : ""
106
+ inline({
107
+ icon: "→",
108
+ title: dir ? `List ${dir}` : "List",
109
+ })
110
+ }
111
+
112
+ function read(info: ToolProps<typeof ReadTool>) {
113
+ const file = normalizePath(info.input.filePath)
114
+ const pairs = Object.entries(info.input).filter(([key, value]) => {
115
+ if (key === "filePath") return false
116
+ return typeof value === "string" || typeof value === "number" || typeof value === "boolean"
117
+ })
118
+ const description = pairs.length ? `[${pairs.map(([key, value]) => `${key}=${value}`).join(", ")}]` : undefined
119
+ inline({
120
+ icon: "→",
121
+ title: `Read ${file}`,
122
+ ...(description && { description }),
123
+ })
124
+ }
125
+
126
+ function write(info: ToolProps<typeof WriteTool>) {
127
+ block(
128
+ {
129
+ icon: "←",
130
+ title: `Write ${normalizePath(info.input.filePath)}`,
131
+ },
132
+ info.part.state.status === "completed" ? info.part.state.output : undefined,
133
+ )
134
+ }
135
+
136
+ function webfetch(info: ToolProps<typeof WebFetchTool>) {
137
+ inline({
138
+ icon: "%",
139
+ title: `WebFetch ${info.input.url}`,
140
+ })
141
+ }
142
+
143
+ function edit(info: ToolProps<typeof EditTool>) {
144
+ const title = normalizePath(info.input.filePath)
145
+ const diff = info.metadata.diff
146
+ block(
147
+ {
148
+ icon: "←",
149
+ title: `Edit ${title}`,
150
+ },
151
+ diff,
152
+ )
153
+ }
154
+
155
+ function codesearch(info: ToolProps<typeof CodeSearchTool>) {
156
+ inline({
157
+ icon: "◇",
158
+ title: `Exa Code Search "${info.input.query}"`,
159
+ })
160
+ }
161
+
162
+ function websearch(info: ToolProps<typeof WebSearchTool>) {
163
+ inline({
164
+ icon: "◈",
165
+ title: `Exa Web Search "${info.input.query}"`,
166
+ })
167
+ }
168
+
169
+ function task(info: ToolProps<typeof TaskTool>) {
170
+ const agent = Locale.titlecase(info.input.subagent_type)
171
+ const desc = info.input.description
172
+ const started = info.part.state.status === "running"
173
+ const name = desc ?? `${agent} Task`
174
+ inline({
175
+ icon: started ? "•" : "✓",
176
+ title: name,
177
+ description: desc ? `${agent} Agent` : undefined,
178
+ })
179
+ }
180
+
181
+ function skill(info: ToolProps<typeof SkillTool>) {
182
+ inline({
183
+ icon: "→",
184
+ title: `Skill "${info.input.name}"`,
185
+ })
186
+ }
187
+
188
+ function bash(info: ToolProps<typeof BashTool>) {
189
+ const output = info.part.state.status === "completed" ? info.part.state.output?.trim() : undefined
190
+ block(
191
+ {
192
+ icon: "$",
193
+ title: `${info.input.command}`,
194
+ },
195
+ output,
196
+ )
197
+ }
198
+
199
+ function todo(info: ToolProps<typeof TodoWriteTool>) {
200
+ block(
201
+ {
202
+ icon: "#",
203
+ title: "Todos",
204
+ },
205
+ info.input.todos.map((item) => `${item.status === "completed" ? "[x]" : "[ ]"} ${item.content}`).join("\n"),
206
+ )
207
+ }
208
+
209
+ function normalizePath(input?: string) {
210
+ if (!input) return ""
211
+ if (path.isAbsolute(input)) return path.relative(process.cwd(), input) || "."
212
+ return input
213
+ }
214
+
215
+ export const RunCommand = cmd({
216
+ command: "run [message..]",
217
+ describe: "run opencode with a message",
218
+ builder: (yargs: Argv) => {
219
+ return yargs
220
+ .positional("message", {
221
+ describe: "message to send",
222
+ type: "string",
223
+ array: true,
224
+ default: [],
225
+ })
226
+ .option("command", {
227
+ describe: "the command to run, use message for args",
228
+ type: "string",
229
+ })
230
+ .option("continue", {
231
+ alias: ["c"],
232
+ describe: "continue the last session",
233
+ type: "boolean",
234
+ })
235
+ .option("session", {
236
+ alias: ["s"],
237
+ describe: "session id to continue",
238
+ type: "string",
239
+ })
240
+ .option("fork", {
241
+ describe: "fork the session before continuing (requires --continue or --session)",
242
+ type: "boolean",
243
+ })
244
+ .option("share", {
245
+ type: "boolean",
246
+ describe: "share the session",
247
+ })
248
+ .option("model", {
249
+ type: "string",
250
+ alias: ["m"],
251
+ describe: "model to use in the format of provider/model",
252
+ })
253
+ .option("agent", {
254
+ type: "string",
255
+ describe: "agent to use",
256
+ })
257
+ .option("mode", {
258
+ type: "string",
259
+ choices: ["build", "plan", "chat"],
260
+ describe: "agent mode to start with (cycles: build → plan → chat). Only available when starting a new session.",
261
+ })
262
+ .option("format", {
263
+ type: "string",
264
+ choices: ["default", "json"],
265
+ default: "default",
266
+ describe: "format: default (formatted) or json (raw JSON events)",
267
+ })
268
+ .option("file", {
269
+ alias: ["f"],
270
+ type: "string",
271
+ array: true,
272
+ describe: "file(s) to attach to message",
273
+ })
274
+ .option("title", {
275
+ type: "string",
276
+ describe: "title for the session (uses truncated prompt if no value provided)",
277
+ })
278
+ .option("attach", {
279
+ type: "string",
280
+ describe: "attach to a running opencode server (e.g., http://localhost:4096)",
281
+ })
282
+ .option("port", {
283
+ type: "number",
284
+ describe: "port for the local server (defaults to random port if no value provided)",
285
+ })
286
+ .option("variant", {
287
+ type: "string",
288
+ describe: "model variant (provider-specific reasoning effort, e.g., high, max, minimal)",
289
+ })
290
+ .option("thinking", {
291
+ type: "boolean",
292
+ describe: "show thinking blocks",
293
+ default: false,
294
+ })
295
+ },
296
+ handler: async (args) => {
297
+ let message = [...args.message, ...(args["--"] || [])]
298
+ .map((arg) => (arg.includes(" ") ? `"${arg.replace(/"/g, '\\"')}"` : arg))
299
+ .join(" ")
300
+
301
+ const files: { type: "file"; url: string; filename: string; mime: string }[] = []
302
+ if (args.file) {
303
+ const list = Array.isArray(args.file) ? args.file : [args.file]
304
+
305
+ for (const filePath of list) {
306
+ const resolvedPath = path.resolve(process.cwd(), filePath)
307
+ const file = Bun.file(resolvedPath)
308
+ const stats = await file.stat().catch(() => {})
309
+ if (!stats) {
310
+ UI.error(`File not found: ${filePath}`)
311
+ process.exit(1)
312
+ }
313
+ if (!(await file.exists())) {
314
+ UI.error(`File not found: ${filePath}`)
315
+ process.exit(1)
316
+ }
317
+
318
+ const stat = await file.stat()
319
+ const mime = stat.isDirectory() ? "application/x-directory" : "text/plain"
320
+
321
+ files.push({
322
+ type: "file",
323
+ url: pathToFileURL(resolvedPath).href,
324
+ filename: path.basename(resolvedPath),
325
+ mime,
326
+ })
327
+ }
328
+ }
329
+
330
+ if (!process.stdin.isTTY) message += "\n" + (await Bun.stdin.text())
331
+
332
+ if (message.trim().length === 0 && !args.command) {
333
+ UI.error("You must provide a message or a command")
334
+ process.exit(1)
335
+ }
336
+
337
+ if (args.fork && !args.continue && !args.session) {
338
+ UI.error("--fork requires --continue or --session")
339
+ process.exit(1)
340
+ }
341
+
342
+ const rules: PermissionNext.Ruleset = [
343
+ {
344
+ permission: "question",
345
+ action: "deny",
346
+ pattern: "*",
347
+ },
348
+ {
349
+ permission: "plan_enter",
350
+ action: "deny",
351
+ pattern: "*",
352
+ },
353
+ {
354
+ permission: "plan_exit",
355
+ action: "deny",
356
+ pattern: "*",
357
+ },
358
+ ]
359
+
360
+ function title() {
361
+ if (args.title === undefined) return
362
+ if (args.title !== "") return args.title
363
+ return message.slice(0, 50) + (message.length > 50 ? "..." : "")
364
+ }
365
+
366
+ async function session(sdk: OpencodeClient) {
367
+ const baseID = args.continue ? (await sdk.session.list()).data?.find((s) => !s.parentID)?.id : args.session
368
+
369
+ if (baseID && args.fork) {
370
+ const forked = await sdk.session.fork({ sessionID: baseID })
371
+ return forked.data?.id
372
+ }
373
+
374
+ if (baseID) return baseID
375
+
376
+ const name = title()
377
+ const result = await sdk.session.create({ title: name, permission: rules })
378
+ return result.data?.id
379
+ }
380
+
381
+ async function share(sdk: OpencodeClient, sessionID: string) {
382
+ const cfg = await sdk.config.get()
383
+ if (!cfg.data) return
384
+ if (cfg.data.share !== "auto" && !Flag.OPENCODE_AUTO_SHARE && !args.share) return
385
+ const res = await sdk.session.share({ sessionID }).catch((error) => {
386
+ if (error instanceof Error && error.message.includes("disabled")) {
387
+ UI.println(UI.Style.TEXT_DANGER_BOLD + "! " + error.message)
388
+ }
389
+ return { error }
390
+ })
391
+ if (!res.error && "data" in res && res.data?.share?.url) {
392
+ UI.println(UI.Style.TEXT_INFO_BOLD + "~ " + res.data.share.url)
393
+ }
394
+ }
395
+
396
+ async function execute(sdk: OpencodeClient) {
397
+ function tool(part: ToolPart) {
398
+ if (part.tool === "bash") return bash(props<typeof BashTool>(part))
399
+ if (part.tool === "glob") return glob(props<typeof GlobTool>(part))
400
+ if (part.tool === "grep") return grep(props<typeof GrepTool>(part))
401
+ if (part.tool === "list") return list(props<typeof ListTool>(part))
402
+ if (part.tool === "read") return read(props<typeof ReadTool>(part))
403
+ if (part.tool === "write") return write(props<typeof WriteTool>(part))
404
+ if (part.tool === "webfetch") return webfetch(props<typeof WebFetchTool>(part))
405
+ if (part.tool === "edit") return edit(props<typeof EditTool>(part))
406
+ if (part.tool === "codesearch") return codesearch(props<typeof CodeSearchTool>(part))
407
+ if (part.tool === "websearch") return websearch(props<typeof WebSearchTool>(part))
408
+ if (part.tool === "task") return task(props<typeof TaskTool>(part))
409
+ if (part.tool === "todowrite") return todo(props<typeof TodoWriteTool>(part))
410
+ if (part.tool === "skill") return skill(props<typeof SkillTool>(part))
411
+ return fallback(part)
412
+ }
413
+
414
+ function emit(type: string, data: Record<string, unknown>) {
415
+ if (args.format === "json") {
416
+ process.stdout.write(JSON.stringify({ type, timestamp: Date.now(), sessionID, ...data }) + EOL)
417
+ return true
418
+ }
419
+ return false
420
+ }
421
+
422
+ const events = await sdk.event.subscribe()
423
+ let error: string | undefined
424
+
425
+ async function loop() {
426
+ const toggles = new Map<string, boolean>()
427
+
428
+ for await (const event of events.stream) {
429
+ if (
430
+ event.type === "message.updated" &&
431
+ event.properties.info.role === "assistant" &&
432
+ args.format !== "json" &&
433
+ toggles.get("start") !== true
434
+ ) {
435
+ UI.empty()
436
+ UI.println(`> ${event.properties.info.agent} · ${event.properties.info.modelID}`)
437
+ UI.empty()
438
+ toggles.set("start", true)
439
+ }
440
+
441
+ if (event.type === "message.part.updated") {
442
+ const part = event.properties.part
443
+ if (part.sessionID !== sessionID) continue
444
+
445
+ if (part.type === "tool" && part.state.status === "completed") {
446
+ if (emit("tool_use", { part })) continue
447
+ tool(part)
448
+ }
449
+
450
+ if (
451
+ part.type === "tool" &&
452
+ part.tool === "task" &&
453
+ part.state.status === "running" &&
454
+ args.format !== "json"
455
+ ) {
456
+ if (toggles.get(part.id) === true) continue
457
+ task(props<typeof TaskTool>(part))
458
+ toggles.set(part.id, true)
459
+ }
460
+
461
+ if (part.type === "step-start") {
462
+ if (emit("step_start", { part })) continue
463
+ }
464
+
465
+ if (part.type === "step-finish") {
466
+ if (emit("step_finish", { part })) continue
467
+ }
468
+
469
+ if (part.type === "text" && part.time?.end) {
470
+ if (emit("text", { part })) continue
471
+ const text = part.text.trim()
472
+ if (!text) continue
473
+ if (!process.stdout.isTTY) {
474
+ process.stdout.write(text + EOL)
475
+ continue
476
+ }
477
+ UI.empty()
478
+ UI.println(text)
479
+ UI.empty()
480
+ }
481
+
482
+ if (part.type === "reasoning" && part.time?.end && args.thinking) {
483
+ if (emit("reasoning", { part })) continue
484
+ const text = part.text.trim()
485
+ if (!text) continue
486
+ const line = `Thinking: ${text}`
487
+ if (process.stdout.isTTY) {
488
+ UI.empty()
489
+ UI.println(`${UI.Style.TEXT_DIM}\u001b[3m${line}\u001b[0m${UI.Style.TEXT_NORMAL}`)
490
+ UI.empty()
491
+ continue
492
+ }
493
+ process.stdout.write(line + EOL)
494
+ }
495
+ }
496
+
497
+ if (event.type === "session.error") {
498
+ const props = event.properties
499
+ if (props.sessionID !== sessionID || !props.error) continue
500
+ let err = String(props.error.name)
501
+ if ("data" in props.error && props.error.data && "message" in props.error.data) {
502
+ err = String(props.error.data.message)
503
+ }
504
+ error = error ? error + EOL + err : err
505
+ if (emit("error", { error: props.error })) continue
506
+ UI.error(err)
507
+ }
508
+
509
+ if (
510
+ event.type === "session.status" &&
511
+ event.properties.sessionID === sessionID &&
512
+ event.properties.status.type === "idle"
513
+ ) {
514
+ break
515
+ }
516
+
517
+ if (event.type === "permission.asked") {
518
+ const permission = event.properties
519
+ if (permission.sessionID !== sessionID) continue
520
+ UI.println(
521
+ UI.Style.TEXT_WARNING_BOLD + "!",
522
+ UI.Style.TEXT_NORMAL +
523
+ `permission requested: ${permission.permission} (${permission.patterns.join(", ")}); auto-rejecting`,
524
+ )
525
+ await sdk.permission.reply({
526
+ requestID: permission.id,
527
+ reply: "reject",
528
+ })
529
+ }
530
+ }
531
+ }
532
+
533
+ // Validate agent if specified, or use mode to determine agent
534
+ const agent = await (async () => {
535
+ // If --mode is specified and we're not continuing a session, use it
536
+ if (args.mode && !args.continue && !args.session) {
537
+ const entry = await Agent.get(args.mode)
538
+ if (!entry) {
539
+ UI.println(
540
+ UI.Style.TEXT_WARNING_BOLD + "!",
541
+ UI.Style.TEXT_NORMAL,
542
+ `mode "${args.mode}" not found. Falling back to default agent`,
543
+ )
544
+ return undefined
545
+ }
546
+ return args.mode
547
+ }
548
+
549
+ if (!args.agent) return undefined
550
+ const entry = await Agent.get(args.agent)
551
+ if (!entry) {
552
+ UI.println(
553
+ UI.Style.TEXT_WARNING_BOLD + "!",
554
+ UI.Style.TEXT_NORMAL,
555
+ `agent "${args.agent}" not found. Falling back to default agent`,
556
+ )
557
+ return undefined
558
+ }
559
+ if (entry.mode === "subagent") {
560
+ UI.println(
561
+ UI.Style.TEXT_WARNING_BOLD + "!",
562
+ UI.Style.TEXT_NORMAL,
563
+ `agent "${args.agent}" is a subagent, not a primary agent. Falling back to default agent`,
564
+ )
565
+ return undefined
566
+ }
567
+ return args.agent
568
+ })()
569
+
570
+ const sessionID = await session(sdk)
571
+ if (!sessionID) {
572
+ UI.error("Session not found")
573
+ process.exit(1)
574
+ }
575
+ await share(sdk, sessionID)
576
+
577
+ loop().catch((e) => {
578
+ console.error(e)
579
+ process.exit(1)
580
+ })
581
+
582
+ if (args.command) {
583
+ await sdk.session.command({
584
+ sessionID,
585
+ agent,
586
+ model: args.model,
587
+ command: args.command,
588
+ arguments: message,
589
+ variant: args.variant,
590
+ })
591
+ } else {
592
+ const model = args.model ? Provider.parseModel(args.model) : undefined
593
+ await sdk.session.prompt({
594
+ sessionID,
595
+ agent,
596
+ model,
597
+ variant: args.variant,
598
+ parts: [...files, { type: "text", text: message }],
599
+ })
600
+ }
601
+ }
602
+
603
+ if (args.attach) {
604
+ const sdk = createOpencodeClient({ baseUrl: args.attach })
605
+ return await execute(sdk)
606
+ }
607
+
608
+ await bootstrap(process.cwd(), async () => {
609
+ const fetchFn = (async (input: RequestInfo | URL, init?: RequestInit) => {
610
+ const request = new Request(input, init)
611
+ return Server.App().fetch(request)
612
+ }) as typeof globalThis.fetch
613
+ const sdk = createOpencodeClient({ baseUrl: "http://opencode.internal", fetch: fetchFn })
614
+ await execute(sdk)
615
+ })
616
+ },
617
+ })
@@ -0,0 +1,20 @@
1
+ import { Server } from "../../server/server"
2
+ import { cmd } from "./cmd"
3
+ import { withNetworkOptions, resolveNetworkOptions } from "../network"
4
+ import { Flag } from "../../flag/flag"
5
+
6
+ export const ServeCommand = cmd({
7
+ command: "serve",
8
+ builder: (yargs) => withNetworkOptions(yargs),
9
+ describe: "starts a headless opencode server",
10
+ handler: async (args) => {
11
+ if (!Flag.OPENCODE_SERVER_PASSWORD) {
12
+ console.log("Warning: OPENCODE_SERVER_PASSWORD is not set; server is unsecured.")
13
+ }
14
+ const opts = await resolveNetworkOptions(args)
15
+ const server = Server.listen(opts)
16
+ console.log(`opencode server listening on http://${server.hostname}:${server.port}`)
17
+ await new Promise(() => {})
18
+ await server.stop()
19
+ },
20
+ })