@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,99 @@
1
+ import { NamedError } from "@opencode-ai/util/error"
2
+ import matter from "gray-matter"
3
+ import { z } from "zod"
4
+ import { Filesystem } from "../util/filesystem"
5
+
6
+ export namespace ConfigMarkdown {
7
+ export const FILE_REGEX = /(?<![\w`])@(\.?[^\s`,.]*(?:\.[^\s`,.]+)*)/g
8
+ export const SHELL_REGEX = /!`([^`]+)`/g
9
+
10
+ export function files(template: string) {
11
+ return Array.from(template.matchAll(FILE_REGEX))
12
+ }
13
+
14
+ export function shell(template: string) {
15
+ return Array.from(template.matchAll(SHELL_REGEX))
16
+ }
17
+
18
+ // other coding agents like claude code allow invalid yaml in their
19
+ // frontmatter, we need to fallback to a more permissive parser for those cases
20
+ export function fallbackSanitization(content: string): string {
21
+ const match = content.match(/^---\r?\n([\s\S]*?)\r?\n---/)
22
+ if (!match) return content
23
+
24
+ const frontmatter = match[1]
25
+ const lines = frontmatter.split(/\r?\n/)
26
+ const result: string[] = []
27
+
28
+ for (const line of lines) {
29
+ // skip comments and empty lines
30
+ if (line.trim().startsWith("#") || line.trim() === "") {
31
+ result.push(line)
32
+ continue
33
+ }
34
+
35
+ // skip lines that are continuations (indented)
36
+ if (line.match(/^\s+/)) {
37
+ result.push(line)
38
+ continue
39
+ }
40
+
41
+ // match key: value pattern
42
+ const kvMatch = line.match(/^([a-zA-Z_][a-zA-Z0-9_]*)\s*:\s*(.*)$/)
43
+ if (!kvMatch) {
44
+ result.push(line)
45
+ continue
46
+ }
47
+
48
+ const key = kvMatch[1]
49
+ const value = kvMatch[2].trim()
50
+
51
+ // skip if value is empty, already quoted, or uses block scalar
52
+ if (value === "" || value === ">" || value === "|" || value.startsWith('"') || value.startsWith("'")) {
53
+ result.push(line)
54
+ continue
55
+ }
56
+
57
+ // if value contains a colon, convert to block scalar
58
+ if (value.includes(":")) {
59
+ result.push(`${key}: |-`)
60
+ result.push(` ${value}`)
61
+ continue
62
+ }
63
+
64
+ result.push(line)
65
+ }
66
+
67
+ const processed = result.join("\n")
68
+ return content.replace(frontmatter, () => processed)
69
+ }
70
+
71
+ export async function parse(filePath: string) {
72
+ const template = await Filesystem.readText(filePath)
73
+
74
+ try {
75
+ const md = matter(template)
76
+ return md
77
+ } catch {
78
+ try {
79
+ return matter(fallbackSanitization(template))
80
+ } catch (err) {
81
+ throw new FrontmatterError(
82
+ {
83
+ path: filePath,
84
+ message: `${filePath}: Failed to parse YAML frontmatter: ${err instanceof Error ? err.message : String(err)}`,
85
+ },
86
+ { cause: err },
87
+ )
88
+ }
89
+ }
90
+ }
91
+
92
+ export const FrontmatterError = NamedError.create(
93
+ "ConfigFrontmatterError",
94
+ z.object({
95
+ path: z.string(),
96
+ message: z.string(),
97
+ }),
98
+ )
99
+ }
@@ -0,0 +1,155 @@
1
+ import path from "path"
2
+ import { type ParseError as JsoncParseError, applyEdits, modify, parse as parseJsonc } from "jsonc-parser"
3
+ import { unique } from "remeda"
4
+ import z from "zod"
5
+ import { ConfigPaths } from "./paths"
6
+ import { TuiInfo, TuiOptions } from "./tui-schema"
7
+ import { Instance } from "@/project/instance"
8
+ import { Flag } from "@/flag/flag"
9
+ import { Log } from "@/util/log"
10
+ import { Filesystem } from "@/util/filesystem"
11
+ import { Global } from "@/global"
12
+
13
+ const log = Log.create({ service: "tui.migrate" })
14
+
15
+ const TUI_SCHEMA_URL = "https://opencode.ai/tui.json"
16
+
17
+ const LegacyTheme = TuiInfo.shape.theme.optional()
18
+ const LegacyRecord = z.record(z.string(), z.unknown()).optional()
19
+
20
+ const TuiLegacy = z
21
+ .object({
22
+ scroll_speed: TuiOptions.shape.scroll_speed.catch(undefined),
23
+ scroll_acceleration: TuiOptions.shape.scroll_acceleration.catch(undefined),
24
+ diff_style: TuiOptions.shape.diff_style.catch(undefined),
25
+ })
26
+ .strip()
27
+
28
+ interface MigrateInput {
29
+ directories: string[]
30
+ custom?: string
31
+ managed: string
32
+ }
33
+
34
+ /**
35
+ * Migrates tui-specific keys (theme, keybinds, tui) from opencode.json files
36
+ * into dedicated tui.json files. Migration is performed per-directory and
37
+ * skips only locations where a tui.json already exists.
38
+ */
39
+ export async function migrateTuiConfig(input: MigrateInput) {
40
+ const opencode = await opencodeFiles(input)
41
+ for (const file of opencode) {
42
+ const source = await Filesystem.readText(file).catch((error) => {
43
+ log.warn("failed to read config for tui migration", { path: file, error })
44
+ return undefined
45
+ })
46
+ if (!source) continue
47
+ const errors: JsoncParseError[] = []
48
+ const data = parseJsonc(source, errors, { allowTrailingComma: true })
49
+ if (errors.length || !data || typeof data !== "object" || Array.isArray(data)) continue
50
+
51
+ const theme = LegacyTheme.safeParse("theme" in data ? data.theme : undefined)
52
+ const keybinds = LegacyRecord.safeParse("keybinds" in data ? data.keybinds : undefined)
53
+ const legacyTui = LegacyRecord.safeParse("tui" in data ? data.tui : undefined)
54
+ const extracted = {
55
+ theme: theme.success ? theme.data : undefined,
56
+ keybinds: keybinds.success ? keybinds.data : undefined,
57
+ tui: legacyTui.success ? legacyTui.data : undefined,
58
+ }
59
+ const tui = extracted.tui ? normalizeTui(extracted.tui) : undefined
60
+ if (extracted.theme === undefined && extracted.keybinds === undefined && !tui) continue
61
+
62
+ const target = path.join(path.dirname(file), "tui.json")
63
+ const targetExists = await Filesystem.exists(target)
64
+ if (targetExists) continue
65
+
66
+ const payload: Record<string, unknown> = {
67
+ $schema: TUI_SCHEMA_URL,
68
+ }
69
+ if (extracted.theme !== undefined) payload.theme = extracted.theme
70
+ if (extracted.keybinds !== undefined) payload.keybinds = extracted.keybinds
71
+ if (tui) Object.assign(payload, tui)
72
+
73
+ const wrote = await Bun.write(target, JSON.stringify(payload, null, 2))
74
+ .then(() => true)
75
+ .catch((error) => {
76
+ log.warn("failed to write tui migration target", { from: file, to: target, error })
77
+ return false
78
+ })
79
+ if (!wrote) continue
80
+
81
+ const stripped = await backupAndStripLegacy(file, source)
82
+ if (!stripped) {
83
+ log.warn("tui config migrated but source file was not stripped", { from: file, to: target })
84
+ continue
85
+ }
86
+ log.info("migrated tui config", { from: file, to: target })
87
+ }
88
+ }
89
+
90
+ function normalizeTui(data: Record<string, unknown>) {
91
+ const parsed = TuiLegacy.parse(data)
92
+ if (
93
+ parsed.scroll_speed === undefined &&
94
+ parsed.diff_style === undefined &&
95
+ parsed.scroll_acceleration === undefined
96
+ ) {
97
+ return
98
+ }
99
+ return parsed
100
+ }
101
+
102
+ async function backupAndStripLegacy(file: string, source: string) {
103
+ const backup = file + ".tui-migration.bak"
104
+ const hasBackup = await Filesystem.exists(backup)
105
+ const backed = hasBackup
106
+ ? true
107
+ : await Bun.write(backup, source)
108
+ .then(() => true)
109
+ .catch((error) => {
110
+ log.warn("failed to backup source config during tui migration", { path: file, backup, error })
111
+ return false
112
+ })
113
+ if (!backed) return false
114
+
115
+ const text = ["theme", "keybinds", "tui"].reduce((acc, key) => {
116
+ const edits = modify(acc, [key], undefined, {
117
+ formattingOptions: {
118
+ insertSpaces: true,
119
+ tabSize: 2,
120
+ },
121
+ })
122
+ if (!edits.length) return acc
123
+ return applyEdits(acc, edits)
124
+ }, source)
125
+
126
+ return Bun.write(file, text)
127
+ .then(() => {
128
+ log.info("stripped tui keys from server config", { path: file, backup })
129
+ return true
130
+ })
131
+ .catch((error) => {
132
+ log.warn("failed to strip legacy tui keys from server config", { path: file, backup, error })
133
+ return false
134
+ })
135
+ }
136
+
137
+ async function opencodeFiles(input: { directories: string[]; managed: string }) {
138
+ const project = Flag.OPENCODE_DISABLE_PROJECT_CONFIG
139
+ ? []
140
+ : await ConfigPaths.projectFiles("opencode", Instance.directory, Instance.worktree)
141
+ const files = [...project, ...ConfigPaths.fileInDirectory(Global.Path.config, "opencode")]
142
+ for (const dir of unique(input.directories)) {
143
+ files.push(...ConfigPaths.fileInDirectory(dir, "opencode"))
144
+ }
145
+ if (Flag.OPENCODE_CONFIG) files.push(Flag.OPENCODE_CONFIG)
146
+ files.push(...ConfigPaths.fileInDirectory(input.managed, "opencode"))
147
+
148
+ const existing = await Promise.all(
149
+ unique(files).map(async (file) => {
150
+ const ok = await Filesystem.exists(file)
151
+ return ok ? file : undefined
152
+ }),
153
+ )
154
+ return existing.filter((file): file is string => !!file)
155
+ }
@@ -0,0 +1,174 @@
1
+ import path from "path"
2
+ import os from "os"
3
+ import z from "zod"
4
+ import { type ParseError as JsoncParseError, parse as parseJsonc, printParseErrorCode } from "jsonc-parser"
5
+ import { NamedError } from "@opencode-ai/util/error"
6
+ import { Filesystem } from "@/util/filesystem"
7
+ import { Flag } from "@/flag/flag"
8
+ import { Global } from "@/global"
9
+
10
+ export namespace ConfigPaths {
11
+ export async function projectFiles(name: string, directory: string, worktree: string) {
12
+ const files: string[] = []
13
+ for (const file of [`${name}.jsonc`, `${name}.json`]) {
14
+ const found = await Filesystem.findUp(file, directory, worktree)
15
+ for (const resolved of found.toReversed()) {
16
+ files.push(resolved)
17
+ }
18
+ }
19
+ return files
20
+ }
21
+
22
+ export async function directories(directory: string, worktree: string) {
23
+ return [
24
+ Global.Path.config,
25
+ ...(!Flag.OPENCODE_DISABLE_PROJECT_CONFIG
26
+ ? await Array.fromAsync(
27
+ Filesystem.up({
28
+ targets: [".opencode"],
29
+ start: directory,
30
+ stop: worktree,
31
+ }),
32
+ )
33
+ : []),
34
+ ...(await Array.fromAsync(
35
+ Filesystem.up({
36
+ targets: [".opencode"],
37
+ start: Global.Path.home,
38
+ stop: Global.Path.home,
39
+ }),
40
+ )),
41
+ ...(Flag.OPENCODE_CONFIG_DIR ? [Flag.OPENCODE_CONFIG_DIR] : []),
42
+ ]
43
+ }
44
+
45
+ export function fileInDirectory(dir: string, name: string) {
46
+ return [path.join(dir, `${name}.jsonc`), path.join(dir, `${name}.json`)]
47
+ }
48
+
49
+ export const JsonError = NamedError.create(
50
+ "ConfigJsonError",
51
+ z.object({
52
+ path: z.string(),
53
+ message: z.string().optional(),
54
+ }),
55
+ )
56
+
57
+ export const InvalidError = NamedError.create(
58
+ "ConfigInvalidError",
59
+ z.object({
60
+ path: z.string(),
61
+ issues: z.custom<z.core.$ZodIssue[]>().optional(),
62
+ message: z.string().optional(),
63
+ }),
64
+ )
65
+
66
+ /** Read a config file, returning undefined for missing files and throwing JsonError for other failures. */
67
+ export async function readFile(filepath: string) {
68
+ return Filesystem.readText(filepath).catch((err: NodeJS.ErrnoException) => {
69
+ if (err.code === "ENOENT") return
70
+ throw new JsonError({ path: filepath }, { cause: err })
71
+ })
72
+ }
73
+
74
+ type ParseSource = string | { source: string; dir: string }
75
+
76
+ function source(input: ParseSource) {
77
+ return typeof input === "string" ? input : input.source
78
+ }
79
+
80
+ function dir(input: ParseSource) {
81
+ return typeof input === "string" ? path.dirname(input) : input.dir
82
+ }
83
+
84
+ /** Apply {env:VAR} and {file:path} substitutions to config text. */
85
+ async function substitute(text: string, input: ParseSource, missing: "error" | "empty" = "error") {
86
+ text = text.replace(/\{env:([^}]+)\}/g, (_, varName) => {
87
+ return process.env[varName] || ""
88
+ })
89
+
90
+ const fileMatches = Array.from(text.matchAll(/\{file:[^}]+\}/g))
91
+ if (!fileMatches.length) return text
92
+
93
+ const configDir = dir(input)
94
+ const configSource = source(input)
95
+ let out = ""
96
+ let cursor = 0
97
+
98
+ for (const match of fileMatches) {
99
+ const token = match[0]
100
+ const index = match.index!
101
+ out += text.slice(cursor, index)
102
+
103
+ const lineStart = text.lastIndexOf("\n", index - 1) + 1
104
+ const prefix = text.slice(lineStart, index).trimStart()
105
+ if (prefix.startsWith("//")) {
106
+ out += token
107
+ cursor = index + token.length
108
+ continue
109
+ }
110
+
111
+ let filePath = token.replace(/^\{file:/, "").replace(/\}$/, "")
112
+ if (filePath.startsWith("~/")) {
113
+ filePath = path.join(os.homedir(), filePath.slice(2))
114
+ }
115
+
116
+ const resolvedPath = path.isAbsolute(filePath) ? filePath : path.resolve(configDir, filePath)
117
+ const fileContent = (
118
+ await Filesystem.readText(resolvedPath).catch((error: NodeJS.ErrnoException) => {
119
+ if (missing === "empty") return ""
120
+
121
+ const errMsg = `bad file reference: "${token}"`
122
+ if (error.code === "ENOENT") {
123
+ throw new InvalidError(
124
+ {
125
+ path: configSource,
126
+ message: errMsg + ` ${resolvedPath} does not exist`,
127
+ },
128
+ { cause: error },
129
+ )
130
+ }
131
+ throw new InvalidError({ path: configSource, message: errMsg }, { cause: error })
132
+ })
133
+ ).trim()
134
+
135
+ out += JSON.stringify(fileContent).slice(1, -1)
136
+ cursor = index + token.length
137
+ }
138
+
139
+ out += text.slice(cursor)
140
+ return out
141
+ }
142
+
143
+ /** Substitute and parse JSONC text, throwing JsonError on syntax errors. */
144
+ export async function parseText(text: string, input: ParseSource, missing: "error" | "empty" = "error") {
145
+ const configSource = source(input)
146
+ text = await substitute(text, input, missing)
147
+
148
+ const errors: JsoncParseError[] = []
149
+ const data = parseJsonc(text, errors, { allowTrailingComma: true })
150
+ if (errors.length) {
151
+ const lines = text.split("\n")
152
+ const errorDetails = errors
153
+ .map((e) => {
154
+ const beforeOffset = text.substring(0, e.offset).split("\n")
155
+ const line = beforeOffset.length
156
+ const column = beforeOffset[beforeOffset.length - 1].length + 1
157
+ const problemLine = lines[line - 1]
158
+
159
+ const error = `${printParseErrorCode(e.error)} at line ${line}, column ${column}`
160
+ if (!problemLine) return error
161
+
162
+ return `${error}\n Line ${line}: ${problemLine}\n${"".padStart(column + 9)}^`
163
+ })
164
+ .join("\n")
165
+
166
+ throw new JsonError({
167
+ path: configSource,
168
+ message: `\n--- JSONC Input ---\n${text}\n--- Errors ---\n${errorDetails}\n--- End ---`,
169
+ })
170
+ }
171
+
172
+ return data
173
+ }
174
+ }
@@ -0,0 +1,34 @@
1
+ import z from "zod"
2
+ import { Config } from "./config"
3
+
4
+ const KeybindOverride = z
5
+ .object(
6
+ Object.fromEntries(Object.keys(Config.Keybinds.shape).map((key) => [key, z.string().optional()])) as Record<
7
+ string,
8
+ z.ZodOptional<z.ZodString>
9
+ >,
10
+ )
11
+ .strict()
12
+
13
+ export const TuiOptions = z.object({
14
+ scroll_speed: z.number().min(0.001).optional().describe("TUI scroll speed"),
15
+ scroll_acceleration: z
16
+ .object({
17
+ enabled: z.boolean().describe("Enable scroll acceleration"),
18
+ })
19
+ .optional()
20
+ .describe("Scroll acceleration settings"),
21
+ diff_style: z
22
+ .enum(["auto", "stacked"])
23
+ .optional()
24
+ .describe("Control diff rendering style: 'auto' adapts to terminal width, 'stacked' always shows single column"),
25
+ })
26
+
27
+ export const TuiInfo = z
28
+ .object({
29
+ $schema: z.string().optional(),
30
+ theme: z.string().optional(),
31
+ keybinds: KeybindOverride.optional(),
32
+ })
33
+ .extend(TuiOptions.shape)
34
+ .strict()
@@ -0,0 +1,118 @@
1
+ import { existsSync } from "fs"
2
+ import z from "zod"
3
+ import { mergeDeep, unique } from "remeda"
4
+ import { Config } from "./config"
5
+ import { ConfigPaths } from "./paths"
6
+ import { migrateTuiConfig } from "./migrate-tui-config"
7
+ import { TuiInfo } from "./tui-schema"
8
+ import { Instance } from "@/project/instance"
9
+ import { Flag } from "@/flag/flag"
10
+ import { Log } from "@/util/log"
11
+ import { Global } from "@/global"
12
+
13
+ export namespace TuiConfig {
14
+ const log = Log.create({ service: "tui.config" })
15
+
16
+ export const Info = TuiInfo
17
+
18
+ export type Info = z.output<typeof Info>
19
+
20
+ function mergeInfo(target: Info, source: Info): Info {
21
+ return mergeDeep(target, source)
22
+ }
23
+
24
+ function customPath() {
25
+ return Flag.OPENCODE_TUI_CONFIG
26
+ }
27
+
28
+ const state = Instance.state(async () => {
29
+ let projectFiles = Flag.OPENCODE_DISABLE_PROJECT_CONFIG
30
+ ? []
31
+ : await ConfigPaths.projectFiles("tui", Instance.directory, Instance.worktree)
32
+ const directories = await ConfigPaths.directories(Instance.directory, Instance.worktree)
33
+ const custom = customPath()
34
+ const managed = Config.managedConfigDir()
35
+ await migrateTuiConfig({ directories, custom, managed })
36
+ // Re-compute after migration since migrateTuiConfig may have created new tui.json files
37
+ projectFiles = Flag.OPENCODE_DISABLE_PROJECT_CONFIG
38
+ ? []
39
+ : await ConfigPaths.projectFiles("tui", Instance.directory, Instance.worktree)
40
+
41
+ let result: Info = {}
42
+
43
+ for (const file of ConfigPaths.fileInDirectory(Global.Path.config, "tui")) {
44
+ result = mergeInfo(result, await loadFile(file))
45
+ }
46
+
47
+ if (custom) {
48
+ result = mergeInfo(result, await loadFile(custom))
49
+ log.debug("loaded custom tui config", { path: custom })
50
+ }
51
+
52
+ for (const file of projectFiles) {
53
+ result = mergeInfo(result, await loadFile(file))
54
+ }
55
+
56
+ for (const dir of unique(directories)) {
57
+ if (!dir.endsWith(".opencode") && dir !== Flag.OPENCODE_CONFIG_DIR) continue
58
+ for (const file of ConfigPaths.fileInDirectory(dir, "tui")) {
59
+ result = mergeInfo(result, await loadFile(file))
60
+ }
61
+ }
62
+
63
+ if (existsSync(managed)) {
64
+ for (const file of ConfigPaths.fileInDirectory(managed, "tui")) {
65
+ result = mergeInfo(result, await loadFile(file))
66
+ }
67
+ }
68
+
69
+ result.keybinds = Config.Keybinds.parse(result.keybinds ?? {})
70
+
71
+ return {
72
+ config: result,
73
+ }
74
+ })
75
+
76
+ export async function get() {
77
+ return state().then((x) => x.config)
78
+ }
79
+
80
+ async function loadFile(filepath: string): Promise<Info> {
81
+ const text = await ConfigPaths.readFile(filepath)
82
+ if (!text) return {}
83
+ return load(text, filepath).catch((error) => {
84
+ log.warn("failed to load tui config", { path: filepath, error })
85
+ return {}
86
+ })
87
+ }
88
+
89
+ async function load(text: string, configFilepath: string): Promise<Info> {
90
+ const data = await ConfigPaths.parseText(text, configFilepath, "empty")
91
+ if (!data || typeof data !== "object" || Array.isArray(data)) return {}
92
+
93
+ // Flatten a nested "tui" key so users who wrote `{ "tui": { ... } }` inside tui.json
94
+ // (mirroring the old opencode.json shape) still get their settings applied.
95
+ const normalized = (() => {
96
+ const copy = { ...(data as Record<string, unknown>) }
97
+ if (!("tui" in copy)) return copy
98
+ if (!copy.tui || typeof copy.tui !== "object" || Array.isArray(copy.tui)) {
99
+ delete copy.tui
100
+ return copy
101
+ }
102
+ const tui = copy.tui as Record<string, unknown>
103
+ delete copy.tui
104
+ return {
105
+ ...tui,
106
+ ...copy,
107
+ }
108
+ })()
109
+
110
+ const parsed = Info.safeParse(normalized)
111
+ if (!parsed.success) {
112
+ log.warn("invalid tui config", { path: configFilepath, issues: parsed.error.issues })
113
+ return {}
114
+ }
115
+
116
+ return parsed.data
117
+ }
118
+ }
@@ -0,0 +1,22 @@
1
+ import { sqliteTable, text, integer, primaryKey, uniqueIndex } from "drizzle-orm/sqlite-core"
2
+ import { eq } from "drizzle-orm"
3
+ import { Timestamps } from "@/storage/schema.sql"
4
+
5
+ export const ControlAccountTable = sqliteTable(
6
+ "control_account",
7
+ {
8
+ email: text().notNull(),
9
+ url: text().notNull(),
10
+ access_token: text().notNull(),
11
+ refresh_token: text().notNull(),
12
+ token_expiry: integer(),
13
+ active: integer({ mode: "boolean" })
14
+ .notNull()
15
+ .$default(() => false),
16
+ ...Timestamps,
17
+ },
18
+ (table) => [
19
+ primaryKey({ columns: [table.email, table.url] }),
20
+ // uniqueIndex("control_account_active_idx").on(table.email).where(eq(table.active, true)),
21
+ ],
22
+ )
@@ -0,0 +1,67 @@
1
+ import { eq, and } from "drizzle-orm"
2
+ import { Database } from "@/storage/db"
3
+ import { ControlAccountTable } from "./control.sql"
4
+ import z from "zod"
5
+
6
+ export * from "./control.sql"
7
+
8
+ export namespace Control {
9
+ export const Account = z.object({
10
+ email: z.string(),
11
+ url: z.string(),
12
+ })
13
+ export type Account = z.infer<typeof Account>
14
+
15
+ function fromRow(row: (typeof ControlAccountTable)["$inferSelect"]): Account {
16
+ return {
17
+ email: row.email,
18
+ url: row.url,
19
+ }
20
+ }
21
+
22
+ export function account(): Account | undefined {
23
+ const row = Database.use((db) =>
24
+ db.select().from(ControlAccountTable).where(eq(ControlAccountTable.active, true)).get(),
25
+ )
26
+ return row ? fromRow(row) : undefined
27
+ }
28
+
29
+ export async function token(): Promise<string | undefined> {
30
+ const row = Database.use((db) =>
31
+ db.select().from(ControlAccountTable).where(eq(ControlAccountTable.active, true)).get(),
32
+ )
33
+ if (!row) return undefined
34
+ if (row.token_expiry && row.token_expiry > Date.now()) return row.access_token
35
+
36
+ const res = await fetch(`${row.url}/oauth/token`, {
37
+ method: "POST",
38
+ headers: { "Content-Type": "application/x-www-form-urlencoded" },
39
+ body: new URLSearchParams({
40
+ grant_type: "refresh_token",
41
+ refresh_token: row.refresh_token,
42
+ }).toString(),
43
+ })
44
+
45
+ if (!res.ok) return
46
+
47
+ const json = (await res.json()) as {
48
+ access_token: string
49
+ refresh_token?: string
50
+ expires_in?: number
51
+ }
52
+
53
+ Database.use((db) =>
54
+ db
55
+ .update(ControlAccountTable)
56
+ .set({
57
+ access_token: json.access_token,
58
+ refresh_token: json.refresh_token ?? row.refresh_token,
59
+ token_expiry: json.expires_in ? Date.now() + json.expires_in * 1000 : undefined,
60
+ })
61
+ .where(and(eq(ControlAccountTable.email, row.email), eq(ControlAccountTable.url, row.url)))
62
+ .run(),
63
+ )
64
+
65
+ return json.access_token
66
+ }
67
+ }
@@ -0,0 +1,10 @@
1
+ import { WorktreeAdaptor } from "./worktree"
2
+ import type { Config } from "../config"
3
+ import type { Adaptor } from "./types"
4
+
5
+ export function getAdaptor(config: Config): Adaptor {
6
+ switch (config.type) {
7
+ case "worktree":
8
+ return WorktreeAdaptor
9
+ }
10
+ }
@@ -0,0 +1,7 @@
1
+ import type { Config } from "../config"
2
+
3
+ export type Adaptor<T extends Config = Config> = {
4
+ create(from: T, branch?: string | null): Promise<{ config: T; init: () => Promise<void> }>
5
+ remove(from: T): Promise<void>
6
+ request(from: T, method: string, url: string, data?: BodyInit, signal?: AbortSignal): Promise<Response | undefined>
7
+ }