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,612 @@
1
+ import z from "zod"
2
+ import { NamedError } from "@opencode-ai/util/error"
3
+ import { Global } from "../global"
4
+ import { Instance } from "../project/instance"
5
+ import { InstanceBootstrap } from "../project/bootstrap"
6
+ import { Project } from "../project/project"
7
+ import { Database, eq } from "../storage/db"
8
+ import { ProjectTable } from "../project/project.sql"
9
+ import type { ProjectID } from "../project/schema"
10
+ import { Log } from "../util/log"
11
+ import { Slug } from "@opencode-ai/util/slug"
12
+ import { errorMessage } from "../util/error"
13
+ import { BusEvent } from "@/bus/bus-event"
14
+ import { GlobalBus } from "@/bus/global"
15
+ import { Git } from "@/git"
16
+ import { Effect, Layer, Path, Scope, ServiceMap, Stream } from "effect"
17
+ import { ChildProcess, ChildProcessSpawner } from "effect/unstable/process"
18
+ import { NodePath } from "@effect/platform-node"
19
+ import { AppFileSystem } from "@/filesystem"
20
+ import { makeRuntime } from "@/effect/run-service"
21
+ import * as CrossSpawnSpawner from "@/effect/cross-spawn-spawner"
22
+ import { InstanceState } from "@/effect/instance-state"
23
+
24
+ export namespace Worktree {
25
+ const log = Log.create({ service: "worktree" })
26
+
27
+ export const Event = {
28
+ Ready: BusEvent.define(
29
+ "worktree.ready",
30
+ z.object({
31
+ name: z.string(),
32
+ branch: z.string(),
33
+ }),
34
+ ),
35
+ Failed: BusEvent.define(
36
+ "worktree.failed",
37
+ z.object({
38
+ message: z.string(),
39
+ }),
40
+ ),
41
+ }
42
+
43
+ export const Info = z
44
+ .object({
45
+ name: z.string(),
46
+ branch: z.string(),
47
+ directory: z.string(),
48
+ })
49
+ .meta({
50
+ ref: "Worktree",
51
+ })
52
+
53
+ export type Info = z.infer<typeof Info>
54
+
55
+ export const CreateInput = z
56
+ .object({
57
+ name: z.string().optional(),
58
+ startCommand: z
59
+ .string()
60
+ .optional()
61
+ .describe("Additional startup script to run after the project's start command"),
62
+ })
63
+ .meta({
64
+ ref: "WorktreeCreateInput",
65
+ })
66
+
67
+ export type CreateInput = z.infer<typeof CreateInput>
68
+
69
+ export const RemoveInput = z
70
+ .object({
71
+ directory: z.string(),
72
+ })
73
+ .meta({
74
+ ref: "WorktreeRemoveInput",
75
+ })
76
+
77
+ export type RemoveInput = z.infer<typeof RemoveInput>
78
+
79
+ export const ResetInput = z
80
+ .object({
81
+ directory: z.string(),
82
+ })
83
+ .meta({
84
+ ref: "WorktreeResetInput",
85
+ })
86
+
87
+ export type ResetInput = z.infer<typeof ResetInput>
88
+
89
+ export const NotGitError = NamedError.create(
90
+ "WorktreeNotGitError",
91
+ z.object({
92
+ message: z.string(),
93
+ }),
94
+ )
95
+
96
+ export const NameGenerationFailedError = NamedError.create(
97
+ "WorktreeNameGenerationFailedError",
98
+ z.object({
99
+ message: z.string(),
100
+ }),
101
+ )
102
+
103
+ export const CreateFailedError = NamedError.create(
104
+ "WorktreeCreateFailedError",
105
+ z.object({
106
+ message: z.string(),
107
+ }),
108
+ )
109
+
110
+ export const StartCommandFailedError = NamedError.create(
111
+ "WorktreeStartCommandFailedError",
112
+ z.object({
113
+ message: z.string(),
114
+ }),
115
+ )
116
+
117
+ export const RemoveFailedError = NamedError.create(
118
+ "WorktreeRemoveFailedError",
119
+ z.object({
120
+ message: z.string(),
121
+ }),
122
+ )
123
+
124
+ export const ResetFailedError = NamedError.create(
125
+ "WorktreeResetFailedError",
126
+ z.object({
127
+ message: z.string(),
128
+ }),
129
+ )
130
+
131
+ function slugify(input: string) {
132
+ return input
133
+ .trim()
134
+ .toLowerCase()
135
+ .replace(/[^a-z0-9]+/g, "-")
136
+ .replace(/^-+/, "")
137
+ .replace(/-+$/, "")
138
+ }
139
+
140
+ function failedRemoves(...chunks: string[]) {
141
+ return chunks.filter(Boolean).flatMap((chunk) =>
142
+ chunk
143
+ .split("\n")
144
+ .map((line) => line.trim())
145
+ .flatMap((line) => {
146
+ const match = line.match(/^warning:\s+failed to remove\s+(.+):\s+/i)
147
+ if (!match) return []
148
+ const value = match[1]?.trim().replace(/^['"]|['"]$/g, "")
149
+ if (!value) return []
150
+ return [value]
151
+ }),
152
+ )
153
+ }
154
+
155
+ // ---------------------------------------------------------------------------
156
+ // Effect service
157
+ // ---------------------------------------------------------------------------
158
+
159
+ export interface Interface {
160
+ readonly makeWorktreeInfo: (name?: string) => Effect.Effect<Info>
161
+ readonly createFromInfo: (info: Info, startCommand?: string) => Effect.Effect<void>
162
+ readonly create: (input?: CreateInput) => Effect.Effect<Info>
163
+ readonly remove: (input: RemoveInput) => Effect.Effect<boolean>
164
+ readonly reset: (input: ResetInput) => Effect.Effect<boolean>
165
+ }
166
+
167
+ export class Service extends ServiceMap.Service<Service, Interface>()("@opencode/Worktree") {}
168
+
169
+ type GitResult = { code: number; text: string; stderr: string }
170
+
171
+ export const layer: Layer.Layer<
172
+ Service,
173
+ never,
174
+ AppFileSystem.Service | Path.Path | ChildProcessSpawner.ChildProcessSpawner | Project.Service
175
+ > = Layer.effect(
176
+ Service,
177
+ Effect.gen(function* () {
178
+ const scope = yield* Scope.Scope
179
+ const fs = yield* AppFileSystem.Service
180
+ const pathSvc = yield* Path.Path
181
+ const spawner = yield* ChildProcessSpawner.ChildProcessSpawner
182
+ const project = yield* Project.Service
183
+
184
+ const git = Effect.fnUntraced(
185
+ function* (args: string[], opts?: { cwd?: string }) {
186
+ const handle = yield* spawner.spawn(
187
+ ChildProcess.make("git", args, { cwd: opts?.cwd, extendEnv: true, stdin: "ignore" }),
188
+ )
189
+ const [text, stderr] = yield* Effect.all(
190
+ [Stream.mkString(Stream.decodeText(handle.stdout)), Stream.mkString(Stream.decodeText(handle.stderr))],
191
+ { concurrency: 2 },
192
+ )
193
+ const code = yield* handle.exitCode
194
+ return { code, text, stderr } satisfies GitResult
195
+ },
196
+ Effect.scoped,
197
+ Effect.catch((e) =>
198
+ Effect.succeed({ code: 1, text: "", stderr: e instanceof Error ? e.message : String(e) } satisfies GitResult),
199
+ ),
200
+ )
201
+
202
+ const MAX_NAME_ATTEMPTS = 26
203
+ const candidate = Effect.fn("Worktree.candidate")(function* (root: string, base?: string) {
204
+ const ctx = yield* InstanceState.context
205
+ for (const attempt of Array.from({ length: MAX_NAME_ATTEMPTS }, (_, i) => i)) {
206
+ const name = base ? (attempt === 0 ? base : `${base}-${Slug.create()}`) : Slug.create()
207
+ const branch = `opencode/${name}`
208
+ const directory = pathSvc.join(root, name)
209
+
210
+ if (yield* fs.exists(directory).pipe(Effect.orDie)) continue
211
+
212
+ const ref = `refs/heads/${branch}`
213
+ const branchCheck = yield* git(["show-ref", "--verify", "--quiet", ref], { cwd: ctx.worktree })
214
+ if (branchCheck.code === 0) continue
215
+
216
+ return Info.parse({ name, branch, directory })
217
+ }
218
+ throw new NameGenerationFailedError({ message: "Failed to generate a unique worktree name" })
219
+ })
220
+
221
+ const makeWorktreeInfo = Effect.fn("Worktree.makeWorktreeInfo")(function* (name?: string) {
222
+ const ctx = yield* InstanceState.context
223
+ if (ctx.project.vcs !== "git") {
224
+ throw new NotGitError({ message: "Worktrees are only supported for git projects" })
225
+ }
226
+
227
+ const root = pathSvc.join(Global.Path.data, "worktree", ctx.project.id)
228
+ yield* fs.makeDirectory(root, { recursive: true }).pipe(Effect.orDie)
229
+
230
+ const base = name ? slugify(name) : ""
231
+ return yield* candidate(root, base || undefined)
232
+ })
233
+
234
+ const setup = Effect.fnUntraced(function* (info: Info) {
235
+ const ctx = yield* InstanceState.context
236
+ const created = yield* git(["worktree", "add", "--no-checkout", "-b", info.branch, info.directory], {
237
+ cwd: ctx.worktree,
238
+ })
239
+ if (created.code !== 0) {
240
+ throw new CreateFailedError({ message: created.stderr || created.text || "Failed to create git worktree" })
241
+ }
242
+
243
+ yield* project.addSandbox(ctx.project.id, info.directory).pipe(Effect.catch(() => Effect.void))
244
+ })
245
+
246
+ const boot = Effect.fnUntraced(function* (info: Info, startCommand?: string) {
247
+ const ctx = yield* InstanceState.context
248
+ const projectID = ctx.project.id
249
+ const extra = startCommand?.trim()
250
+
251
+ const populated = yield* git(["reset", "--hard"], { cwd: info.directory })
252
+ if (populated.code !== 0) {
253
+ const message = populated.stderr || populated.text || "Failed to populate worktree"
254
+ log.error("worktree checkout failed", { directory: info.directory, message })
255
+ GlobalBus.emit("event", {
256
+ directory: info.directory,
257
+ payload: { type: Event.Failed.type, properties: { message } },
258
+ })
259
+ return
260
+ }
261
+
262
+ const booted = yield* Effect.promise(() =>
263
+ Instance.provide({
264
+ directory: info.directory,
265
+ init: InstanceBootstrap,
266
+ fn: () => undefined,
267
+ })
268
+ .then(() => true)
269
+ .catch((error) => {
270
+ const message = errorMessage(error)
271
+ log.error("worktree bootstrap failed", { directory: info.directory, message })
272
+ GlobalBus.emit("event", {
273
+ directory: info.directory,
274
+ payload: { type: Event.Failed.type, properties: { message } },
275
+ })
276
+ return false
277
+ }),
278
+ )
279
+ if (!booted) return
280
+
281
+ GlobalBus.emit("event", {
282
+ directory: info.directory,
283
+ payload: {
284
+ type: Event.Ready.type,
285
+ properties: { name: info.name, branch: info.branch },
286
+ },
287
+ })
288
+
289
+ yield* runStartScripts(info.directory, { projectID, extra })
290
+ })
291
+
292
+ const createFromInfo = Effect.fn("Worktree.createFromInfo")(function* (info: Info, startCommand?: string) {
293
+ yield* setup(info)
294
+ yield* boot(info, startCommand)
295
+ })
296
+
297
+ const create = Effect.fn("Worktree.create")(function* (input?: CreateInput) {
298
+ const info = yield* makeWorktreeInfo(input?.name)
299
+ yield* setup(info)
300
+ yield* boot(info, input?.startCommand).pipe(
301
+ Effect.catchCause((cause) => Effect.sync(() => log.error("worktree bootstrap failed", { cause }))),
302
+ Effect.forkIn(scope),
303
+ )
304
+ return info
305
+ })
306
+
307
+ const canonical = Effect.fnUntraced(function* (input: string) {
308
+ const abs = pathSvc.resolve(input)
309
+ const real = yield* fs.realPath(abs).pipe(Effect.catch(() => Effect.succeed(abs)))
310
+ const normalized = pathSvc.normalize(real)
311
+ return process.platform === "win32" ? normalized.toLowerCase() : normalized
312
+ })
313
+
314
+ function parseWorktreeList(text: string) {
315
+ return text
316
+ .split("\n")
317
+ .map((line) => line.trim())
318
+ .reduce<{ path?: string; branch?: string }[]>((acc, line) => {
319
+ if (!line) return acc
320
+ if (line.startsWith("worktree ")) {
321
+ acc.push({ path: line.slice("worktree ".length).trim() })
322
+ return acc
323
+ }
324
+ const current = acc[acc.length - 1]
325
+ if (!current) return acc
326
+ if (line.startsWith("branch ")) {
327
+ current.branch = line.slice("branch ".length).trim()
328
+ }
329
+ return acc
330
+ }, [])
331
+ }
332
+
333
+ const locateWorktree = Effect.fnUntraced(function* (
334
+ entries: { path?: string; branch?: string }[],
335
+ directory: string,
336
+ ) {
337
+ for (const item of entries) {
338
+ if (!item.path) continue
339
+ const key = yield* canonical(item.path)
340
+ if (key === directory) return item
341
+ }
342
+ return undefined
343
+ })
344
+
345
+ function stopFsmonitor(target: string) {
346
+ return fs.exists(target).pipe(
347
+ Effect.orDie,
348
+ Effect.flatMap((exists) => (exists ? git(["fsmonitor--daemon", "stop"], { cwd: target }) : Effect.void)),
349
+ )
350
+ }
351
+
352
+ function cleanDirectory(target: string) {
353
+ return Effect.promise(() =>
354
+ import("fs/promises")
355
+ .then((fsp) => fsp.rm(target, { recursive: true, force: true, maxRetries: 5, retryDelay: 100 }))
356
+ .catch((error) => {
357
+ const message = errorMessage(error)
358
+ throw new RemoveFailedError({ message: message || "Failed to remove git worktree directory" })
359
+ }),
360
+ )
361
+ }
362
+
363
+ const remove = Effect.fn("Worktree.remove")(function* (input: RemoveInput) {
364
+ if (Instance.project.vcs !== "git") {
365
+ throw new NotGitError({ message: "Worktrees are only supported for git projects" })
366
+ }
367
+
368
+ const directory = yield* canonical(input.directory)
369
+
370
+ const list = yield* git(["worktree", "list", "--porcelain"], { cwd: Instance.worktree })
371
+ if (list.code !== 0) {
372
+ throw new RemoveFailedError({ message: list.stderr || list.text || "Failed to read git worktrees" })
373
+ }
374
+
375
+ const entries = parseWorktreeList(list.text)
376
+ const entry = yield* locateWorktree(entries, directory)
377
+
378
+ if (!entry?.path) {
379
+ const directoryExists = yield* fs.exists(directory).pipe(Effect.orDie)
380
+ if (directoryExists) {
381
+ yield* stopFsmonitor(directory)
382
+ yield* cleanDirectory(directory)
383
+ }
384
+ return true
385
+ }
386
+
387
+ yield* stopFsmonitor(entry.path)
388
+ const removed = yield* git(["worktree", "remove", "--force", entry.path], { cwd: Instance.worktree })
389
+ if (removed.code !== 0) {
390
+ const next = yield* git(["worktree", "list", "--porcelain"], { cwd: Instance.worktree })
391
+ if (next.code !== 0) {
392
+ throw new RemoveFailedError({
393
+ message: removed.stderr || removed.text || next.stderr || next.text || "Failed to remove git worktree",
394
+ })
395
+ }
396
+
397
+ const stale = yield* locateWorktree(parseWorktreeList(next.text), directory)
398
+ if (stale?.path) {
399
+ throw new RemoveFailedError({ message: removed.stderr || removed.text || "Failed to remove git worktree" })
400
+ }
401
+ }
402
+
403
+ yield* cleanDirectory(entry.path)
404
+
405
+ const branch = entry.branch?.replace(/^refs\/heads\//, "")
406
+ if (branch) {
407
+ const deleted = yield* git(["branch", "-D", branch], { cwd: Instance.worktree })
408
+ if (deleted.code !== 0) {
409
+ throw new RemoveFailedError({
410
+ message: deleted.stderr || deleted.text || "Failed to delete worktree branch",
411
+ })
412
+ }
413
+ }
414
+
415
+ return true
416
+ })
417
+
418
+ const gitExpect = Effect.fnUntraced(function* (
419
+ args: string[],
420
+ opts: { cwd: string },
421
+ error: (r: GitResult) => Error,
422
+ ) {
423
+ const result = yield* git(args, opts)
424
+ if (result.code !== 0) throw error(result)
425
+ return result
426
+ })
427
+
428
+ const runStartCommand = Effect.fnUntraced(
429
+ function* (directory: string, cmd: string) {
430
+ const [shell, args] = process.platform === "win32" ? ["cmd", ["/c", cmd]] : ["bash", ["-lc", cmd]]
431
+ const handle = yield* spawner.spawn(
432
+ ChildProcess.make(shell, args, { cwd: directory, extendEnv: true, stdin: "ignore" }),
433
+ )
434
+ // Drain stdout, capture stderr for error reporting
435
+ const [, stderr] = yield* Effect.all(
436
+ [Stream.runDrain(handle.stdout), Stream.mkString(Stream.decodeText(handle.stderr))],
437
+ { concurrency: 2 },
438
+ ).pipe(Effect.orDie)
439
+ const code = yield* handle.exitCode
440
+ return { code, stderr }
441
+ },
442
+ Effect.scoped,
443
+ Effect.catch(() => Effect.succeed({ code: 1, stderr: "" })),
444
+ )
445
+
446
+ const runStartScript = Effect.fnUntraced(function* (directory: string, cmd: string, kind: string) {
447
+ const text = cmd.trim()
448
+ if (!text) return true
449
+ const result = yield* runStartCommand(directory, text)
450
+ if (result.code === 0) return true
451
+ log.error("worktree start command failed", { kind, directory, message: result.stderr })
452
+ return false
453
+ })
454
+
455
+ const runStartScripts = Effect.fnUntraced(function* (
456
+ directory: string,
457
+ input: { projectID: ProjectID; extra?: string },
458
+ ) {
459
+ const row = yield* Effect.sync(() =>
460
+ Database.use((db) => db.select().from(ProjectTable).where(eq(ProjectTable.id, input.projectID)).get()),
461
+ )
462
+ const project = row ? Project.fromRow(row) : undefined
463
+ const startup = project?.commands?.start?.trim() ?? ""
464
+ const ok = yield* runStartScript(directory, startup, "project")
465
+ if (!ok) return false
466
+ yield* runStartScript(directory, input.extra ?? "", "worktree")
467
+ return true
468
+ })
469
+
470
+ const prune = Effect.fnUntraced(function* (root: string, entries: string[]) {
471
+ const base = yield* canonical(root)
472
+ yield* Effect.forEach(
473
+ entries,
474
+ (entry) =>
475
+ Effect.gen(function* () {
476
+ const target = yield* canonical(pathSvc.resolve(root, entry))
477
+ if (target === base) return
478
+ if (!target.startsWith(`${base}${pathSvc.sep}`)) return
479
+ yield* fs.remove(target, { recursive: true }).pipe(Effect.ignore)
480
+ }),
481
+ { concurrency: "unbounded" },
482
+ )
483
+ })
484
+
485
+ const sweep = Effect.fnUntraced(function* (root: string) {
486
+ const first = yield* git(["clean", "-ffdx"], { cwd: root })
487
+ if (first.code === 0) return first
488
+
489
+ const entries = failedRemoves(first.stderr, first.text)
490
+ if (!entries.length) return first
491
+
492
+ yield* prune(root, entries)
493
+ return yield* git(["clean", "-ffdx"], { cwd: root })
494
+ })
495
+
496
+ const reset = Effect.fn("Worktree.reset")(function* (input: ResetInput) {
497
+ if (Instance.project.vcs !== "git") {
498
+ throw new NotGitError({ message: "Worktrees are only supported for git projects" })
499
+ }
500
+
501
+ const directory = yield* canonical(input.directory)
502
+ const primary = yield* canonical(Instance.worktree)
503
+ if (directory === primary) {
504
+ throw new ResetFailedError({ message: "Cannot reset the primary workspace" })
505
+ }
506
+
507
+ const list = yield* git(["worktree", "list", "--porcelain"], { cwd: Instance.worktree })
508
+ if (list.code !== 0) {
509
+ throw new ResetFailedError({ message: list.stderr || list.text || "Failed to read git worktrees" })
510
+ }
511
+
512
+ const entry = yield* locateWorktree(parseWorktreeList(list.text), directory)
513
+ if (!entry?.path) {
514
+ throw new ResetFailedError({ message: "Worktree not found" })
515
+ }
516
+
517
+ const worktreePath = entry.path
518
+
519
+ const base = yield* Effect.promise(() => Git.defaultBranch(Instance.worktree))
520
+ if (!base) {
521
+ throw new ResetFailedError({ message: "Default branch not found" })
522
+ }
523
+
524
+ const sep = base.ref.indexOf("/")
525
+ if (base.ref !== base.name && sep > 0) {
526
+ const remote = base.ref.slice(0, sep)
527
+ const branch = base.ref.slice(sep + 1)
528
+ yield* gitExpect(
529
+ ["fetch", remote, branch],
530
+ { cwd: Instance.worktree },
531
+ (r) => new ResetFailedError({ message: r.stderr || r.text || `Failed to fetch ${base.ref}` }),
532
+ )
533
+ }
534
+
535
+ yield* gitExpect(
536
+ ["reset", "--hard", base.ref],
537
+ { cwd: worktreePath },
538
+ (r) => new ResetFailedError({ message: r.stderr || r.text || "Failed to reset worktree to target" }),
539
+ )
540
+
541
+ const cleanResult = yield* sweep(worktreePath)
542
+ if (cleanResult.code !== 0) {
543
+ throw new ResetFailedError({ message: cleanResult.stderr || cleanResult.text || "Failed to clean worktree" })
544
+ }
545
+
546
+ yield* gitExpect(
547
+ ["submodule", "update", "--init", "--recursive", "--force"],
548
+ { cwd: worktreePath },
549
+ (r) => new ResetFailedError({ message: r.stderr || r.text || "Failed to update submodules" }),
550
+ )
551
+
552
+ yield* gitExpect(
553
+ ["submodule", "foreach", "--recursive", "git", "reset", "--hard"],
554
+ { cwd: worktreePath },
555
+ (r) => new ResetFailedError({ message: r.stderr || r.text || "Failed to reset submodules" }),
556
+ )
557
+
558
+ yield* gitExpect(
559
+ ["submodule", "foreach", "--recursive", "git", "clean", "-fdx"],
560
+ { cwd: worktreePath },
561
+ (r) => new ResetFailedError({ message: r.stderr || r.text || "Failed to clean submodules" }),
562
+ )
563
+
564
+ const status = yield* git(["-c", "core.fsmonitor=false", "status", "--porcelain=v1"], { cwd: worktreePath })
565
+ if (status.code !== 0) {
566
+ throw new ResetFailedError({ message: status.stderr || status.text || "Failed to read git status" })
567
+ }
568
+
569
+ if (status.text.trim()) {
570
+ throw new ResetFailedError({ message: `Worktree reset left local changes:\n${status.text.trim()}` })
571
+ }
572
+
573
+ yield* runStartScripts(worktreePath, { projectID: Instance.project.id }).pipe(
574
+ Effect.catchCause((cause) => Effect.sync(() => log.error("worktree start task failed", { cause }))),
575
+ Effect.forkIn(scope),
576
+ )
577
+
578
+ return true
579
+ })
580
+
581
+ return Service.of({ makeWorktreeInfo, createFromInfo, create, remove, reset })
582
+ }),
583
+ )
584
+
585
+ const defaultLayer = layer.pipe(
586
+ Layer.provide(CrossSpawnSpawner.defaultLayer),
587
+ Layer.provide(Project.defaultLayer),
588
+ Layer.provide(AppFileSystem.defaultLayer),
589
+ Layer.provide(NodePath.layer),
590
+ )
591
+ const { runPromise } = makeRuntime(Service, defaultLayer)
592
+
593
+ export async function makeWorktreeInfo(name?: string) {
594
+ return runPromise((svc) => svc.makeWorktreeInfo(name))
595
+ }
596
+
597
+ export async function createFromInfo(info: Info, startCommand?: string) {
598
+ return runPromise((svc) => svc.createFromInfo(info, startCommand))
599
+ }
600
+
601
+ export async function create(input?: CreateInput) {
602
+ return runPromise((svc) => svc.create(input))
603
+ }
604
+
605
+ export async function remove(input: RemoveInput) {
606
+ return runPromise((svc) => svc.remove(input))
607
+ }
608
+
609
+ export async function reset(input: ResetInput) {
610
+ return runPromise((svc) => svc.reset(input))
611
+ }
612
+ }
package/sst-env.d.ts ADDED
@@ -0,0 +1,10 @@
1
+ /* This file is auto-generated by SST. Do not edit. */
2
+ /* tslint:disable */
3
+ /* eslint-disable */
4
+ /* deno-fmt-ignore-file */
5
+ /* biome-ignore-all lint: auto-generated */
6
+
7
+ /// <reference path="../../sst-env.d.ts" />
8
+
9
+ import "sst"
10
+ export {}
package/test/AGENTS.md ADDED
@@ -0,0 +1,81 @@
1
+ # Test Fixtures Guide
2
+
3
+ ## Temporary Directory Fixture
4
+
5
+ The `tmpdir` function in `fixture/fixture.ts` creates temporary directories for tests with automatic cleanup.
6
+
7
+ ### Basic Usage
8
+
9
+ ```typescript
10
+ import { tmpdir } from "./fixture/fixture"
11
+
12
+ test("example", async () => {
13
+ await using tmp = await tmpdir()
14
+ // tmp.path is the temp directory path
15
+ // automatically cleaned up when test ends
16
+ })
17
+ ```
18
+
19
+ ### Options
20
+
21
+ - `git?: boolean` - Initialize a git repo with a root commit
22
+ - `config?: Partial<Config.Info>` - Write an `opencode.json` config file
23
+ - `init?: (dir: string) => Promise<T>` - Custom setup function, returns value accessible as `tmp.extra`
24
+ - `dispose?: (dir: string) => Promise<T>` - Custom cleanup function
25
+
26
+ ### Examples
27
+
28
+ **Git repository:**
29
+
30
+ ```typescript
31
+ await using tmp = await tmpdir({ git: true })
32
+ ```
33
+
34
+ **With config file:**
35
+
36
+ ```typescript
37
+ await using tmp = await tmpdir({
38
+ config: { model: "test/model", username: "testuser" },
39
+ })
40
+ ```
41
+
42
+ **Custom initialization (returns extra data):**
43
+
44
+ ```typescript
45
+ await using tmp = await tmpdir<string>({
46
+ init: async (dir) => {
47
+ await Bun.write(path.join(dir, "file.txt"), "content")
48
+ return "extra data"
49
+ },
50
+ })
51
+ // Access extra data via tmp.extra
52
+ console.log(tmp.extra) // "extra data"
53
+ ```
54
+
55
+ **With cleanup:**
56
+
57
+ ```typescript
58
+ await using tmp = await tmpdir({
59
+ init: async (dir) => {
60
+ const specialDir = path.join(dir, "special")
61
+ await fs.mkdir(specialDir)
62
+ return specialDir
63
+ },
64
+ dispose: async (dir) => {
65
+ // Custom cleanup logic
66
+ await fs.rm(path.join(dir, "special"), { recursive: true })
67
+ },
68
+ })
69
+ ```
70
+
71
+ ### Returned Object
72
+
73
+ - `path: string` - Absolute path to the temp directory (realpath resolved)
74
+ - `extra: T` - Value returned by the `init` function
75
+ - `[Symbol.asyncDispose]` - Enables automatic cleanup via `await using`
76
+
77
+ ### Notes
78
+
79
+ - Directories are created in the system temp folder with prefix `opencode-test-`
80
+ - Use `await using` for automatic cleanup when the variable goes out of scope
81
+ - Paths are sanitized to strip null bytes (defensive fix for CI environments)