koro-ai 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (670) hide show
  1. package/AGENTS.md +69 -0
  2. package/BUN_SHELL_MIGRATION_PLAN.md +136 -0
  3. package/Dockerfile +18 -0
  4. package/README.md +15 -0
  5. package/bin/koro +16 -0
  6. package/bin/opencode +179 -0
  7. package/bunfig.toml +7 -0
  8. package/drizzle.config.ts +10 -0
  9. package/git +0 -0
  10. package/migration/20260127222353_familiar_lady_ursula/migration.sql +90 -0
  11. package/migration/20260127222353_familiar_lady_ursula/snapshot.json +796 -0
  12. package/migration/20260211171708_add_project_commands/migration.sql +1 -0
  13. package/migration/20260211171708_add_project_commands/snapshot.json +806 -0
  14. package/migration/20260213144116_wakeful_the_professor/migration.sql +11 -0
  15. package/migration/20260213144116_wakeful_the_professor/snapshot.json +897 -0
  16. package/migration/20260225215848_workspace/migration.sql +7 -0
  17. package/migration/20260225215848_workspace/snapshot.json +959 -0
  18. package/migration/20260227213759_add_session_workspace_id/migration.sql +2 -0
  19. package/migration/20260227213759_add_session_workspace_id/snapshot.json +983 -0
  20. package/migration/20260228203230_blue_harpoon/migration.sql +17 -0
  21. package/migration/20260228203230_blue_harpoon/snapshot.json +1102 -0
  22. package/migration/20260303231226_add_workspace_fields/migration.sql +5 -0
  23. package/migration/20260303231226_add_workspace_fields/snapshot.json +1013 -0
  24. package/migration/20260309230000_move_org_to_state/migration.sql +3 -0
  25. package/migration/20260309230000_move_org_to_state/snapshot.json +1156 -0
  26. package/migration/20260312043431_session_message_cursor/migration.sql +4 -0
  27. package/migration/20260312043431_session_message_cursor/snapshot.json +1168 -0
  28. package/migration/20260323234822_events/migration.sql +13 -0
  29. package/migration/20260323234822_events/snapshot.json +1271 -0
  30. package/package.json +163 -0
  31. package/parsers-config.ts +290 -0
  32. package/script/build-node.ts +56 -0
  33. package/script/build.ts +276 -0
  34. package/script/check-migrations.ts +16 -0
  35. package/script/postinstall.mjs +131 -0
  36. package/script/publish.ts +181 -0
  37. package/script/schema.ts +63 -0
  38. package/script/seed-e2e.ts +60 -0
  39. package/script/upgrade-opentui.ts +64 -0
  40. package/specs/effect-migration.md +294 -0
  41. package/specs/tui-plugins.md +436 -0
  42. package/src/account/account.sql.ts +39 -0
  43. package/src/account/index.ts +424 -0
  44. package/src/account/repo.ts +163 -0
  45. package/src/account/schema.ts +91 -0
  46. package/src/acp/README.md +174 -0
  47. package/src/acp/agent.ts +1763 -0
  48. package/src/acp/session.ts +116 -0
  49. package/src/acp/types.ts +24 -0
  50. package/src/agent/agent.ts +476 -0
  51. package/src/agent/generate.txt +75 -0
  52. package/src/agent/prompt/compaction.txt +15 -0
  53. package/src/agent/prompt/explore.txt +18 -0
  54. package/src/agent/prompt/summary.txt +11 -0
  55. package/src/agent/prompt/title.txt +44 -0
  56. package/src/auth/index.ts +109 -0
  57. package/src/bus/bus-event.ts +40 -0
  58. package/src/bus/global.ts +10 -0
  59. package/src/bus/index.ts +185 -0
  60. package/src/cli/bootstrap.ts +17 -0
  61. package/src/cli/cmd/account.ts +257 -0
  62. package/src/cli/cmd/acp.ts +70 -0
  63. package/src/cli/cmd/agent.ts +245 -0
  64. package/src/cli/cmd/cmd.ts +7 -0
  65. package/src/cli/cmd/db.ts +119 -0
  66. package/src/cli/cmd/debug/agent.ts +167 -0
  67. package/src/cli/cmd/debug/config.ts +16 -0
  68. package/src/cli/cmd/debug/file.ts +97 -0
  69. package/src/cli/cmd/debug/index.ts +48 -0
  70. package/src/cli/cmd/debug/lsp.ts +53 -0
  71. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  72. package/src/cli/cmd/debug/scrap.ts +16 -0
  73. package/src/cli/cmd/debug/skill.ts +16 -0
  74. package/src/cli/cmd/debug/snapshot.ts +52 -0
  75. package/src/cli/cmd/export.ts +89 -0
  76. package/src/cli/cmd/generate.ts +38 -0
  77. package/src/cli/cmd/github.ts +1646 -0
  78. package/src/cli/cmd/import.ts +207 -0
  79. package/src/cli/cmd/mcp.ts +754 -0
  80. package/src/cli/cmd/models.ts +78 -0
  81. package/src/cli/cmd/plug.ts +233 -0
  82. package/src/cli/cmd/pr.ts +127 -0
  83. package/src/cli/cmd/providers.ts +478 -0
  84. package/src/cli/cmd/run.ts +676 -0
  85. package/src/cli/cmd/serve.ts +24 -0
  86. package/src/cli/cmd/session.ts +159 -0
  87. package/src/cli/cmd/stats.ts +410 -0
  88. package/src/cli/cmd/tui/app.tsx +919 -0
  89. package/src/cli/cmd/tui/attach.ts +88 -0
  90. package/src/cli/cmd/tui/component/border.tsx +21 -0
  91. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  92. package/src/cli/cmd/tui/component/dialog-command.tsx +171 -0
  93. package/src/cli/cmd/tui/component/dialog-import-share.tsx +118 -0
  94. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  95. package/src/cli/cmd/tui/component/dialog-model.tsx +179 -0
  96. package/src/cli/cmd/tui/component/dialog-provider.tsx +329 -0
  97. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  98. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  99. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  100. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  101. package/src/cli/cmd/tui/component/dialog-status.tsx +168 -0
  102. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  103. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  104. package/src/cli/cmd/tui/component/dialog-variant.tsx +39 -0
  105. package/src/cli/cmd/tui/component/dialog-workspace-list.tsx +320 -0
  106. package/src/cli/cmd/tui/component/error-component.tsx +92 -0
  107. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  108. package/src/cli/cmd/tui/component/plugin-route-missing.tsx +14 -0
  109. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +672 -0
  110. package/src/cli/cmd/tui/component/prompt/frecency.tsx +90 -0
  111. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  112. package/src/cli/cmd/tui/component/prompt/index.tsx +1310 -0
  113. package/src/cli/cmd/tui/component/prompt/part.ts +16 -0
  114. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  115. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  116. package/src/cli/cmd/tui/component/startup-loading.tsx +63 -0
  117. package/src/cli/cmd/tui/component/task-panel.tsx +44 -0
  118. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  119. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  120. package/src/cli/cmd/tui/component/token-bar.tsx +60 -0
  121. package/src/cli/cmd/tui/component/workspace/dialog-session-list.tsx +151 -0
  122. package/src/cli/cmd/tui/context/args.tsx +15 -0
  123. package/src/cli/cmd/tui/context/directory.ts +13 -0
  124. package/src/cli/cmd/tui/context/exit.tsx +60 -0
  125. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  126. package/src/cli/cmd/tui/context/keybind.tsx +105 -0
  127. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  128. package/src/cli/cmd/tui/context/local.tsx +412 -0
  129. package/src/cli/cmd/tui/context/plugin-keybinds.ts +41 -0
  130. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  131. package/src/cli/cmd/tui/context/route.tsx +52 -0
  132. package/src/cli/cmd/tui/context/sdk.tsx +128 -0
  133. package/src/cli/cmd/tui/context/sync.tsx +504 -0
  134. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  135. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  136. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  137. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  138. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  139. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  140. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  141. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  142. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  143. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  144. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  145. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  146. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  147. package/src/cli/cmd/tui/context/theme/gtr.json +245 -0
  148. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  149. package/src/cli/cmd/tui/context/theme/koro.json +241 -0
  150. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  151. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  152. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  153. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  154. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  155. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  156. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  157. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  158. package/src/cli/cmd/tui/context/theme/opencode.json +245 -0
  159. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  160. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  161. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  162. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  163. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  164. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  165. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  166. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  167. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  168. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  169. package/src/cli/cmd/tui/context/theme.tsx +1240 -0
  170. package/src/cli/cmd/tui/context/tui-config.tsx +9 -0
  171. package/src/cli/cmd/tui/event.ts +49 -0
  172. package/src/cli/cmd/tui/feature-plugins/home/footer.tsx +93 -0
  173. package/src/cli/cmd/tui/feature-plugins/home/tips-view.tsx +152 -0
  174. package/src/cli/cmd/tui/feature-plugins/home/tips.tsx +50 -0
  175. package/src/cli/cmd/tui/feature-plugins/sidebar/context.tsx +63 -0
  176. package/src/cli/cmd/tui/feature-plugins/sidebar/files.tsx +62 -0
  177. package/src/cli/cmd/tui/feature-plugins/sidebar/footer.tsx +93 -0
  178. package/src/cli/cmd/tui/feature-plugins/sidebar/lsp.tsx +66 -0
  179. package/src/cli/cmd/tui/feature-plugins/sidebar/mcp.tsx +96 -0
  180. package/src/cli/cmd/tui/feature-plugins/sidebar/todo.tsx +48 -0
  181. package/src/cli/cmd/tui/feature-plugins/system/plugins.tsx +270 -0
  182. package/src/cli/cmd/tui/plugin/api.tsx +430 -0
  183. package/src/cli/cmd/tui/plugin/index.ts +3 -0
  184. package/src/cli/cmd/tui/plugin/internal.ts +27 -0
  185. package/src/cli/cmd/tui/plugin/runtime.ts +1033 -0
  186. package/src/cli/cmd/tui/plugin/slots.tsx +60 -0
  187. package/src/cli/cmd/tui/routes/home.tsx +84 -0
  188. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +65 -0
  189. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +110 -0
  190. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  191. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  192. package/src/cli/cmd/tui/routes/session/footer.tsx +93 -0
  193. package/src/cli/cmd/tui/routes/session/index.tsx +2270 -0
  194. package/src/cli/cmd/tui/routes/session/permission.tsx +691 -0
  195. package/src/cli/cmd/tui/routes/session/question.tsx +468 -0
  196. package/src/cli/cmd/tui/routes/session/sidebar.tsx +74 -0
  197. package/src/cli/cmd/tui/routes/session/subagent-footer.tsx +131 -0
  198. package/src/cli/cmd/tui/thread.ts +232 -0
  199. package/src/cli/cmd/tui/ui/dialog-alert.tsx +59 -0
  200. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +89 -0
  201. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +211 -0
  202. package/src/cli/cmd/tui/ui/dialog-help.tsx +40 -0
  203. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +130 -0
  204. package/src/cli/cmd/tui/ui/dialog-select.tsx +409 -0
  205. package/src/cli/cmd/tui/ui/dialog.tsx +192 -0
  206. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  207. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  208. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  209. package/src/cli/cmd/tui/util/clipboard.ts +192 -0
  210. package/src/cli/cmd/tui/util/editor.ts +37 -0
  211. package/src/cli/cmd/tui/util/model.ts +23 -0
  212. package/src/cli/cmd/tui/util/scroll.ts +23 -0
  213. package/src/cli/cmd/tui/util/selection.ts +25 -0
  214. package/src/cli/cmd/tui/util/signal.ts +7 -0
  215. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  216. package/src/cli/cmd/tui/util/transcript.ts +112 -0
  217. package/src/cli/cmd/tui/win32.ts +129 -0
  218. package/src/cli/cmd/tui/worker.ts +175 -0
  219. package/src/cli/cmd/uninstall.ts +353 -0
  220. package/src/cli/cmd/upgrade.ts +73 -0
  221. package/src/cli/cmd/web.ts +81 -0
  222. package/src/cli/effect/prompt.ts +25 -0
  223. package/src/cli/error.ts +46 -0
  224. package/src/cli/heap.ts +59 -0
  225. package/src/cli/logo.ts +14 -0
  226. package/src/cli/network.ts +60 -0
  227. package/src/cli/ui.ts +133 -0
  228. package/src/cli/upgrade.ts +31 -0
  229. package/src/command/index.ts +195 -0
  230. package/src/command/template/initialize.txt +66 -0
  231. package/src/command/template/review.txt +101 -0
  232. package/src/config/config.ts +1591 -0
  233. package/src/config/markdown.ts +99 -0
  234. package/src/config/paths.ts +181 -0
  235. package/src/config/tui-migrate.ts +155 -0
  236. package/src/config/tui-schema.ts +36 -0
  237. package/src/config/tui.ts +171 -0
  238. package/src/control-plane/adaptors/index.ts +20 -0
  239. package/src/control-plane/adaptors/worktree.ts +38 -0
  240. package/src/control-plane/schema.ts +17 -0
  241. package/src/control-plane/sse.ts +66 -0
  242. package/src/control-plane/types.ts +21 -0
  243. package/src/control-plane/workspace.sql.ts +17 -0
  244. package/src/control-plane/workspace.ts +154 -0
  245. package/src/effect/cross-spawn-spawner.ts +502 -0
  246. package/src/effect/instance-ref.ts +6 -0
  247. package/src/effect/instance-registry.ts +12 -0
  248. package/src/effect/instance-state.ts +82 -0
  249. package/src/effect/run-service.ts +33 -0
  250. package/src/effect/runner.ts +216 -0
  251. package/src/env/index.ts +28 -0
  252. package/src/file/ignore.ts +82 -0
  253. package/src/file/index.ts +686 -0
  254. package/src/file/protected.ts +59 -0
  255. package/src/file/ripgrep.ts +376 -0
  256. package/src/file/time.ts +133 -0
  257. package/src/file/watcher.ts +171 -0
  258. package/src/filesystem/index.ts +226 -0
  259. package/src/flag/flag.ts +155 -0
  260. package/src/format/formatter.ts +413 -0
  261. package/src/format/index.ts +203 -0
  262. package/src/git/index.ts +303 -0
  263. package/src/global/index.ts +161 -0
  264. package/src/id/id.ts +85 -0
  265. package/src/ide/index.ts +74 -0
  266. package/src/index.ts +240 -0
  267. package/src/installation/index.ts +355 -0
  268. package/src/installation/meta.ts +7 -0
  269. package/src/lsp/client.ts +252 -0
  270. package/src/lsp/index.ts +558 -0
  271. package/src/lsp/language.ts +120 -0
  272. package/src/lsp/launch.ts +21 -0
  273. package/src/lsp/server.ts +1958 -0
  274. package/src/mcp/auth.ts +173 -0
  275. package/src/mcp/index.ts +921 -0
  276. package/src/mcp/oauth-callback.ts +215 -0
  277. package/src/mcp/oauth-provider.ts +185 -0
  278. package/src/memory/index.ts +117 -0
  279. package/src/node.ts +1 -0
  280. package/src/npm/index.ts +180 -0
  281. package/src/orchestrator/agent-registry.ts +38 -0
  282. package/src/orchestrator/conflict.ts +25 -0
  283. package/src/orchestrator/context-manager.ts +22 -0
  284. package/src/orchestrator/index.ts +9 -0
  285. package/src/orchestrator/scheduler.ts +30 -0
  286. package/src/orchestrator/state-tracker.ts +71 -0
  287. package/src/orchestrator/task-manager.ts +69 -0
  288. package/src/patch/index.ts +680 -0
  289. package/src/permission/arity.ts +163 -0
  290. package/src/permission/evaluate.ts +15 -0
  291. package/src/permission/index.ts +325 -0
  292. package/src/permission/schema.ts +17 -0
  293. package/src/plugin/codex.ts +596 -0
  294. package/src/plugin/github-copilot/copilot.ts +353 -0
  295. package/src/plugin/github-copilot/models.ts +144 -0
  296. package/src/plugin/index.ts +281 -0
  297. package/src/plugin/install.ts +439 -0
  298. package/src/plugin/loader.ts +174 -0
  299. package/src/plugin/meta.ts +188 -0
  300. package/src/plugin/shared.ts +307 -0
  301. package/src/project/bootstrap.ts +31 -0
  302. package/src/project/instance.ts +175 -0
  303. package/src/project/project.sql.ts +16 -0
  304. package/src/project/project.ts +519 -0
  305. package/src/project/schema.ts +16 -0
  306. package/src/project/state.ts +70 -0
  307. package/src/project/vcs.ts +240 -0
  308. package/src/provider/auth.ts +253 -0
  309. package/src/provider/error.ts +197 -0
  310. package/src/provider/models-snapshot.ts +60410 -0
  311. package/src/provider/models.ts +162 -0
  312. package/src/provider/provider.ts +1677 -0
  313. package/src/provider/schema.ts +38 -0
  314. package/src/provider/sdk/copilot/README.md +5 -0
  315. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +170 -0
  316. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  317. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +19 -0
  318. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  319. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +815 -0
  320. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  321. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  322. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +83 -0
  323. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  324. package/src/provider/sdk/copilot/index.ts +2 -0
  325. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  326. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +335 -0
  327. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  328. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  329. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  330. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +214 -0
  331. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1769 -0
  332. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +173 -0
  333. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  334. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +87 -0
  335. package/src/provider/sdk/copilot/responses/tool/file-search.ts +127 -0
  336. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +114 -0
  337. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +64 -0
  338. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +103 -0
  339. package/src/provider/sdk/copilot/responses/tool/web-search.ts +102 -0
  340. package/src/provider/transform.ts +1046 -0
  341. package/src/pty/index.ts +401 -0
  342. package/src/pty/schema.ts +17 -0
  343. package/src/question/index.ts +224 -0
  344. package/src/question/schema.ts +17 -0
  345. package/src/server/error.ts +36 -0
  346. package/src/server/event.ts +7 -0
  347. package/src/server/instance.ts +314 -0
  348. package/src/server/mdns.ts +60 -0
  349. package/src/server/middleware.ts +33 -0
  350. package/src/server/projectors.ts +28 -0
  351. package/src/server/router.ts +99 -0
  352. package/src/server/routes/config.ts +92 -0
  353. package/src/server/routes/event.ts +83 -0
  354. package/src/server/routes/experimental.ts +271 -0
  355. package/src/server/routes/file.ts +197 -0
  356. package/src/server/routes/global.ts +312 -0
  357. package/src/server/routes/mcp.ts +225 -0
  358. package/src/server/routes/permission.ts +69 -0
  359. package/src/server/routes/project.ts +118 -0
  360. package/src/server/routes/provider.ts +171 -0
  361. package/src/server/routes/pty.ts +211 -0
  362. package/src/server/routes/question.ts +99 -0
  363. package/src/server/routes/session.ts +1031 -0
  364. package/src/server/routes/tui.ts +379 -0
  365. package/src/server/routes/workspace.ts +94 -0
  366. package/src/server/server.ts +312 -0
  367. package/src/session/compaction.ts +428 -0
  368. package/src/session/index.ts +887 -0
  369. package/src/session/instruction.ts +258 -0
  370. package/src/session/llm.ts +370 -0
  371. package/src/session/message-v2.ts +1031 -0
  372. package/src/session/message.ts +191 -0
  373. package/src/session/overflow.ts +22 -0
  374. package/src/session/processor.ts +523 -0
  375. package/src/session/projectors.ts +135 -0
  376. package/src/session/prompt/anthropic.txt +105 -0
  377. package/src/session/prompt/beast.txt +147 -0
  378. package/src/session/prompt/build-switch.txt +5 -0
  379. package/src/session/prompt/codex.txt +79 -0
  380. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  381. package/src/session/prompt/default.txt +105 -0
  382. package/src/session/prompt/gemini.txt +155 -0
  383. package/src/session/prompt/gpt.txt +107 -0
  384. package/src/session/prompt/kimi.txt +114 -0
  385. package/src/session/prompt/max-steps.txt +16 -0
  386. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  387. package/src/session/prompt/plan.txt +26 -0
  388. package/src/session/prompt/trinity.txt +97 -0
  389. package/src/session/prompt.ts +1908 -0
  390. package/src/session/retry.ts +106 -0
  391. package/src/session/revert.ts +176 -0
  392. package/src/session/schema.ts +38 -0
  393. package/src/session/session.sql.ts +103 -0
  394. package/src/session/status.ts +102 -0
  395. package/src/session/summary.ts +177 -0
  396. package/src/session/system.ts +76 -0
  397. package/src/session/todo.ts +95 -0
  398. package/src/share/share-next.ts +369 -0
  399. package/src/share/share.sql.ts +13 -0
  400. package/src/shell/shell.ts +110 -0
  401. package/src/skill/discovery.ts +116 -0
  402. package/src/skill/index.ts +277 -0
  403. package/src/snapshot/index.ts +571 -0
  404. package/src/sql.d.ts +4 -0
  405. package/src/storage/db.bun.ts +8 -0
  406. package/src/storage/db.node.ts +8 -0
  407. package/src/storage/db.ts +174 -0
  408. package/src/storage/json-migration.ts +425 -0
  409. package/src/storage/schema.sql.ts +10 -0
  410. package/src/storage/schema.ts +5 -0
  411. package/src/storage/storage.ts +353 -0
  412. package/src/sync/README.md +179 -0
  413. package/src/sync/event.sql.ts +16 -0
  414. package/src/sync/index.ts +263 -0
  415. package/src/sync/schema.ts +14 -0
  416. package/src/token/index.ts +77 -0
  417. package/src/tool/apply_patch.ts +281 -0
  418. package/src/tool/apply_patch.txt +33 -0
  419. package/src/tool/bash.ts +496 -0
  420. package/src/tool/bash.txt +117 -0
  421. package/src/tool/batch.ts +183 -0
  422. package/src/tool/batch.txt +24 -0
  423. package/src/tool/codesearch.ts +132 -0
  424. package/src/tool/codesearch.txt +12 -0
  425. package/src/tool/edit.ts +667 -0
  426. package/src/tool/edit.txt +10 -0
  427. package/src/tool/external-directory.ts +37 -0
  428. package/src/tool/glob.ts +78 -0
  429. package/src/tool/glob.txt +6 -0
  430. package/src/tool/grep.ts +156 -0
  431. package/src/tool/grep.txt +8 -0
  432. package/src/tool/invalid.ts +17 -0
  433. package/src/tool/ls.ts +121 -0
  434. package/src/tool/ls.txt +1 -0
  435. package/src/tool/lsp.ts +97 -0
  436. package/src/tool/lsp.txt +19 -0
  437. package/src/tool/multiedit.ts +46 -0
  438. package/src/tool/multiedit.txt +41 -0
  439. package/src/tool/plan-enter.txt +14 -0
  440. package/src/tool/plan-exit.txt +13 -0
  441. package/src/tool/plan.ts +131 -0
  442. package/src/tool/question.ts +46 -0
  443. package/src/tool/question.txt +10 -0
  444. package/src/tool/read.ts +296 -0
  445. package/src/tool/read.txt +14 -0
  446. package/src/tool/registry.ts +248 -0
  447. package/src/tool/schema.ts +17 -0
  448. package/src/tool/skill.ts +105 -0
  449. package/src/tool/task.ts +166 -0
  450. package/src/tool/task.txt +60 -0
  451. package/src/tool/todo.ts +48 -0
  452. package/src/tool/todowrite.txt +167 -0
  453. package/src/tool/tool.ts +112 -0
  454. package/src/tool/truncate.ts +144 -0
  455. package/src/tool/truncation-dir.ts +4 -0
  456. package/src/tool/webfetch.ts +206 -0
  457. package/src/tool/webfetch.txt +13 -0
  458. package/src/tool/websearch.ts +150 -0
  459. package/src/tool/websearch.txt +14 -0
  460. package/src/tool/write.ts +84 -0
  461. package/src/tool/write.txt +8 -0
  462. package/src/util/abort.ts +35 -0
  463. package/src/util/archive.ts +17 -0
  464. package/src/util/color.ts +19 -0
  465. package/src/util/context.ts +25 -0
  466. package/src/util/data-url.ts +9 -0
  467. package/src/util/defer.ts +12 -0
  468. package/src/util/effect-http-client.ts +11 -0
  469. package/src/util/effect-zod.ts +98 -0
  470. package/src/util/error.ts +77 -0
  471. package/src/util/filesystem.ts +245 -0
  472. package/src/util/flock.ts +333 -0
  473. package/src/util/fn.ts +21 -0
  474. package/src/util/format.ts +20 -0
  475. package/src/util/glob.ts +34 -0
  476. package/src/util/hash.ts +7 -0
  477. package/src/util/iife.ts +3 -0
  478. package/src/util/keybind.ts +103 -0
  479. package/src/util/lazy.ts +23 -0
  480. package/src/util/locale.ts +81 -0
  481. package/src/util/lock.ts +98 -0
  482. package/src/util/log.ts +182 -0
  483. package/src/util/network.ts +9 -0
  484. package/src/util/process.ts +176 -0
  485. package/src/util/queue.ts +32 -0
  486. package/src/util/record.ts +3 -0
  487. package/src/util/rpc.ts +66 -0
  488. package/src/util/schema.ts +53 -0
  489. package/src/util/scrap.ts +10 -0
  490. package/src/util/signal.ts +12 -0
  491. package/src/util/timeout.ts +14 -0
  492. package/src/util/token.ts +7 -0
  493. package/src/util/update-schema.ts +13 -0
  494. package/src/util/which.ts +14 -0
  495. package/src/util/wildcard.ts +59 -0
  496. package/src/worktree/index.ts +612 -0
  497. package/sst-env.d.ts +10 -0
  498. package/test/AGENTS.md +81 -0
  499. package/test/account/repo.test.ts +326 -0
  500. package/test/account/service.test.ts +393 -0
  501. package/test/acp/agent-interface.test.ts +51 -0
  502. package/test/acp/event-subscription.test.ts +685 -0
  503. package/test/agent/agent.test.ts +717 -0
  504. package/test/auth/auth.test.ts +58 -0
  505. package/test/bus/bus-effect.test.ts +164 -0
  506. package/test/bus/bus-integration.test.ts +87 -0
  507. package/test/bus/bus.test.ts +219 -0
  508. package/test/cli/account.test.ts +26 -0
  509. package/test/cli/cmd/tui/prompt-part.test.ts +47 -0
  510. package/test/cli/github-action.test.ts +198 -0
  511. package/test/cli/github-remote.test.ts +80 -0
  512. package/test/cli/import.test.ts +54 -0
  513. package/test/cli/plugin-auth-picker.test.ts +120 -0
  514. package/test/cli/tui/keybind-plugin.test.ts +90 -0
  515. package/test/cli/tui/plugin-add.test.ts +107 -0
  516. package/test/cli/tui/plugin-install.test.ts +89 -0
  517. package/test/cli/tui/plugin-lifecycle.test.ts +225 -0
  518. package/test/cli/tui/plugin-loader-entrypoint.test.ts +492 -0
  519. package/test/cli/tui/plugin-loader-pure.test.ts +72 -0
  520. package/test/cli/tui/plugin-loader.test.ts +752 -0
  521. package/test/cli/tui/plugin-toggle.test.ts +159 -0
  522. package/test/cli/tui/slot-replace.test.tsx +47 -0
  523. package/test/cli/tui/theme-store.test.ts +51 -0
  524. package/test/cli/tui/thread.test.ts +128 -0
  525. package/test/cli/tui/transcript.test.ts +426 -0
  526. package/test/config/agent-color.test.ts +71 -0
  527. package/test/config/config.test.ts +2348 -0
  528. package/test/config/fixtures/empty-frontmatter.md +4 -0
  529. package/test/config/fixtures/frontmatter.md +28 -0
  530. package/test/config/fixtures/markdown-header.md +11 -0
  531. package/test/config/fixtures/no-frontmatter.md +1 -0
  532. package/test/config/fixtures/weird-model-id.md +13 -0
  533. package/test/config/markdown.test.ts +228 -0
  534. package/test/config/tui.test.ts +752 -0
  535. package/test/control-plane/sse.test.ts +56 -0
  536. package/test/effect/cross-spawn-spawner.test.ts +412 -0
  537. package/test/effect/instance-state.test.ts +482 -0
  538. package/test/effect/run-service.test.ts +46 -0
  539. package/test/effect/runner.test.ts +523 -0
  540. package/test/fake/provider.ts +81 -0
  541. package/test/file/fsmonitor.test.ts +62 -0
  542. package/test/file/ignore.test.ts +10 -0
  543. package/test/file/index.test.ts +946 -0
  544. package/test/file/path-traversal.test.ts +198 -0
  545. package/test/file/ripgrep.test.ts +54 -0
  546. package/test/file/time.test.ts +445 -0
  547. package/test/file/watcher.test.ts +247 -0
  548. package/test/filesystem/filesystem.test.ts +319 -0
  549. package/test/fixture/db.ts +11 -0
  550. package/test/fixture/fixture.test.ts +26 -0
  551. package/test/fixture/fixture.ts +172 -0
  552. package/test/fixture/flock-worker.ts +72 -0
  553. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  554. package/test/fixture/plug-worker.ts +93 -0
  555. package/test/fixture/plugin-meta-worker.ts +26 -0
  556. package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
  557. package/test/fixture/skills/agents-sdk/references/callable.md +92 -0
  558. package/test/fixture/skills/cloudflare/SKILL.md +211 -0
  559. package/test/fixture/skills/index.json +6 -0
  560. package/test/fixture/tui-plugin.ts +337 -0
  561. package/test/fixture/tui-runtime.ts +27 -0
  562. package/test/format/format.test.ts +171 -0
  563. package/test/git/git.test.ts +128 -0
  564. package/test/ide/ide.test.ts +82 -0
  565. package/test/installation/installation.test.ts +151 -0
  566. package/test/keybind.test.ts +421 -0
  567. package/test/lib/effect.ts +53 -0
  568. package/test/lib/filesystem.ts +10 -0
  569. package/test/lib/llm-server.ts +795 -0
  570. package/test/lsp/client.test.ts +95 -0
  571. package/test/lsp/index.test.ts +55 -0
  572. package/test/lsp/launch.test.ts +22 -0
  573. package/test/lsp/lifecycle.test.ts +147 -0
  574. package/test/mcp/headers.test.ts +153 -0
  575. package/test/mcp/lifecycle.test.ts +750 -0
  576. package/test/mcp/oauth-auto-connect.test.ts +199 -0
  577. package/test/mcp/oauth-browser.test.ts +249 -0
  578. package/test/memory/abort-leak.test.ts +137 -0
  579. package/test/patch/patch.test.ts +348 -0
  580. package/test/permission/arity.test.ts +33 -0
  581. package/test/permission/next.test.ts +1148 -0
  582. package/test/permission-task.test.ts +323 -0
  583. package/test/plugin/auth-override.test.ts +74 -0
  584. package/test/plugin/codex.test.ts +123 -0
  585. package/test/plugin/github-copilot-models.test.ts +117 -0
  586. package/test/plugin/install-concurrency.test.ts +140 -0
  587. package/test/plugin/install.test.ts +570 -0
  588. package/test/plugin/loader-shared.test.ts +1136 -0
  589. package/test/plugin/meta.test.ts +137 -0
  590. package/test/plugin/trigger.test.ts +111 -0
  591. package/test/preload.ts +90 -0
  592. package/test/project/migrate-global.test.ts +140 -0
  593. package/test/project/project.test.ts +459 -0
  594. package/test/project/state.test.ts +115 -0
  595. package/test/project/vcs.test.ts +228 -0
  596. package/test/project/worktree-remove.test.ts +96 -0
  597. package/test/project/worktree.test.ts +173 -0
  598. package/test/provider/amazon-bedrock.test.ts +447 -0
  599. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  600. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  601. package/test/provider/gitlab-duo.test.ts +412 -0
  602. package/test/provider/provider.test.ts +2284 -0
  603. package/test/provider/transform.test.ts +2839 -0
  604. package/test/pty/pty-output-isolation.test.ts +141 -0
  605. package/test/pty/pty-session.test.ts +92 -0
  606. package/test/pty/pty-shell.test.ts +59 -0
  607. package/test/question/question.test.ts +453 -0
  608. package/test/server/global-session-list.test.ts +89 -0
  609. package/test/server/project-init-git.test.ts +121 -0
  610. package/test/server/session-actions.test.ts +83 -0
  611. package/test/server/session-list.test.ts +98 -0
  612. package/test/server/session-messages.test.ts +159 -0
  613. package/test/server/session-select.test.ts +84 -0
  614. package/test/session/compaction.test.ts +1212 -0
  615. package/test/session/instruction.test.ts +286 -0
  616. package/test/session/llm.test.ts +1098 -0
  617. package/test/session/message-v2.test.ts +957 -0
  618. package/test/session/messages-pagination.test.ts +885 -0
  619. package/test/session/processor-effect.test.ts +747 -0
  620. package/test/session/prompt-effect.test.ts +1241 -0
  621. package/test/session/prompt.test.ts +518 -0
  622. package/test/session/retry.test.ts +232 -0
  623. package/test/session/revert-compact.test.ts +621 -0
  624. package/test/session/session.test.ts +142 -0
  625. package/test/session/snapshot-tool-race.test.ts +242 -0
  626. package/test/session/structured-output-integration.test.ts +233 -0
  627. package/test/session/structured-output.test.ts +391 -0
  628. package/test/session/system.test.ts +59 -0
  629. package/test/share/share-next.test.ts +333 -0
  630. package/test/shell/shell.test.ts +73 -0
  631. package/test/skill/discovery.test.ts +116 -0
  632. package/test/skill/skill.test.ts +392 -0
  633. package/test/snapshot/snapshot.test.ts +1312 -0
  634. package/test/storage/db.test.ts +14 -0
  635. package/test/storage/json-migration.test.ts +849 -0
  636. package/test/storage/storage.test.ts +295 -0
  637. package/test/sync/index.test.ts +191 -0
  638. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  639. package/test/tool/apply_patch.test.ts +567 -0
  640. package/test/tool/bash.test.ts +1099 -0
  641. package/test/tool/edit.test.ts +681 -0
  642. package/test/tool/external-directory.test.ts +198 -0
  643. package/test/tool/fixtures/large-image.png +0 -0
  644. package/test/tool/fixtures/models-api.json +65179 -0
  645. package/test/tool/grep.test.ts +111 -0
  646. package/test/tool/question.test.ts +126 -0
  647. package/test/tool/read.test.ts +546 -0
  648. package/test/tool/registry.test.ts +126 -0
  649. package/test/tool/skill.test.ts +167 -0
  650. package/test/tool/task.test.ts +49 -0
  651. package/test/tool/tool-define.test.ts +101 -0
  652. package/test/tool/truncation.test.ts +161 -0
  653. package/test/tool/webfetch.test.ts +101 -0
  654. package/test/tool/write.test.ts +353 -0
  655. package/test/util/data-url.test.ts +14 -0
  656. package/test/util/effect-zod.test.ts +61 -0
  657. package/test/util/error.test.ts +38 -0
  658. package/test/util/filesystem.test.ts +656 -0
  659. package/test/util/flock.test.ts +383 -0
  660. package/test/util/format.test.ts +59 -0
  661. package/test/util/glob.test.ts +164 -0
  662. package/test/util/iife.test.ts +36 -0
  663. package/test/util/lazy.test.ts +50 -0
  664. package/test/util/lock.test.ts +72 -0
  665. package/test/util/module.test.ts +59 -0
  666. package/test/util/process.test.ts +128 -0
  667. package/test/util/timeout.test.ts +21 -0
  668. package/test/util/which.test.ts +100 -0
  669. package/test/util/wildcard.test.ts +90 -0
  670. package/tsconfig.json +23 -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,181 @@
