@bloxystudios/bloxycode 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 (344) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +256 -0
  3. package/bin/bloxycode +84 -0
  4. package/package.json +133 -0
  5. package/src/acp/README.md +164 -0
  6. package/src/acp/agent.ts +1437 -0
  7. package/src/acp/session.ts +105 -0
  8. package/src/acp/types.ts +22 -0
  9. package/src/agent/agent.ts +356 -0
  10. package/src/agent/generate.txt +75 -0
  11. package/src/agent/prompt/bloxy.txt +46 -0
  12. package/src/agent/prompt/compaction.txt +12 -0
  13. package/src/agent/prompt/explore.txt +18 -0
  14. package/src/agent/prompt/summary.txt +11 -0
  15. package/src/agent/prompt/title.txt +44 -0
  16. package/src/auth/index.ts +73 -0
  17. package/src/bloxy/event.ts +41 -0
  18. package/src/bloxy/index.ts +5 -0
  19. package/src/bloxy/parser.ts +263 -0
  20. package/src/bloxy/prompt.ts +121 -0
  21. package/src/bloxy/runner.ts +193 -0
  22. package/src/bloxy/state.ts +246 -0
  23. package/src/bun/index.ts +134 -0
  24. package/src/bus/bus-event.ts +43 -0
  25. package/src/bus/global.ts +10 -0
  26. package/src/bus/index.ts +105 -0
  27. package/src/cli/bootstrap.ts +17 -0
  28. package/src/cli/cmd/acp.ts +69 -0
  29. package/src/cli/cmd/agent.ts +257 -0
  30. package/src/cli/cmd/auth.ts +400 -0
  31. package/src/cli/cmd/cmd.ts +7 -0
  32. package/src/cli/cmd/debug/agent.ts +167 -0
  33. package/src/cli/cmd/debug/config.ts +16 -0
  34. package/src/cli/cmd/debug/file.ts +97 -0
  35. package/src/cli/cmd/debug/index.ts +48 -0
  36. package/src/cli/cmd/debug/lsp.ts +52 -0
  37. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  38. package/src/cli/cmd/debug/scrap.ts +16 -0
  39. package/src/cli/cmd/debug/skill.ts +16 -0
  40. package/src/cli/cmd/debug/snapshot.ts +52 -0
  41. package/src/cli/cmd/export.ts +88 -0
  42. package/src/cli/cmd/generate.ts +38 -0
  43. package/src/cli/cmd/github.ts +1548 -0
  44. package/src/cli/cmd/import.ts +98 -0
  45. package/src/cli/cmd/mcp.ts +755 -0
  46. package/src/cli/cmd/models.ts +77 -0
  47. package/src/cli/cmd/pr.ts +112 -0
  48. package/src/cli/cmd/run.ts +395 -0
  49. package/src/cli/cmd/serve.ts +20 -0
  50. package/src/cli/cmd/session.ts +135 -0
  51. package/src/cli/cmd/stats.ts +402 -0
  52. package/src/cli/cmd/tui/app.tsx +771 -0
  53. package/src/cli/cmd/tui/attach.ts +39 -0
  54. package/src/cli/cmd/tui/component/border.tsx +21 -0
  55. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  56. package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
  57. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  58. package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
  59. package/src/cli/cmd/tui/component/dialog-provider.tsx +256 -0
  60. package/src/cli/cmd/tui/component/dialog-session-list.tsx +114 -0
  61. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  62. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  63. package/src/cli/cmd/tui/component/dialog-status.tsx +164 -0
  64. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  65. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  66. package/src/cli/cmd/tui/component/logo.tsx +102 -0
  67. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +653 -0
  68. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  69. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  70. package/src/cli/cmd/tui/component/prompt/index.tsx +1138 -0
  71. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  72. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  73. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  74. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  75. package/src/cli/cmd/tui/context/args.tsx +14 -0
  76. package/src/cli/cmd/tui/context/directory.ts +13 -0
  77. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  78. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  79. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  80. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  81. package/src/cli/cmd/tui/context/local.tsx +402 -0
  82. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  83. package/src/cli/cmd/tui/context/route.tsx +46 -0
  84. package/src/cli/cmd/tui/context/sdk.tsx +94 -0
  85. package/src/cli/cmd/tui/context/sync.tsx +470 -0
  86. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  87. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  88. package/src/cli/cmd/tui/context/theme/bloxycode.json +245 -0
  89. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  90. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  91. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  92. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  93. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  94. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  95. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  96. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  97. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  98. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  99. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  100. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  101. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  102. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  103. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  104. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  105. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  106. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  107. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  108. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  109. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  110. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  111. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  112. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  113. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  114. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  115. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  116. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  117. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  118. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  119. package/src/cli/cmd/tui/context/theme.tsx +1152 -0
  120. package/src/cli/cmd/tui/event.ts +48 -0
  121. package/src/cli/cmd/tui/routes/home.tsx +140 -0
  122. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  123. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  124. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  125. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  126. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  127. package/src/cli/cmd/tui/routes/session/header.tsx +142 -0
  128. package/src/cli/cmd/tui/routes/session/index.tsx +2048 -0
  129. package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
  130. package/src/cli/cmd/tui/routes/session/question.tsx +453 -0
  131. package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
  132. package/src/cli/cmd/tui/thread.ts +165 -0
  133. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  134. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  135. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +204 -0
  136. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  137. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  138. package/src/cli/cmd/tui/ui/dialog-select.tsx +385 -0
  139. package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
  140. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  141. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  142. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  143. package/src/cli/cmd/tui/util/clipboard.ts +160 -0
  144. package/src/cli/cmd/tui/util/editor.ts +32 -0
  145. package/src/cli/cmd/tui/util/signal.ts +7 -0
  146. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  147. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  148. package/src/cli/cmd/tui/worker.ts +152 -0
  149. package/src/cli/cmd/uninstall.ts +357 -0
  150. package/src/cli/cmd/upgrade.ts +73 -0
  151. package/src/cli/cmd/web.ts +81 -0
  152. package/src/cli/error.ts +57 -0
  153. package/src/cli/network.ts +53 -0
  154. package/src/cli/ui.ts +86 -0
  155. package/src/cli/upgrade.ts +25 -0
  156. package/src/command/index.ts +173 -0
  157. package/src/command/template/bloxy-resume.txt +15 -0
  158. package/src/command/template/bloxy-status.txt +25 -0
  159. package/src/command/template/bloxy-validate.txt +22 -0
  160. package/src/command/template/bloxy.txt +14 -0
  161. package/src/command/template/initialize.txt +10 -0
  162. package/src/command/template/review.txt +99 -0
  163. package/src/config/config.ts +1367 -0
  164. package/src/config/markdown.ts +93 -0
  165. package/src/env/index.ts +26 -0
  166. package/src/file/ignore.ts +83 -0
  167. package/src/file/index.ts +415 -0
  168. package/src/file/ripgrep.ts +407 -0
  169. package/src/file/time.ts +69 -0
  170. package/src/file/watcher.ts +127 -0
  171. package/src/flag/flag.ts +79 -0
  172. package/src/format/formatter.ts +357 -0
  173. package/src/format/index.ts +137 -0
  174. package/src/global/index.ts +55 -0
  175. package/src/id/id.ts +83 -0
  176. package/src/ide/index.ts +76 -0
  177. package/src/index.ts +159 -0
  178. package/src/installation/index.ts +246 -0
  179. package/src/lsp/client.ts +252 -0
  180. package/src/lsp/index.ts +485 -0
  181. package/src/lsp/language.ts +119 -0
  182. package/src/lsp/server.ts +2046 -0
  183. package/src/mcp/auth.ts +135 -0
  184. package/src/mcp/index.ts +934 -0
  185. package/src/mcp/oauth-callback.ts +200 -0
  186. package/src/mcp/oauth-provider.ts +154 -0
  187. package/src/patch/index.ts +680 -0
  188. package/src/permission/arity.ts +163 -0
  189. package/src/permission/index.ts +210 -0
  190. package/src/permission/next.ts +280 -0
  191. package/src/plugin/antigravity.ts +378 -0
  192. package/src/plugin/codex.ts +506 -0
  193. package/src/plugin/copilot.ts +298 -0
  194. package/src/plugin/index.ts +136 -0
  195. package/src/project/bootstrap.ts +35 -0
  196. package/src/project/instance.ts +91 -0
  197. package/src/project/project.ts +371 -0
  198. package/src/project/state.ts +66 -0
  199. package/src/project/vcs.ts +76 -0
  200. package/src/provider/auth.ts +147 -0
  201. package/src/provider/models-snapshot.ts +2 -0
  202. package/src/provider/models.ts +133 -0
  203. package/src/provider/provider.ts +1241 -0
  204. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  205. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  206. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  207. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  208. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
  209. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  210. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  211. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  212. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1732 -0
  213. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  214. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  215. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  216. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  217. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  218. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  219. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  220. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  221. package/src/provider/transform.ts +741 -0
  222. package/src/pty/index.ts +241 -0
  223. package/src/question/index.ts +171 -0
  224. package/src/scheduler/index.ts +61 -0
  225. package/src/server/error.ts +36 -0
  226. package/src/server/event.ts +7 -0
  227. package/src/server/mdns.ts +59 -0
  228. package/src/server/routes/config.ts +92 -0
  229. package/src/server/routes/experimental.ts +208 -0
  230. package/src/server/routes/file.ts +197 -0
  231. package/src/server/routes/global.ts +135 -0
  232. package/src/server/routes/mcp.ts +225 -0
  233. package/src/server/routes/permission.ts +68 -0
  234. package/src/server/routes/project.ts +82 -0
  235. package/src/server/routes/provider.ts +165 -0
  236. package/src/server/routes/pty.ts +169 -0
  237. package/src/server/routes/question.ts +98 -0
  238. package/src/server/routes/session.ts +939 -0
  239. package/src/server/routes/tui.ts +379 -0
  240. package/src/server/server.ts +604 -0
  241. package/src/session/compaction.ts +225 -0
  242. package/src/session/fallback.ts +246 -0
  243. package/src/session/index.ts +498 -0
  244. package/src/session/instruction.ts +164 -0
  245. package/src/session/llm.ts +298 -0
  246. package/src/session/message-v2.ts +747 -0
  247. package/src/session/message.ts +189 -0
  248. package/src/session/processor.ts +450 -0
  249. package/src/session/prompt/anthropic-20250930.txt +166 -0
  250. package/src/session/prompt/anthropic.txt +105 -0
  251. package/src/session/prompt/beast.txt +147 -0
  252. package/src/session/prompt/build-switch.txt +5 -0
  253. package/src/session/prompt/codex_header.txt +79 -0
  254. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  255. package/src/session/prompt/gemini.txt +155 -0
  256. package/src/session/prompt/max-steps.txt +16 -0
  257. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  258. package/src/session/prompt/plan.txt +26 -0
  259. package/src/session/prompt/qwen.txt +109 -0
  260. package/src/session/prompt.ts +1822 -0
  261. package/src/session/retry.ts +99 -0
  262. package/src/session/revert.ts +121 -0
  263. package/src/session/status.ts +100 -0
  264. package/src/session/summary.ts +217 -0
  265. package/src/session/system.ts +52 -0
  266. package/src/session/todo.ts +37 -0
  267. package/src/share/share-next.ts +200 -0
  268. package/src/share/share.ts +92 -0
  269. package/src/shell/shell.ts +67 -0
  270. package/src/skill/index.ts +1 -0
  271. package/src/skill/skill.ts +135 -0
  272. package/src/snapshot/index.ts +236 -0
  273. package/src/storage/storage.ts +227 -0
  274. package/src/tool/apply_patch.ts +281 -0
  275. package/src/tool/apply_patch.txt +33 -0
  276. package/src/tool/bash.ts +258 -0
  277. package/src/tool/bash.txt +115 -0
  278. package/src/tool/batch.ts +175 -0
  279. package/src/tool/batch.txt +24 -0
  280. package/src/tool/bloxy-control.ts +123 -0
  281. package/src/tool/bloxy-control.txt +13 -0
  282. package/src/tool/codesearch.ts +132 -0
  283. package/src/tool/codesearch.txt +12 -0
  284. package/src/tool/edit.ts +655 -0
  285. package/src/tool/edit.txt +10 -0
  286. package/src/tool/external-directory.ts +32 -0
  287. package/src/tool/glob.ts +77 -0
  288. package/src/tool/glob.txt +6 -0
  289. package/src/tool/grep.ts +154 -0
  290. package/src/tool/grep.txt +8 -0
  291. package/src/tool/invalid.ts +17 -0
  292. package/src/tool/ls.ts +121 -0
  293. package/src/tool/ls.txt +1 -0
  294. package/src/tool/lsp.ts +96 -0
  295. package/src/tool/lsp.txt +19 -0
  296. package/src/tool/multiedit.ts +46 -0
  297. package/src/tool/multiedit.txt +41 -0
  298. package/src/tool/plan-enter.txt +14 -0
  299. package/src/tool/plan-exit.txt +13 -0
  300. package/src/tool/plan.ts +130 -0
  301. package/src/tool/question.ts +33 -0
  302. package/src/tool/question.txt +10 -0
  303. package/src/tool/read.ts +211 -0
  304. package/src/tool/read.txt +12 -0
  305. package/src/tool/registry.ts +161 -0
  306. package/src/tool/skill.ts +82 -0
  307. package/src/tool/task.ts +191 -0
  308. package/src/tool/task.txt +60 -0
  309. package/src/tool/todo.ts +53 -0
  310. package/src/tool/todoread.txt +14 -0
  311. package/src/tool/todowrite.txt +167 -0
  312. package/src/tool/tool.ts +89 -0
  313. package/src/tool/truncation.ts +106 -0
  314. package/src/tool/webfetch.ts +188 -0
  315. package/src/tool/webfetch.txt +13 -0
  316. package/src/tool/websearch.ts +150 -0
  317. package/src/tool/websearch.txt +14 -0
  318. package/src/tool/write.ts +85 -0
  319. package/src/tool/write.txt +8 -0
  320. package/src/util/archive.ts +16 -0
  321. package/src/util/binary.ts +41 -0
  322. package/src/util/color.ts +19 -0
  323. package/src/util/context.ts +25 -0
  324. package/src/util/defer.ts +12 -0
  325. package/src/util/error.ts +54 -0
  326. package/src/util/eventloop.ts +20 -0
  327. package/src/util/filesystem.ts +93 -0
  328. package/src/util/fn.ts +11 -0
  329. package/src/util/format.ts +20 -0
  330. package/src/util/iife.ts +3 -0
  331. package/src/util/keybind.ts +103 -0
  332. package/src/util/lazy.ts +23 -0
  333. package/src/util/locale.ts +81 -0
  334. package/src/util/lock.ts +98 -0
  335. package/src/util/log.ts +180 -0
  336. package/src/util/queue.ts +32 -0
  337. package/src/util/rpc.ts +66 -0
  338. package/src/util/scrap.ts +10 -0
  339. package/src/util/signal.ts +12 -0
  340. package/src/util/slug.ts +74 -0
  341. package/src/util/timeout.ts +14 -0
  342. package/src/util/token.ts +7 -0
  343. package/src/util/wildcard.ts +56 -0
  344. package/src/worktree/index.ts +549 -0
