@stonerzju/opencode 1.2.16-offline.1

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