1
+ import path from "path"
2
+ import os from "os"
3
+ import { xdgConfig } from "xdg-basedir"
4
+ import z from "zod"
5
+ import { type ParseError as JsoncParseError, parse as parseJsonc, printParseErrorCode } from "jsonc-parser"
6
+ import { NamedError } from "@opencode-ai/util/error"
7
+ import { Filesystem } from "@/util/filesystem"
8
+ import { Flag } from "@/flag/flag"
9
+ import { Global } from "@/global"
10
+
11
+ export namespace ConfigPaths {
12
+ export async function projectFiles(name: string, directory: string, worktree: string) {
13
+ const files = [`${name}.json`, `${name}.jsonc`]
14
+ // Also look for opencode config files for OpenCode compatibility
15
+ if (name === "koro") {
16
+ files.push("opencode.json", "opencode.jsonc")
17
+ }
18
+ return Filesystem.findUp(files, directory, worktree, { rootFirst: true })
19
+ }
20
+
21
+ export async function directories(directory: string, worktree: string) {
22
+ const dirs = [
23
+ Global.Path.config,
24
+ ...(!Flag.OPENCODE_DISABLE_PROJECT_CONFIG
25
+ ? await Array.fromAsync(
26
+ Filesystem.up({
27
+ targets: [".koro", ".opencode"],
28
+ start: directory,
29
+ stop: worktree,
30
+ }),
31
+ )
32
+ : []),
33
+ ...(await Array.fromAsync(
34
+ Filesystem.up({
35
+ targets: [".koro", ".opencode"],
36
+ start: Global.Path.home,
37
+ stop: Global.Path.home,
38
+ }),
39
+ )),
40
+ ...(Flag.OPENCODE_CONFIG_DIR ? [Flag.OPENCODE_CONFIG_DIR] : []),
41
+ ]
42
+
43
+ // Also include OpenCode config directory for plugin/skill compatibility
44
+ const opencodeConfig = path.join(xdgConfig!, "opencode")
45
+ if (opencodeConfig !== Global.Path.config && !dirs.includes(opencodeConfig)) {
46
+ dirs.push(opencodeConfig)
47
+ }
48
+
49
+ return dirs
50
+ }
51
+
52
+ export function fileInDirectory(dir: string, name: string) {
53
+ return [path.join(dir, `${name}.json`), path.join(dir, `${name}.jsonc`)]
54
+ }
55
+
56
+ export const JsonError = NamedError.create(
57
+ "ConfigJsonError",
58
+ z.object({
59
+ path: z.string(),
60
+ message: z.string().optional(),
61
+ }),
62
+ )
63
+
64
+ export const InvalidError = NamedError.create(
65
+ "ConfigInvalidError",
66
+ z.object({
67
+ path: z.string(),
68
+ issues: z.custom<z.core.$ZodIssue[]>().optional(),
69
+ message: z.string().optional(),
70
+ }),
71
+ )
72
+
73
+ /** Read a config file, returning undefined for missing files and throwing JsonError for other failures. */
74
+ export async function readFile(filepath: string) {
75
+ return Filesystem.readText(filepath).catch((err: NodeJS.ErrnoException) => {
76
+ if (err.code === "ENOENT") return
77
+ throw new JsonError({ path: filepath }, { cause: err })
78
+ })
79
+ }
80
+
81
+ type ParseSource = string | { source: string; dir: string }
82
+
83
+ function source(input: ParseSource) {
84
+ return typeof input === "string" ? input : input.source
85
+ }
86
+
87
+ function dir(input: ParseSource) {
88
+ return typeof input === "string" ? path.dirname(input) : input.dir
89
+ }
90
+
91
+ /** Apply {env:VAR} and {file:path} substitutions to config text. */
92
+ async function substitute(text: string, input: ParseSource, missing: "error" | "empty" = "error") {
93
+ text = text.replace(/\{env:([^}]+)\}/g, (_, varName) => {
94
+ return process.env[varName] || ""
95
+ })
96
+
97
+ const fileMatches = Array.from(text.matchAll(/\{file:[^}]+\}/g))
98
+ if (!fileMatches.length) return text
99
+
100
+ const configDir = dir(input)
101
+ const configSource = source(input)
102
+ let out = ""
103
+ let cursor = 0
104
+
105
+ for (const match of fileMatches) {
106
+ const token = match[0]
107
+ const index = match.index!
108
+ out += text.slice(cursor, index)
109
+
110
+ const lineStart = text.lastIndexOf("\n", index - 1) + 1
111
+ const prefix = text.slice(lineStart, index).trimStart()
112
+ if (prefix.startsWith("//")) {
113
+ out += token
114
+ cursor = index + token.length
115
+ continue
116
+ }
117
+
118
+ let filePath = token.replace(/^\{file:/, "").replace(/\}$/, "")
119
+ if (filePath.startsWith("~/")) {
120
+ filePath = path.join(os.homedir(), filePath.slice(2))
121
+ }
122
+
123
+ const resolvedPath = path.isAbsolute(filePath) ? filePath : path.resolve(configDir, filePath)
124
+ const fileContent = (
125
+ await Filesystem.readText(resolvedPath).catch((error: NodeJS.ErrnoException) => {
126
+ if (missing === "empty") return ""
127
+
128
+ const errMsg = `bad file reference: "${token}"`
129
+ if (error.code === "ENOENT") {
130
+ throw new InvalidError(
131
+ {
132
+ path: configSource,
133
+ message: errMsg + ` ${resolvedPath} does not exist`,
134
+ },
135
+ { cause: error },
136
+ )
137
+ }
138
+ throw new InvalidError({ path: configSource, message: errMsg }, { cause: error })
139
+ })
140
+ ).trim()
141
+
142
+ out += JSON.stringify(fileContent).slice(1, -1)
143
+ cursor = index + token.length
144
+ }
145
+
146
+ out += text.slice(cursor)
147
+ return out
148
+ }
149
+
150
+ /** Substitute and parse JSONC text, throwing JsonError on syntax errors. */
151
+ export async function parseText(text: string, input: ParseSource, missing: "error" | "empty" = "error") {
152
+ const configSource = source(input)
153
+ text = await substitute(text, input, missing)
154
+
155
+ const errors: JsoncParseError[] = []
156
+ const data = parseJsonc(text, errors, { allowTrailingComma: true })
157
+ if (errors.length) {
158
+ const lines = text.split("\n")
159
+ const errorDetails = errors
160
+ .map((e) => {
161
+ const beforeOffset = text.substring(0, e.offset).split("\n")
162
+ const line = beforeOffset.length
163
+ const column = beforeOffset[beforeOffset.length - 1].length + 1
164
+ const problemLine = lines[line - 1]
165
+
166
+ const error = `${printParseErrorCode(e.error)} at line ${line}, column ${column}`
167
+ if (!problemLine) return error
168
+
169
+ return `${error}\n Line ${line}: ${problemLine}\n${"".padStart(column + 9)}^`
170
+ })
171
+ .join("\n")
172
+
173
+ throw new JsonError({
174
+ path: configSource,
175
+ message: `\n--- JSONC Input ---\n${text}\n--- Errors ---\n${errorDetails}\n--- End ---`,
176
+ })
177
+ }
178
+
179
+ return data
180
+ }
181
+ }
@@ -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 Filesystem.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 Filesystem.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 Filesystem.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,36 @@
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
+ plugin: Config.PluginSpec.array().optional(),
33
+ plugin_enabled: z.record(z.string(), z.boolean()).optional(),
34
+ })
35
+ .extend(TuiOptions.shape)
36
+ .strict()
@@ -0,0 +1,171 @@
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 "./tui-migrate"
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 { isRecord } from "@/util/record"
12
+ import { Global } from "@/global"
13
+
14
+ export namespace TuiConfig {
15
+ const log = Log.create({ service: "tui.config" })
16
+
17
+ export const Info = TuiInfo
18
+
19
+ type Acc = {
20
+ result: Info
21
+ }
22
+
23
+ export type Info = z.output<typeof Info> & {
24
+ // Internal resolved plugin list used by runtime loading.
25
+ plugin_origins?: Config.PluginOrigin[]
26
+ }
27
+
28
+ function pluginScope(file: string): Config.PluginScope {
29
+ if (Instance.containsPath(file)) return "local"
30
+ return "global"
31
+ }
32
+
33
+ function customPath() {
34
+ return Flag.OPENCODE_TUI_CONFIG
35
+ }
36
+
37
+ function normalize(raw: Record<string, unknown>) {
38
+ const data = { ...raw }
39
+ if (!("tui" in data)) return data
40
+ if (!isRecord(data.tui)) {
41
+ delete data.tui
42
+ return data
43
+ }
44
+
45
+ const tui = data.tui
46
+ delete data.tui
47
+ return {
48
+ ...tui,
49
+ ...data,
50
+ }
51
+ }
52
+
53
+ function installDeps(dir: string): Promise<void> {
54
+ return Config.installDependencies(dir)
55
+ }
56
+
57
+ async function mergeFile(acc: Acc, file: string) {
58
+ const data = await loadFile(file)
59
+ acc.result = mergeDeep(acc.result, data)
60
+ if (!data.plugin?.length) return
61
+
62
+ const scope = pluginScope(file)
63
+ const plugins = Config.deduplicatePluginOrigins([
64
+ ...(acc.result.plugin_origins ?? []),
65
+ ...data.plugin.map((spec) => ({ spec, scope, source: file })),
66
+ ])
67
+ acc.result.plugin = plugins.map((item) => item.spec)
68
+ acc.result.plugin_origins = plugins
69
+ }
70
+
71
+ const state = Instance.state(async () => {
72
+ let projectFiles = Flag.OPENCODE_DISABLE_PROJECT_CONFIG
73
+ ? []
74
+ : await ConfigPaths.projectFiles("tui", Instance.directory, Instance.worktree)
75
+ const directories = await ConfigPaths.directories(Instance.directory, Instance.worktree)
76
+ const custom = customPath()
77
+ const managed = Config.managedConfigDir()
78
+ await migrateTuiConfig({ directories, custom, managed })
79
+ // Re-compute after migration since migrateTuiConfig may have created new tui.json files
80
+ projectFiles = Flag.OPENCODE_DISABLE_PROJECT_CONFIG
81
+ ? []
82
+ : await ConfigPaths.projectFiles("tui", Instance.directory, Instance.worktree)
83
+
84
+ const acc: Acc = {
85
+ result: {},
86
+ }
87
+
88
+ for (const file of ConfigPaths.fileInDirectory(Global.Path.config, "tui")) {
89
+ await mergeFile(acc, file)
90
+ }
91
+
92
+ if (custom) {
93
+ await mergeFile(acc, custom)
94
+ log.debug("loaded custom tui config", { path: custom })
95
+ }
96
+
97
+ for (const file of projectFiles) {
98
+ await mergeFile(acc, file)
99
+ }
100
+
101
+ for (const dir of unique(directories)) {
102
+ if (!dir.endsWith(".opencode") && dir !== Flag.OPENCODE_CONFIG_DIR) continue
103
+ for (const file of ConfigPaths.fileInDirectory(dir, "tui")) {
104
+ await mergeFile(acc, file)
105
+ }
106
+ }
107
+
108
+ if (existsSync(managed)) {
109
+ for (const file of ConfigPaths.fileInDirectory(managed, "tui")) {
110
+ await mergeFile(acc, file)
111
+ }
112
+ }
113
+
114
+ acc.result.keybinds = Config.Keybinds.parse(acc.result.keybinds ?? {})
115
+
116
+ const deps: Promise<void>[] = []
117
+ if (acc.result.plugin?.length) {
118
+ for (const dir of unique(directories)) {
119
+ if (!dir.endsWith(".opencode") && dir !== Flag.OPENCODE_CONFIG_DIR) continue
120
+ deps.push(installDeps(dir))
121
+ }
122
+ }
123
+
124
+ return {
125
+ config: acc.result,
126
+ deps,
127
+ }
128
+ })
129
+
130
+ export async function get() {
131
+ return state().then((x) => x.config)
132
+ }
133
+
134
+ export async function waitForDependencies() {
135
+ const deps = await state().then((x) => x.deps)
136
+ await Promise.all(deps)
137
+ }
138
+
139
+ async function loadFile(filepath: string): Promise<Info> {
140
+ const text = await ConfigPaths.readFile(filepath)
141
+ if (!text) return {}
142
+ return load(text, filepath).catch((error) => {
143
+ log.warn("failed to load tui config", { path: filepath, error })
144
+ return {}
145
+ })
146
+ }
147
+
148
+ async function load(text: string, configFilepath: string): Promise<Info> {
149
+ const raw = await ConfigPaths.parseText(text, configFilepath, "empty")
150
+ if (!isRecord(raw)) return {}
151
+
152
+ // Flatten a nested "tui" key so users who wrote `{ "tui": { ... } }` inside tui.json
153
+ // (mirroring the old opencode.json shape) still get their settings applied.
154
+ const normalized = normalize(raw)
155
+
156
+ const parsed = Info.safeParse(normalized)
157
+ if (!parsed.success) {
158
+ log.warn("invalid tui config", { path: configFilepath, issues: parsed.error.issues })
159
+ return {}
160
+ }
161
+
162
+ const data = parsed.data
163
+ if (data.plugin) {
164
+ for (let i = 0; i < data.plugin.length; i++) {
165
+ data.plugin[i] = await Config.resolvePluginSpec(data.plugin[i], configFilepath)
166
+ }
167
+ }
168
+
169
+ return data
170
+ }
171
+ }
@@ -0,0 +1,20 @@
1
+ import { lazy } from "@/util/lazy"
2
+ import type { Adaptor } from "../types"
3
+
4
+ const ADAPTORS: Record<string, () => Promise<Adaptor>> = {
5
+ worktree: lazy(async () => (await import("./worktree")).WorktreeAdaptor),
6
+ }
7
+
8
+ export function getAdaptor(type: string): Promise<Adaptor> {
9
+ return ADAPTORS[type]()
10
+ }
11
+
12
+ export function installAdaptor(type: string, adaptor: Adaptor) {
13
+ // This is experimental: mostly used for testing right now, but we
14
+ // will likely allow this in the future. Need to figure out the
15
+ // TypeScript story
16
+
17
+ // @ts-expect-error we force the builtin types right now, but we
18
+ // will implement a way to extend the types for custom adaptors
19
+ ADAPTORS[type] = () => adaptor
20
+ }
@@ -0,0 +1,38 @@
1
+ import z from "zod"
2
+ import { Worktree } from "@/worktree"
3
+ import { type Adaptor, WorkspaceInfo } from "../types"
4
+
5
+ const Config = WorkspaceInfo.extend({
6
+ name: WorkspaceInfo.shape.name.unwrap(),
7
+ branch: WorkspaceInfo.shape.branch.unwrap(),
8
+ directory: WorkspaceInfo.shape.directory.unwrap(),
9
+ })
10
+
11
+ type Config = z.infer<typeof Config>
12
+
13
+ export const WorktreeAdaptor: Adaptor = {
14
+ async configure(info) {
15
+ const worktree = await Worktree.makeWorktreeInfo(info.name ?? undefined)
16
+ return {
17
+ ...info,
18
+ name: worktree.name,
19
+ branch: worktree.branch,
20
+ directory: worktree.directory,
21
+ }
22
+ },
23
+ async create(info) {
24
+ const config = Config.parse(info)
25
+ await Worktree.createFromInfo({
26
+ name: config.name,
27
+ directory: config.directory,
28
+ branch: config.branch,
29
+ })
30
+ },
31
+ async remove(info) {
32
+ const config = Config.parse(info)
33
+ await Worktree.remove({ directory: config.directory })
34
+ },
35
+ async fetch(_info, _input: RequestInfo | URL, _init?: RequestInit) {
36
+ throw new Error("fetch not implemented")
37
+ },
38
+ }