@@ -0,0 +1,549 @@
1
+ import { $ } from "bun"
2
+ import fs from "fs/promises"
3
+ import path from "path"
4
+ import z from "zod"
5
+ import { NamedError } from "@/util/error"
6
+ import { Global } from "../global"
7
+ import { Instance } from "../project/instance"
8
+ import { InstanceBootstrap } from "../project/bootstrap"
9
+ import { Project } from "../project/project"
10
+ import { Storage } from "../storage/storage"
11
+ import { fn } from "../util/fn"
12
+ import { Log } from "../util/log"
13
+ import { BusEvent } from "@/bus/bus-event"
14
+ import { GlobalBus } from "@/bus/global"
15
+
16
+ export namespace Worktree {
17
+ const log = Log.create({ service: "worktree" })
18
+
19
+ export const Event = {
20
+ Ready: BusEvent.define(
21
+ "worktree.ready",
22
+ z.object({
23
+ name: z.string(),
24
+ branch: z.string(),
25
+ }),
26
+ ),
27
+ Failed: BusEvent.define(
28
+ "worktree.failed",
29
+ z.object({
30
+ message: z.string(),
31
+ }),
32
+ ),
33
+ }
34
+
35
+ export const Info = z
36
+ .object({
37
+ name: z.string(),
38
+ branch: z.string(),
39
+ directory: z.string(),
40
+ })
41
+ .meta({
42
+ ref: "Worktree",
43
+ })
44
+
45
+ export type Info = z.infer<typeof Info>
46
+
47
+ export const CreateInput = z
48
+ .object({
49
+ name: z.string().optional(),
50
+ startCommand: z
51
+ .string()
52
+ .optional()
53
+ .describe("Additional startup script to run after the project's start command"),
54
+ })
55
+ .meta({
56
+ ref: "WorktreeCreateInput",
57
+ })
58
+
59
+ export type CreateInput = z.infer<typeof CreateInput>
60
+
61
+ export const RemoveInput = z
62
+ .object({
63
+ directory: z.string(),
64
+ })
65
+ .meta({
66
+ ref: "WorktreeRemoveInput",
67
+ })
68
+
69
+ export type RemoveInput = z.infer<typeof RemoveInput>
70
+
71
+ export const ResetInput = z
72
+ .object({
73
+ directory: z.string(),
74
+ })
75
+ .meta({
76
+ ref: "WorktreeResetInput",
77
+ })
78
+
79
+ export type ResetInput = z.infer<typeof ResetInput>
80
+
81
+ export const NotGitError = NamedError.create(
82
+ "WorktreeNotGitError",
83
+ z.object({
84
+ message: z.string(),
85
+ }),
86
+ )
87
+
88
+ export const NameGenerationFailedError = NamedError.create(
89
+ "WorktreeNameGenerationFailedError",
90
+ z.object({
91
+ message: z.string(),
92
+ }),
93
+ )
94
+
95
+ export const CreateFailedError = NamedError.create(
96
+ "WorktreeCreateFailedError",
97
+ z.object({
98
+ message: z.string(),
99
+ }),
100
+ )
101
+
102
+ export const StartCommandFailedError = NamedError.create(
103
+ "WorktreeStartCommandFailedError",
104
+ z.object({
105
+ message: z.string(),
106
+ }),
107
+ )
108
+
109
+ export const RemoveFailedError = NamedError.create(
110
+ "WorktreeRemoveFailedError",
111
+ z.object({
112
+ message: z.string(),
113
+ }),
114
+ )
115
+
116
+ export const ResetFailedError = NamedError.create(
117
+ "WorktreeResetFailedError",
118
+ z.object({
119
+ message: z.string(),
120
+ }),
121
+ )
122
+
123
+ const ADJECTIVES = [
124
+ "brave",
125
+ "calm",
126
+ "clever",
127
+ "cosmic",
128
+ "crisp",
129
+ "curious",
130
+ "eager",
131
+ "gentle",
132
+ "glowing",
133
+ "happy",
134
+ "hidden",
135
+ "jolly",
136
+ "kind",
137
+ "lucky",
138
+ "mighty",
139
+ "misty",
140
+ "neon",
141
+ "nimble",
142
+ "playful",
143
+ "proud",
144
+ "quick",
145
+ "quiet",
146
+ "shiny",
147
+ "silent",
148
+ "stellar",
149
+ "sunny",
150
+ "swift",
151
+ "tidy",
152
+ "witty",
153
+ ] as const
154
+
155
+ const NOUNS = [
156
+ "cabin",
157
+ "cactus",
158
+ "canyon",
159
+ "circuit",
160
+ "comet",
161
+ "eagle",
162
+ "engine",
163
+ "falcon",
164
+ "forest",
165
+ "garden",
166
+ "harbor",
167
+ "island",
168
+ "knight",
169
+ "lagoon",
170
+ "meadow",
171
+ "moon",
172
+ "mountain",
173
+ "nebula",
174
+ "orchid",
175
+ "otter",
176
+ "panda",
177
+ "pixel",
178
+ "planet",
179
+ "river",
180
+ "rocket",
181
+ "sailor",
182
+ "squid",
183
+ "star",
184
+ "tiger",
185
+ "wizard",
186
+ "wolf",
187
+ ] as const
188
+
189
+ function pick<const T extends readonly string[]>(list: T) {
190
+ return list[Math.floor(Math.random() * list.length)]
191
+ }
192
+
193
+ function slug(input: string) {
194
+ return input
195
+ .trim()
196
+ .toLowerCase()
197
+ .replace(/[^a-z0-9]+/g, "-")
198
+ .replace(/^-+/, "")
199
+ .replace(/-+$/, "")
200
+ }
201
+
202
+ function randomName() {
203
+ return `${pick(ADJECTIVES)}-${pick(NOUNS)}`
204
+ }
205
+
206
+ async function exists(target: string) {
207
+ return fs
208
+ .stat(target)
209
+ .then(() => true)
210
+ .catch(() => false)
211
+ }
212
+
213
+ function outputText(input: Uint8Array | undefined) {
214
+ if (!input?.length) return ""
215
+ return new TextDecoder().decode(input).trim()
216
+ }
217
+
218
+ function errorText(result: { stdout?: Uint8Array; stderr?: Uint8Array }) {
219
+ return [outputText(result.stderr), outputText(result.stdout)].filter(Boolean).join("\n")
220
+ }
221
+
222
+ async function candidate(root: string, base?: string) {
223
+ for (const attempt of Array.from({ length: 26 }, (_, i) => i)) {
224
+ const name = base ? (attempt === 0 ? base : `${base}-${randomName()}`) : randomName()
225
+ const branch = `opencode/${name}`
226
+ const directory = path.join(root, name)
227
+
228
+ if (await exists(directory)) continue
229
+
230
+ const ref = `refs/heads/${branch}`
231
+ const branchCheck = await $`git show-ref --verify --quiet ${ref}`.quiet().nothrow().cwd(Instance.worktree)
232
+ if (branchCheck.exitCode === 0) continue
233
+
234
+ return Info.parse({ name, branch, directory })
235
+ }
236
+
237
+ throw new NameGenerationFailedError({ message: "Failed to generate a unique worktree name" })
238
+ }
239
+
240
+ async function runStartCommand(directory: string, cmd: string) {
241
+ if (process.platform === "win32") {
242
+ return $`cmd /c ${cmd}`.nothrow().cwd(directory)
243
+ }
244
+ return $`bash -lc ${cmd}`.nothrow().cwd(directory)
245
+ }
246
+
247
+ type StartKind = "project" | "worktree"
248
+
249
+ async function runStartScript(directory: string, cmd: string, kind: StartKind) {
250
+ const text = cmd.trim()
251
+ if (!text) return true
252
+
253
+ const ran = await runStartCommand(directory, text)
254
+ if (ran.exitCode === 0) return true
255
+
256
+ log.error("worktree start command failed", {
257
+ kind,
258
+ directory,
259
+ message: errorText(ran),
260
+ })
261
+ return false
262
+ }
263
+
264
+ async function runStartScripts(directory: string, input: { projectID: string; extra?: string }) {
265
+ const project = await Storage.read<Project.Info>(["project", input.projectID]).catch(() => undefined)
266
+ const startup = project?.commands?.start?.trim() ?? ""
267
+ const ok = await runStartScript(directory, startup, "project")
268
+ if (!ok) return false
269
+
270
+ const extra = input.extra ?? ""
271
+ await runStartScript(directory, extra, "worktree")
272
+ return true
273
+ }
274
+
275
+ function queueStartScripts(directory: string, input: { projectID: string; extra?: string }) {
276
+ setTimeout(() => {
277
+ const start = async () => {
278
+ await runStartScripts(directory, input)
279
+ }
280
+
281
+ void start().catch((error) => {
282
+ log.error("worktree start task failed", { directory, error })
283
+ })
284
+ }, 0)
285
+ }
286
+
287
+ export const create = fn(CreateInput.optional(), async (input) => {
288
+ if (Instance.project.vcs !== "git") {
289
+ throw new NotGitError({ message: "Worktrees are only supported for git projects" })
290
+ }
291
+
292
+ const root = path.join(Global.Path.data, "worktree", Instance.project.id)
293
+ await fs.mkdir(root, { recursive: true })
294
+
295
+ const base = input?.name ? slug(input.name) : ""
296
+ const info = await candidate(root, base || undefined)
297
+
298
+ const created = await $`git worktree add --no-checkout -b ${info.branch} ${info.directory}`
299
+ .quiet()
300
+ .nothrow()
301
+ .cwd(Instance.worktree)
302
+ if (created.exitCode !== 0) {
303
+ throw new CreateFailedError({ message: errorText(created) || "Failed to create git worktree" })
304
+ }
305
+
306
+ await Project.addSandbox(Instance.project.id, info.directory).catch(() => undefined)
307
+
308
+ const projectID = Instance.project.id
309
+ const extra = input?.startCommand?.trim()
310
+ setTimeout(() => {
311
+ const start = async () => {
312
+ const populated = await $`git reset --hard`.quiet().nothrow().cwd(info.directory)
313
+ if (populated.exitCode !== 0) {
314
+ const message = errorText(populated) || "Failed to populate worktree"
315
+ log.error("worktree checkout failed", { directory: info.directory, message })
316
+ GlobalBus.emit("event", {
317
+ directory: info.directory,
318
+ payload: {
319
+ type: Event.Failed.type,
320
+ properties: {
321
+ message,
322
+ },
323
+ },
324
+ })
325
+ return
326
+ }
327
+
328
+ const booted = await Instance.provide({
329
+ directory: info.directory,
330
+ init: InstanceBootstrap,
331
+ fn: () => undefined,
332
+ })
333
+ .then(() => true)
334
+ .catch((error) => {
335
+ const message = error instanceof Error ? error.message : String(error)
336
+ log.error("worktree bootstrap failed", { directory: info.directory, message })
337
+ GlobalBus.emit("event", {
338
+ directory: info.directory,
339
+ payload: {
340
+ type: Event.Failed.type,
341
+ properties: {
342
+ message,
343
+ },
344
+ },
345
+ })
346
+ return false
347
+ })
348
+ if (!booted) return
349
+
350
+ GlobalBus.emit("event", {
351
+ directory: info.directory,
352
+ payload: {
353
+ type: Event.Ready.type,
354
+ properties: {
355
+ name: info.name,
356
+ branch: info.branch,
357
+ },
358
+ },
359
+ })
360
+
361
+ await runStartScripts(info.directory, { projectID, extra })
362
+ }
363
+
364
+ void start().catch((error) => {
365
+ log.error("worktree start task failed", { directory: info.directory, error })
366
+ })
367
+ }, 0)
368
+
369
+ return info
370
+ })
371
+
372
+ export const remove = fn(RemoveInput, async (input) => {
373
+ if (Instance.project.vcs !== "git") {
374
+ throw new NotGitError({ message: "Worktrees are only supported for git projects" })
375
+ }
376
+
377
+ const directory = path.resolve(input.directory)
378
+ const list = await $`git worktree list --porcelain`.quiet().nothrow().cwd(Instance.worktree)
379
+ if (list.exitCode !== 0) {
380
+ throw new RemoveFailedError({ message: errorText(list) || "Failed to read git worktrees" })
381
+ }
382
+
383
+ const lines = outputText(list.stdout)
384
+ .split("\n")
385
+ .map((line) => line.trim())
386
+ const entries = lines.reduce<{ path?: string; branch?: string }[]>((acc, line) => {
387
+ if (!line) return acc
388
+ if (line.startsWith("worktree ")) {
389
+ acc.push({ path: line.slice("worktree ".length).trim() })
390
+ return acc
391
+ }
392
+ const current = acc[acc.length - 1]
393
+ if (!current) return acc
394
+ if (line.startsWith("branch ")) {
395
+ current.branch = line.slice("branch ".length).trim()
396
+ }
397
+ return acc
398
+ }, [])
399
+
400
+ const entry = entries.find((item) => item.path && path.resolve(item.path) === directory)
401
+ if (!entry?.path) {
402
+ throw new RemoveFailedError({ message: "Worktree not found" })
403
+ }
404
+
405
+ const removed = await $`git worktree remove --force ${entry.path}`.quiet().nothrow().cwd(Instance.worktree)
406
+ if (removed.exitCode !== 0) {
407
+ throw new RemoveFailedError({ message: errorText(removed) || "Failed to remove git worktree" })
408
+ }
409
+
410
+ const branch = entry.branch?.replace(/^refs\/heads\//, "")
411
+ if (branch) {
412
+ const deleted = await $`git branch -D ${branch}`.quiet().nothrow().cwd(Instance.worktree)
413
+ if (deleted.exitCode !== 0) {
414
+ throw new RemoveFailedError({ message: errorText(deleted) || "Failed to delete worktree branch" })
415
+ }
416
+ }
417
+
418
+ return true
419
+ })
420
+
421
+ export const reset = fn(ResetInput, async (input) => {
422
+ if (Instance.project.vcs !== "git") {
423
+ throw new NotGitError({ message: "Worktrees are only supported for git projects" })
424
+ }
425
+
426
+ const directory = path.resolve(input.directory)
427
+ if (directory === path.resolve(Instance.worktree)) {
428
+ throw new ResetFailedError({ message: "Cannot reset the primary workspace" })
429
+ }
430
+
431
+ const list = await $`git worktree list --porcelain`.quiet().nothrow().cwd(Instance.worktree)
432
+ if (list.exitCode !== 0) {
433
+ throw new ResetFailedError({ message: errorText(list) || "Failed to read git worktrees" })
434
+ }
435
+
436
+ const lines = outputText(list.stdout)
437
+ .split("\n")
438
+ .map((line) => line.trim())
439
+ const entries = lines.reduce<{ path?: string; branch?: string }[]>((acc, line) => {
440
+ if (!line) return acc
441
+ if (line.startsWith("worktree ")) {
442
+ acc.push({ path: line.slice("worktree ".length).trim() })
443
+ return acc
444
+ }
445
+ const current = acc[acc.length - 1]
446
+ if (!current) return acc
447
+ if (line.startsWith("branch ")) {
448
+ current.branch = line.slice("branch ".length).trim()
449
+ }
450
+ return acc
451
+ }, [])
452
+
453
+ const entry = entries.find((item) => item.path && path.resolve(item.path) === directory)
454
+ if (!entry?.path) {
455
+ throw new ResetFailedError({ message: "Worktree not found" })
456
+ }
457
+
458
+ const remoteList = await $`git remote`.quiet().nothrow().cwd(Instance.worktree)
459
+ if (remoteList.exitCode !== 0) {
460
+ throw new ResetFailedError({ message: errorText(remoteList) || "Failed to list git remotes" })
461
+ }
462
+
463
+ const remotes = outputText(remoteList.stdout)
464
+ .split("\n")
465
+ .map((line) => line.trim())
466
+ .filter(Boolean)
467
+
468
+ const remote = remotes.includes("origin")
469
+ ? "origin"
470
+ : remotes.length === 1
471
+ ? remotes[0]
472
+ : remotes.includes("upstream")
473
+ ? "upstream"
474
+ : ""
475
+
476
+ const remoteHead = remote
477
+ ? await $`git symbolic-ref refs/remotes/${remote}/HEAD`.quiet().nothrow().cwd(Instance.worktree)
478
+ : { exitCode: 1, stdout: undefined, stderr: undefined }
479
+
480
+ const remoteRef = remoteHead.exitCode === 0 ? outputText(remoteHead.stdout) : ""
481
+ const remoteTarget = remoteRef ? remoteRef.replace(/^refs\/remotes\//, "") : ""
482
+ const remoteBranch = remote && remoteTarget.startsWith(`${remote}/`) ? remoteTarget.slice(`${remote}/`.length) : ""
483
+
484
+ const mainCheck = await $`git show-ref --verify --quiet refs/heads/main`.quiet().nothrow().cwd(Instance.worktree)
485
+ const masterCheck = await $`git show-ref --verify --quiet refs/heads/master`
486
+ .quiet()
487
+ .nothrow()
488
+ .cwd(Instance.worktree)
489
+ const localBranch = mainCheck.exitCode === 0 ? "main" : masterCheck.exitCode === 0 ? "master" : ""
490
+
491
+ const target = remoteBranch ? `${remote}/${remoteBranch}` : localBranch
492
+ if (!target) {
493
+ throw new ResetFailedError({ message: "Default branch not found" })
494
+ }
495
+
496
+ if (remoteBranch) {
497
+ const fetch = await $`git fetch ${remote} ${remoteBranch}`.quiet().nothrow().cwd(Instance.worktree)
498
+ if (fetch.exitCode !== 0) {
499
+ throw new ResetFailedError({ message: errorText(fetch) || `Failed to fetch ${target}` })
500
+ }
501
+ }
502
+
503
+ if (!entry.path) {
504
+ throw new ResetFailedError({ message: "Worktree path not found" })
505
+ }
506
+
507
+ const worktreePath = entry.path
508
+
509
+ const resetToTarget = await $`git reset --hard ${target}`.quiet().nothrow().cwd(worktreePath)
510
+ if (resetToTarget.exitCode !== 0) {
511
+ throw new ResetFailedError({ message: errorText(resetToTarget) || "Failed to reset worktree to target" })
512
+ }
513
+
514
+ const clean = await $`git clean -fdx`.quiet().nothrow().cwd(worktreePath)
515
+ if (clean.exitCode !== 0) {
516
+ throw new ResetFailedError({ message: errorText(clean) || "Failed to clean worktree" })
517
+ }
518
+
519
+ const update = await $`git submodule update --init --recursive --force`.quiet().nothrow().cwd(worktreePath)
520
+ if (update.exitCode !== 0) {
521
+ throw new ResetFailedError({ message: errorText(update) || "Failed to update submodules" })
522
+ }
523
+
524
+ const subReset = await $`git submodule foreach --recursive git reset --hard`.quiet().nothrow().cwd(worktreePath)
525
+ if (subReset.exitCode !== 0) {
526
+ throw new ResetFailedError({ message: errorText(subReset) || "Failed to reset submodules" })
527
+ }
528
+
529
+ const subClean = await $`git submodule foreach --recursive git clean -fdx`.quiet().nothrow().cwd(worktreePath)
530
+ if (subClean.exitCode !== 0) {
531
+ throw new ResetFailedError({ message: errorText(subClean) || "Failed to clean submodules" })
532
+ }
533
+
534
+ const status = await $`git status --porcelain=v1`.quiet().nothrow().cwd(worktreePath)
535
+ if (status.exitCode !== 0) {
536
+ throw new ResetFailedError({ message: errorText(status) || "Failed to read git status" })
537
+ }
538
+
539
+ const dirty = outputText(status.stdout)
540
+ if (dirty) {
541
+ throw new ResetFailedError({ message: `Worktree reset left local changes:\n${dirty}` })
542
+ }
543
+
544
+ const projectID = Instance.project.id
545
+ queueStartScripts(worktreePath, { projectID })
546
+
547
+ return true
548
+ })
549
+ }