@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,236 @@
1
+ import { $ } from "bun"
2
+ import path from "path"
3
+ import fs from "fs/promises"
4
+ import { Log } from "../util/log"
5
+ import { Global } from "../global"
6
+ import z from "zod"
7
+ import { Config } from "../config/config"
8
+ import { Instance } from "../project/instance"
9
+ import { Scheduler } from "../scheduler"
10
+
11
+ export namespace Snapshot {
12
+ const log = Log.create({ service: "snapshot" })
13
+ const hour = 60 * 60 * 1000
14
+ const prune = "7.days"
15
+
16
+ export function init() {
17
+ Scheduler.register({
18
+ id: "snapshot.cleanup",
19
+ interval: hour,
20
+ run: cleanup,
21
+ scope: "instance",
22
+ })
23
+ }
24
+
25
+ export async function cleanup() {
26
+ if (Instance.project.vcs !== "git") return
27
+ const cfg = await Config.get()
28
+ if (cfg.snapshot === false) return
29
+ const git = gitdir()
30
+ const exists = await fs
31
+ .stat(git)
32
+ .then(() => true)
33
+ .catch(() => false)
34
+ if (!exists) return
35
+ const result = await $`git --git-dir ${git} --work-tree ${Instance.worktree} gc --prune=${prune}`
36
+ .quiet()
37
+ .cwd(Instance.directory)
38
+ .nothrow()
39
+ if (result.exitCode !== 0) {
40
+ log.warn("cleanup failed", {
41
+ exitCode: result.exitCode,
42
+ stderr: result.stderr.toString(),
43
+ stdout: result.stdout.toString(),
44
+ })
45
+ return
46
+ }
47
+ log.info("cleanup", { prune })
48
+ }
49
+
50
+ export async function track() {
51
+ if (Instance.project.vcs !== "git") return
52
+ const cfg = await Config.get()
53
+ if (cfg.snapshot === false) return
54
+ const git = gitdir()
55
+ if (await fs.mkdir(git, { recursive: true })) {
56
+ await $`git init`
57
+ .env({
58
+ ...process.env,
59
+ GIT_DIR: git,
60
+ GIT_WORK_TREE: Instance.worktree,
61
+ })
62
+ .quiet()
63
+ .nothrow()
64
+ // Configure git to not convert line endings on Windows
65
+ await $`git --git-dir ${git} config core.autocrlf false`.quiet().nothrow()
66
+ log.info("initialized")
67
+ }
68
+ await $`git --git-dir ${git} --work-tree ${Instance.worktree} add .`.quiet().cwd(Instance.directory).nothrow()
69
+ const hash = await $`git --git-dir ${git} --work-tree ${Instance.worktree} write-tree`
70
+ .quiet()
71
+ .cwd(Instance.directory)
72
+ .nothrow()
73
+ .text()
74
+ log.info("tracking", { hash, cwd: Instance.directory, git })
75
+ return hash.trim()
76
+ }
77
+
78
+ export const Patch = z.object({
79
+ hash: z.string(),
80
+ files: z.string().array(),
81
+ })
82
+ export type Patch = z.infer<typeof Patch>
83
+
84
+ export async function patch(hash: string): Promise<Patch> {
85
+ const git = gitdir()
86
+ await $`git --git-dir ${git} --work-tree ${Instance.worktree} add .`.quiet().cwd(Instance.directory).nothrow()
87
+ const result =
88
+ await $`git -c core.autocrlf=false -c core.quotepath=false --git-dir ${git} --work-tree ${Instance.worktree} diff --no-ext-diff --name-only ${hash} -- .`
89
+ .quiet()
90
+ .cwd(Instance.directory)
91
+ .nothrow()
92
+
93
+ // If git diff fails, return empty patch
94
+ if (result.exitCode !== 0) {
95
+ log.warn("failed to get diff", { hash, exitCode: result.exitCode })
96
+ return { hash, files: [] }
97
+ }
98
+
99
+ const files = result.text()
100
+ return {
101
+ hash,
102
+ files: files
103
+ .trim()
104
+ .split("\n")
105
+ .map((x) => x.trim())
106
+ .filter(Boolean)
107
+ .map((x) => path.join(Instance.worktree, x)),
108
+ }
109
+ }
110
+
111
+ export async function restore(snapshot: string) {
112
+ log.info("restore", { commit: snapshot })
113
+ const git = gitdir()
114
+ const result =
115
+ await $`git --git-dir ${git} --work-tree ${Instance.worktree} read-tree ${snapshot} && git --git-dir ${git} --work-tree ${Instance.worktree} checkout-index -a -f`
116
+ .quiet()
117
+ .cwd(Instance.worktree)
118
+ .nothrow()
119
+
120
+ if (result.exitCode !== 0) {
121
+ log.error("failed to restore snapshot", {
122
+ snapshot,
123
+ exitCode: result.exitCode,
124
+ stderr: result.stderr.toString(),
125
+ stdout: result.stdout.toString(),
126
+ })
127
+ }
128
+ }
129
+
130
+ export async function revert(patches: Patch[]) {
131
+ const files = new Set<string>()
132
+ const git = gitdir()
133
+ for (const item of patches) {
134
+ for (const file of item.files) {
135
+ if (files.has(file)) continue
136
+ log.info("reverting", { file, hash: item.hash })
137
+ const result = await $`git --git-dir ${git} --work-tree ${Instance.worktree} checkout ${item.hash} -- ${file}`
138
+ .quiet()
139
+ .cwd(Instance.worktree)
140
+ .nothrow()
141
+ if (result.exitCode !== 0) {
142
+ const relativePath = path.relative(Instance.worktree, file)
143
+ const checkTree =
144
+ await $`git --git-dir ${git} --work-tree ${Instance.worktree} ls-tree ${item.hash} -- ${relativePath}`
145
+ .quiet()
146
+ .cwd(Instance.worktree)
147
+ .nothrow()
148
+ if (checkTree.exitCode === 0 && checkTree.text().trim()) {
149
+ log.info("file existed in snapshot but checkout failed, keeping", {
150
+ file,
151
+ })
152
+ } else {
153
+ log.info("file did not exist in snapshot, deleting", { file })
154
+ await fs.unlink(file).catch(() => {})
155
+ }
156
+ }
157
+ files.add(file)
158
+ }
159
+ }
160
+ }
161
+
162
+ export async function diff(hash: string) {
163
+ const git = gitdir()
164
+ await $`git --git-dir ${git} --work-tree ${Instance.worktree} add .`.quiet().cwd(Instance.directory).nothrow()
165
+ const result =
166
+ await $`git -c core.autocrlf=false -c core.quotepath=false --git-dir ${git} --work-tree ${Instance.worktree} diff --no-ext-diff ${hash} -- .`
167
+ .quiet()
168
+ .cwd(Instance.worktree)
169
+ .nothrow()
170
+
171
+ if (result.exitCode !== 0) {
172
+ log.warn("failed to get diff", {
173
+ hash,
174
+ exitCode: result.exitCode,
175
+ stderr: result.stderr.toString(),
176
+ stdout: result.stdout.toString(),
177
+ })
178
+ return ""
179
+ }
180
+
181
+ return result.text().trim()
182
+ }
183
+
184
+ export const FileDiff = z
185
+ .object({
186
+ file: z.string(),
187
+ before: z.string(),
188
+ after: z.string(),
189
+ additions: z.number(),
190
+ deletions: z.number(),
191
+ })
192
+ .meta({
193
+ ref: "FileDiff",
194
+ })
195
+ export type FileDiff = z.infer<typeof FileDiff>
196
+ export async function diffFull(from: string, to: string): Promise<FileDiff[]> {
197
+ const git = gitdir()
198
+ const result: FileDiff[] = []
199
+ for await (const line of $`git -c core.autocrlf=false -c core.quotepath=false --git-dir ${git} --work-tree ${Instance.worktree} diff --no-ext-diff --no-renames --numstat ${from} ${to} -- .`
200
+ .quiet()
201
+ .cwd(Instance.directory)
202
+ .nothrow()
203
+ .lines()) {
204
+ if (!line) continue
205
+ const [additions, deletions, file] = line.split("\t")
206
+ const isBinaryFile = additions === "-" && deletions === "-"
207
+ const before = isBinaryFile
208
+ ? ""
209
+ : await $`git -c core.autocrlf=false --git-dir ${git} --work-tree ${Instance.worktree} show ${from}:${file}`
210
+ .quiet()
211
+ .nothrow()
212
+ .text()
213
+ const after = isBinaryFile
214
+ ? ""
215
+ : await $`git -c core.autocrlf=false --git-dir ${git} --work-tree ${Instance.worktree} show ${to}:${file}`
216
+ .quiet()
217
+ .nothrow()
218
+ .text()
219
+ const added = isBinaryFile ? 0 : parseInt(additions)
220
+ const deleted = isBinaryFile ? 0 : parseInt(deletions)
221
+ result.push({
222
+ file,
223
+ before,
224
+ after,
225
+ additions: Number.isFinite(added) ? added : 0,
226
+ deletions: Number.isFinite(deleted) ? deleted : 0,
227
+ })
228
+ }
229
+ return result
230
+ }
231
+
232
+ function gitdir() {
233
+ const project = Instance.project
234
+ return path.join(Global.Path.data, "snapshot", project.id)
235
+ }
236
+ }
@@ -0,0 +1,227 @@
1
+ import { Log } from "../util/log"
2
+ import path from "path"
3
+ import fs from "fs/promises"
4
+ import { Global } from "../global"
5
+ import { Filesystem } from "../util/filesystem"
6
+ import { lazy } from "../util/lazy"
7
+ import { Lock } from "../util/lock"
8
+ import { $ } from "bun"
9
+ import { NamedError } from "@/util/error"
10
+ import z from "zod"
11
+
12
+ export namespace Storage {
13
+ const log = Log.create({ service: "storage" })
14
+
15
+ type Migration = (dir: string) => Promise<void>
16
+
17
+ export const NotFoundError = NamedError.create(
18
+ "NotFoundError",
19
+ z.object({
20
+ message: z.string(),
21
+ }),
22
+ )
23
+
24
+ const MIGRATIONS: Migration[] = [
25
+ async (dir) => {
26
+ const project = path.resolve(dir, "../project")
27
+ if (!(await Filesystem.isDir(project))) return
28
+ for await (const projectDir of new Bun.Glob("*").scan({
29
+ cwd: project,
30
+ onlyFiles: false,
31
+ })) {
32
+ log.info(`migrating project ${projectDir}`)
33
+ let projectID = projectDir
34
+ const fullProjectDir = path.join(project, projectDir)
35
+ let worktree = "/"
36
+
37
+ if (projectID !== "global") {
38
+ for await (const msgFile of new Bun.Glob("storage/session/message/*/*.json").scan({
39
+ cwd: path.join(project, projectDir),
40
+ absolute: true,
41
+ })) {
42
+ const json = await Bun.file(msgFile).json()
43
+ worktree = json.path?.root
44
+ if (worktree) break
45
+ }
46
+ if (!worktree) continue
47
+ if (!(await Filesystem.isDir(worktree))) continue
48
+ const [id] = await $`git rev-list --max-parents=0 --all`
49
+ .quiet()
50
+ .nothrow()
51
+ .cwd(worktree)
52
+ .text()
53
+ .then((x) =>
54
+ x
55
+ .split("\n")
56
+ .filter(Boolean)
57
+ .map((x) => x.trim())
58
+ .toSorted(),
59
+ )
60
+ if (!id) continue
61
+ projectID = id
62
+
63
+ await Bun.write(
64
+ path.join(dir, "project", projectID + ".json"),
65
+ JSON.stringify({
66
+ id,
67
+ vcs: "git",
68
+ worktree,
69
+ time: {
70
+ created: Date.now(),
71
+ initialized: Date.now(),
72
+ },
73
+ }),
74
+ )
75
+
76
+ log.info(`migrating sessions for project ${projectID}`)
77
+ for await (const sessionFile of new Bun.Glob("storage/session/info/*.json").scan({
78
+ cwd: fullProjectDir,
79
+ absolute: true,
80
+ })) {
81
+ const dest = path.join(dir, "session", projectID, path.basename(sessionFile))
82
+ log.info("copying", {
83
+ sessionFile,
84
+ dest,
85
+ })
86
+ const session = await Bun.file(sessionFile).json()
87
+ await Bun.write(dest, JSON.stringify(session))
88
+ log.info(`migrating messages for session ${session.id}`)
89
+ for await (const msgFile of new Bun.Glob(`storage/session/message/${session.id}/*.json`).scan({
90
+ cwd: fullProjectDir,
91
+ absolute: true,
92
+ })) {
93
+ const dest = path.join(dir, "message", session.id, path.basename(msgFile))
94
+ log.info("copying", {
95
+ msgFile,
96
+ dest,
97
+ })
98
+ const message = await Bun.file(msgFile).json()
99
+ await Bun.write(dest, JSON.stringify(message))
100
+
101
+ log.info(`migrating parts for message ${message.id}`)
102
+ for await (const partFile of new Bun.Glob(`storage/session/part/${session.id}/${message.id}/*.json`).scan(
103
+ {
104
+ cwd: fullProjectDir,
105
+ absolute: true,
106
+ },
107
+ )) {
108
+ const dest = path.join(dir, "part", message.id, path.basename(partFile))
109
+ const part = await Bun.file(partFile).json()
110
+ log.info("copying", {
111
+ partFile,
112
+ dest,
113
+ })
114
+ await Bun.write(dest, JSON.stringify(part))
115
+ }
116
+ }
117
+ }
118
+ }
119
+ }
120
+ },
121
+ async (dir) => {
122
+ for await (const item of new Bun.Glob("session/*/*.json").scan({
123
+ cwd: dir,
124
+ absolute: true,
125
+ })) {
126
+ const session = await Bun.file(item).json()
127
+ if (!session.projectID) continue
128
+ if (!session.summary?.diffs) continue
129
+ const { diffs } = session.summary
130
+ await Bun.file(path.join(dir, "session_diff", session.id + ".json")).write(JSON.stringify(diffs))
131
+ await Bun.file(path.join(dir, "session", session.projectID, session.id + ".json")).write(
132
+ JSON.stringify({
133
+ ...session,
134
+ summary: {
135
+ additions: diffs.reduce((sum: any, x: any) => sum + x.additions, 0),
136
+ deletions: diffs.reduce((sum: any, x: any) => sum + x.deletions, 0),
137
+ },
138
+ }),
139
+ )
140
+ }
141
+ },
142
+ ]
143
+
144
+ const state = lazy(async () => {
145
+ const dir = path.join(Global.Path.data, "storage")
146
+ const migration = await Bun.file(path.join(dir, "migration"))
147
+ .json()
148
+ .then((x) => parseInt(x))
149
+ .catch(() => 0)
150
+ for (let index = migration; index < MIGRATIONS.length; index++) {
151
+ log.info("running migration", { index })
152
+ const migration = MIGRATIONS[index]
153
+ await migration(dir).catch(() => log.error("failed to run migration", { index }))
154
+ await Bun.write(path.join(dir, "migration"), (index + 1).toString())
155
+ }
156
+ return {
157
+ dir,
158
+ }
159
+ })
160
+
161
+ export async function remove(key: string[]) {
162
+ const dir = await state().then((x) => x.dir)
163
+ const target = path.join(dir, ...key) + ".json"
164
+ return withErrorHandling(async () => {
165
+ await fs.unlink(target).catch(() => {})
166
+ })
167
+ }
168
+
169
+ export async function read<T>(key: string[]) {
170
+ const dir = await state().then((x) => x.dir)
171
+ const target = path.join(dir, ...key) + ".json"
172
+ return withErrorHandling(async () => {
173
+ using _ = await Lock.read(target)
174
+ const result = await Bun.file(target).json()
175
+ return result as T
176
+ })
177
+ }
178
+
179
+ export async function update<T>(key: string[], fn: (draft: T) => void) {
180
+ const dir = await state().then((x) => x.dir)
181
+ const target = path.join(dir, ...key) + ".json"
182
+ return withErrorHandling(async () => {
183
+ using _ = await Lock.write(target)
184
+ const content = await Bun.file(target).json()
185
+ fn(content)
186
+ await Bun.write(target, JSON.stringify(content, null, 2))
187
+ return content as T
188
+ })
189
+ }
190
+
191
+ export async function write<T>(key: string[], content: T) {
192
+ const dir = await state().then((x) => x.dir)
193
+ const target = path.join(dir, ...key) + ".json"
194
+ return withErrorHandling(async () => {
195
+ using _ = await Lock.write(target)
196
+ await Bun.write(target, JSON.stringify(content, null, 2))
197
+ })
198
+ }
199
+
200
+ async function withErrorHandling<T>(body: () => Promise<T>) {
201
+ return body().catch((e) => {
202
+ if (!(e instanceof Error)) throw e
203
+ const errnoException = e as NodeJS.ErrnoException
204
+ if (errnoException.code === "ENOENT") {
205
+ throw new NotFoundError({ message: `Resource not found: ${errnoException.path}` })
206
+ }
207
+ throw e
208
+ })
209
+ }
210
+
211
+ const glob = new Bun.Glob("**/*")
212
+ export async function list(prefix: string[]) {
213
+ const dir = await state().then((x) => x.dir)
214
+ try {
215
+ const result = await Array.fromAsync(
216
+ glob.scan({
217
+ cwd: path.join(dir, ...prefix),
218
+ onlyFiles: true,
219
+ }),
220
+ ).then((results) => results.map((x) => [...prefix, ...x.slice(0, -5).split(path.sep)]))
221
+ result.sort()
222
+ return result
223
+ } catch {
224
+ return []
225
+ }
226
+ }
227
+ }