rird 1.0.200

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 (350) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/opencode +336 -0
  5. package/bin/pty-wrapper.js +285 -0
  6. package/bunfig.toml +4 -0
  7. package/facebook_ads_library.png +0 -0
  8. package/nul`nif +0 -0
  9. package/package.json +111 -0
  10. package/parsers-config.ts +239 -0
  11. package/rird-1.0.199.tgz +0 -0
  12. package/script/build-windows.ts +54 -0
  13. package/script/build.ts +167 -0
  14. package/script/postinstall.mjs +544 -0
  15. package/script/publish-registries.ts +187 -0
  16. package/script/publish.ts +72 -0
  17. package/script/schema.ts +47 -0
  18. package/src/acp/README.md +164 -0
  19. package/src/acp/agent.ts +1063 -0
  20. package/src/acp/session.ts +101 -0
  21. package/src/acp/types.ts +22 -0
  22. package/src/agent/agent.ts +367 -0
  23. package/src/agent/generate.txt +75 -0
  24. package/src/agent/prompt/compaction.txt +12 -0
  25. package/src/agent/prompt/explore.txt +18 -0
  26. package/src/agent/prompt/summary.txt +10 -0
  27. package/src/agent/prompt/title.txt +36 -0
  28. package/src/auth/index.ts +70 -0
  29. package/src/bun/index.ts +114 -0
  30. package/src/bus/bus-event.ts +43 -0
  31. package/src/bus/global.ts +10 -0
  32. package/src/bus/index.ts +105 -0
  33. package/src/cli/bootstrap.ts +17 -0
  34. package/src/cli/cmd/acp.ts +88 -0
  35. package/src/cli/cmd/agent.ts +256 -0
  36. package/src/cli/cmd/auth.ts +391 -0
  37. package/src/cli/cmd/cmd.ts +7 -0
  38. package/src/cli/cmd/debug/config.ts +15 -0
  39. package/src/cli/cmd/debug/file.ts +91 -0
  40. package/src/cli/cmd/debug/index.ts +43 -0
  41. package/src/cli/cmd/debug/lsp.ts +48 -0
  42. package/src/cli/cmd/debug/ripgrep.ts +83 -0
  43. package/src/cli/cmd/debug/scrap.ts +15 -0
  44. package/src/cli/cmd/debug/skill.ts +15 -0
  45. package/src/cli/cmd/debug/snapshot.ts +48 -0
  46. package/src/cli/cmd/export.ts +88 -0
  47. package/src/cli/cmd/generate.ts +38 -0
  48. package/src/cli/cmd/github.ts +1400 -0
  49. package/src/cli/cmd/import.ts +98 -0
  50. package/src/cli/cmd/mcp.ts +654 -0
  51. package/src/cli/cmd/models.ts +77 -0
  52. package/src/cli/cmd/pr.ts +112 -0
  53. package/src/cli/cmd/run.ts +368 -0
  54. package/src/cli/cmd/serve.ts +31 -0
  55. package/src/cli/cmd/session.ts +106 -0
  56. package/src/cli/cmd/stats.ts +298 -0
  57. package/src/cli/cmd/tui/app.tsx +696 -0
  58. package/src/cli/cmd/tui/attach.ts +30 -0
  59. package/src/cli/cmd/tui/component/border.tsx +21 -0
  60. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  61. package/src/cli/cmd/tui/component/dialog-command.tsx +124 -0
  62. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  63. package/src/cli/cmd/tui/component/dialog-model.tsx +245 -0
  64. package/src/cli/cmd/tui/component/dialog-provider.tsx +224 -0
  65. package/src/cli/cmd/tui/component/dialog-session-list.tsx +102 -0
  66. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  67. package/src/cli/cmd/tui/component/dialog-stash.tsx +86 -0
  68. package/src/cli/cmd/tui/component/dialog-status.tsx +162 -0
  69. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  70. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  71. package/src/cli/cmd/tui/component/did-you-know.tsx +85 -0
  72. package/src/cli/cmd/tui/component/logo.tsx +35 -0
  73. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +574 -0
  74. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  75. package/src/cli/cmd/tui/component/prompt/index.tsx +1090 -0
  76. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  77. package/src/cli/cmd/tui/component/tips.ts +27 -0
  78. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  79. package/src/cli/cmd/tui/context/args.tsx +14 -0
  80. package/src/cli/cmd/tui/context/directory.ts +13 -0
  81. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  82. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  83. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  84. package/src/cli/cmd/tui/context/kv.tsx +49 -0
  85. package/src/cli/cmd/tui/context/local.tsx +354 -0
  86. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  87. package/src/cli/cmd/tui/context/route.tsx +46 -0
  88. package/src/cli/cmd/tui/context/sdk.tsx +74 -0
  89. package/src/cli/cmd/tui/context/sync.tsx +372 -0
  90. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  91. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  92. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  93. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  94. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  95. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  96. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  97. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  98. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  99. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  100. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  101. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  102. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  103. package/src/cli/cmd/tui/context/theme/lucent-orng.json +227 -0
  104. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  105. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  106. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  107. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  108. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  109. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  110. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  111. package/src/cli/cmd/tui/context/theme/orng.json +245 -0
  112. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  113. package/src/cli/cmd/tui/context/theme/rird.json +245 -0
  114. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  115. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  116. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  117. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  118. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  119. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  120. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  121. package/src/cli/cmd/tui/context/theme.tsx +1109 -0
  122. package/src/cli/cmd/tui/event.ts +40 -0
  123. package/src/cli/cmd/tui/routes/home.tsx +138 -0
  124. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  125. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  126. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  128. package/src/cli/cmd/tui/routes/session/footer.tsx +88 -0
  129. package/src/cli/cmd/tui/routes/session/header.tsx +125 -0
  130. package/src/cli/cmd/tui/routes/session/index.tsx +1864 -0
  131. package/src/cli/cmd/tui/routes/session/sidebar.tsx +318 -0
  132. package/src/cli/cmd/tui/spawn.ts +60 -0
  133. package/src/cli/cmd/tui/thread.ts +142 -0
  134. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  135. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -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 +332 -0
  139. package/src/cli/cmd/tui/ui/dialog.tsx +170 -0
  140. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  141. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  142. package/src/cli/cmd/tui/util/clipboard.ts +127 -0
  143. package/src/cli/cmd/tui/util/editor.ts +32 -0
  144. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  145. package/src/cli/cmd/tui/worker.ts +63 -0
  146. package/src/cli/cmd/uninstall.ts +344 -0
  147. package/src/cli/cmd/upgrade.ts +100 -0
  148. package/src/cli/cmd/web.ts +84 -0
  149. package/src/cli/error.ts +56 -0
  150. package/src/cli/ui.ts +84 -0
  151. package/src/cli/upgrade.ts +25 -0
  152. package/src/command/index.ts +80 -0
  153. package/src/command/template/initialize.txt +10 -0
  154. package/src/command/template/review.txt +97 -0
  155. package/src/config/config.ts +995 -0
  156. package/src/config/markdown.ts +41 -0
  157. package/src/env/index.ts +26 -0
  158. package/src/file/ignore.ts +83 -0
  159. package/src/file/index.ts +328 -0
  160. package/src/file/ripgrep.ts +393 -0
  161. package/src/file/time.ts +64 -0
  162. package/src/file/watcher.ts +103 -0
  163. package/src/flag/flag.ts +46 -0
  164. package/src/format/formatter.ts +315 -0
  165. package/src/format/index.ts +137 -0
  166. package/src/global/index.ts +52 -0
  167. package/src/id/id.ts +73 -0
  168. package/src/ide/index.ts +76 -0
  169. package/src/index.ts +240 -0
  170. package/src/installation/index.ts +239 -0
  171. package/src/lsp/client.ts +229 -0
  172. package/src/lsp/index.ts +485 -0
  173. package/src/lsp/language.ts +116 -0
  174. package/src/lsp/server.ts +1895 -0
  175. package/src/mcp/auth.ts +135 -0
  176. package/src/mcp/index.ts +690 -0
  177. package/src/mcp/oauth-callback.ts +200 -0
  178. package/src/mcp/oauth-provider.ts +154 -0
  179. package/src/patch/index.ts +622 -0
  180. package/src/permission/index.ts +199 -0
  181. package/src/plugin/index.ts +91 -0
  182. package/src/project/bootstrap.ts +31 -0
  183. package/src/project/instance.ts +78 -0
  184. package/src/project/project.ts +221 -0
  185. package/src/project/state.ts +65 -0
  186. package/src/project/vcs.ts +76 -0
  187. package/src/provider/auth.ts +143 -0
  188. package/src/provider/models-macro.ts +11 -0
  189. package/src/provider/models.ts +106 -0
  190. package/src/provider/provider.ts +1071 -0
  191. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  192. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  193. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  194. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  195. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
  196. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  197. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  198. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  199. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1713 -0
  200. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  201. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  202. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  203. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  204. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  205. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  206. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  207. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  208. package/src/provider/transform.ts +455 -0
  209. package/src/pty/index.ts +231 -0
  210. package/src/security/guardrails.test.ts +341 -0
  211. package/src/security/guardrails.ts +558 -0
  212. package/src/security/index.ts +19 -0
  213. package/src/server/error.ts +36 -0
  214. package/src/server/project.ts +79 -0
  215. package/src/server/server.ts +2642 -0
  216. package/src/server/tui.ts +71 -0
  217. package/src/session/compaction.ts +223 -0
  218. package/src/session/index.ts +461 -0
  219. package/src/session/llm.ts +201 -0
  220. package/src/session/message-v2.ts +690 -0
  221. package/src/session/message.ts +189 -0
  222. package/src/session/processor.ts +409 -0
  223. package/src/session/prompt/act-switch.txt +5 -0
  224. package/src/session/prompt/anthropic-20250930.txt +166 -0
  225. package/src/session/prompt/anthropic.txt +85 -0
  226. package/src/session/prompt/anthropic_spoof.txt +1 -0
  227. package/src/session/prompt/beast.txt +103 -0
  228. package/src/session/prompt/codex.txt +304 -0
  229. package/src/session/prompt/copilot-gpt-5.txt +138 -0
  230. package/src/session/prompt/gemini.txt +85 -0
  231. package/src/session/prompt/max-steps.txt +16 -0
  232. package/src/session/prompt/plan-reminder-anthropic.txt +35 -0
  233. package/src/session/prompt/plan.txt +24 -0
  234. package/src/session/prompt/polaris.txt +84 -0
  235. package/src/session/prompt/qwen.txt +106 -0
  236. package/src/session/prompt.ts +1509 -0
  237. package/src/session/retry.ts +86 -0
  238. package/src/session/revert.ts +108 -0
  239. package/src/session/sensitive-filter.test.ts +327 -0
  240. package/src/session/sensitive-filter.ts +466 -0
  241. package/src/session/status.ts +76 -0
  242. package/src/session/summary.ts +194 -0
  243. package/src/session/system.ts +120 -0
  244. package/src/session/todo.ts +37 -0
  245. package/src/share/share-next.ts +194 -0
  246. package/src/share/share.ts +87 -0
  247. package/src/shell/shell.ts +67 -0
  248. package/src/skill/index.ts +1 -0
  249. package/src/skill/skill.ts +83 -0
  250. package/src/snapshot/index.ts +197 -0
  251. package/src/storage/storage.ts +226 -0
  252. package/src/tests/agent.test.ts +308 -0
  253. package/src/tests/build-guards.test.ts +267 -0
  254. package/src/tests/config.test.ts +664 -0
  255. package/src/tests/tool-registry.test.ts +589 -0
  256. package/src/tool/bash.ts +317 -0
  257. package/src/tool/bash.txt +158 -0
  258. package/src/tool/batch.ts +175 -0
  259. package/src/tool/batch.txt +24 -0
  260. package/src/tool/codesearch.ts +168 -0
  261. package/src/tool/codesearch.txt +12 -0
  262. package/src/tool/edit.ts +675 -0
  263. package/src/tool/edit.txt +10 -0
  264. package/src/tool/glob.ts +65 -0
  265. package/src/tool/glob.txt +6 -0
  266. package/src/tool/grep.ts +121 -0
  267. package/src/tool/grep.txt +8 -0
  268. package/src/tool/invalid.ts +17 -0
  269. package/src/tool/ls.ts +110 -0
  270. package/src/tool/ls.txt +1 -0
  271. package/src/tool/lsp-diagnostics.ts +26 -0
  272. package/src/tool/lsp-diagnostics.txt +1 -0
  273. package/src/tool/lsp-hover.ts +31 -0
  274. package/src/tool/lsp-hover.txt +1 -0
  275. package/src/tool/lsp.ts +87 -0
  276. package/src/tool/lsp.txt +19 -0
  277. package/src/tool/multiedit.ts +46 -0
  278. package/src/tool/multiedit.txt +41 -0
  279. package/src/tool/patch.ts +233 -0
  280. package/src/tool/patch.txt +1 -0
  281. package/src/tool/read.ts +219 -0
  282. package/src/tool/read.txt +12 -0
  283. package/src/tool/registry.ts +162 -0
  284. package/src/tool/skill.ts +100 -0
  285. package/src/tool/task.ts +136 -0
  286. package/src/tool/task.txt +51 -0
  287. package/src/tool/todo.ts +39 -0
  288. package/src/tool/todoread.txt +14 -0
  289. package/src/tool/todowrite.txt +167 -0
  290. package/src/tool/tool.ts +71 -0
  291. package/src/tool/webfetch.ts +198 -0
  292. package/src/tool/webfetch.txt +13 -0
  293. package/src/tool/websearch.ts +180 -0
  294. package/src/tool/websearch.txt +11 -0
  295. package/src/tool/write.ts +110 -0
  296. package/src/tool/write.txt +8 -0
  297. package/src/util/archive.ts +16 -0
  298. package/src/util/color.ts +19 -0
  299. package/src/util/context.ts +25 -0
  300. package/src/util/defer.ts +12 -0
  301. package/src/util/eventloop.ts +20 -0
  302. package/src/util/filesystem.ts +83 -0
  303. package/src/util/fn.ts +11 -0
  304. package/src/util/iife.ts +3 -0
  305. package/src/util/keybind.ts +102 -0
  306. package/src/util/lazy.ts +11 -0
  307. package/src/util/license.ts +325 -0
  308. package/src/util/locale.ts +81 -0
  309. package/src/util/lock.ts +98 -0
  310. package/src/util/log.ts +180 -0
  311. package/src/util/queue.ts +32 -0
  312. package/src/util/rpc.ts +42 -0
  313. package/src/util/scrap.ts +10 -0
  314. package/src/util/signal.ts +12 -0
  315. package/src/util/timeout.ts +14 -0
  316. package/src/util/token.ts +7 -0
  317. package/src/util/wildcard.ts +54 -0
  318. package/sst-env.d.ts +9 -0
  319. package/test/agent/agent.test.ts +146 -0
  320. package/test/bun.test.ts +53 -0
  321. package/test/cli/github-remote.test.ts +80 -0
  322. package/test/config/agent-color.test.ts +66 -0
  323. package/test/config/config.test.ts +535 -0
  324. package/test/config/markdown.test.ts +89 -0
  325. package/test/file/ignore.test.ts +10 -0
  326. package/test/fixture/fixture.ts +36 -0
  327. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  328. package/test/ide/ide.test.ts +82 -0
  329. package/test/keybind.test.ts +421 -0
  330. package/test/lsp/client.test.ts +95 -0
  331. package/test/mcp/headers.test.ts +153 -0
  332. package/test/patch/patch.test.ts +348 -0
  333. package/test/preload.ts +57 -0
  334. package/test/project/project.test.ts +72 -0
  335. package/test/provider/provider.test.ts +1809 -0
  336. package/test/provider/transform.test.ts +411 -0
  337. package/test/session/retry.test.ts +111 -0
  338. package/test/session/session.test.ts +71 -0
  339. package/test/skill/skill.test.ts +131 -0
  340. package/test/snapshot/snapshot.test.ts +939 -0
  341. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  342. package/test/tool/bash.test.ts +434 -0
  343. package/test/tool/grep.test.ts +108 -0
  344. package/test/tool/patch.test.ts +259 -0
  345. package/test/tool/read.test.ts +42 -0
  346. package/test/util/iife.test.ts +36 -0
  347. package/test/util/lazy.test.ts +50 -0
  348. package/test/util/timeout.test.ts +21 -0
  349. package/test/util/wildcard.test.ts +55 -0
  350. package/tsconfig.json +16 -0
@@ -0,0 +1,83 @@
1
+ import z from "zod"
2
+ import { Config } from "../config/config"
3
+ import { Instance } from "../project/instance"
4
+ import { NamedError } from "@opencode-ai/util/error"
5
+ import { ConfigMarkdown } from "../config/markdown"
6
+ import { Log } from "../util/log"
7
+
8
+ export namespace Skill {
9
+ const log = Log.create({ service: "skill" })
10
+ export const Info = z.object({
11
+ name: z.string(),
12
+ description: z.string(),
13
+ location: z.string(),
14
+ })
15
+ export type Info = z.infer<typeof Info>
16
+
17
+ export const InvalidError = NamedError.create(
18
+ "SkillInvalidError",
19
+ z.object({
20
+ path: z.string(),
21
+ message: z.string().optional(),
22
+ issues: z.custom<z.core.$ZodIssue[]>().optional(),
23
+ }),
24
+ )
25
+
26
+ export const NameMismatchError = NamedError.create(
27
+ "SkillNameMismatchError",
28
+ z.object({
29
+ path: z.string(),
30
+ expected: z.string(),
31
+ actual: z.string(),
32
+ }),
33
+ )
34
+
35
+ const SKILL_GLOB = new Bun.Glob("skill/**/SKILL.md")
36
+
37
+ export const state = Instance.state(async () => {
38
+ const directories = await Config.directories()
39
+ const skills: Record<string, Info> = {}
40
+
41
+ for (const dir of directories) {
42
+ for await (const match of SKILL_GLOB.scan({
43
+ cwd: dir,
44
+ absolute: true,
45
+ onlyFiles: true,
46
+ followSymlinks: true,
47
+ })) {
48
+ const md = await ConfigMarkdown.parse(match)
49
+ if (!md) {
50
+ continue
51
+ }
52
+
53
+ const parsed = Info.pick({ name: true, description: true }).safeParse(md.data)
54
+ if (!parsed.success) continue
55
+
56
+ // Warn on duplicate skill names
57
+ if (skills[parsed.data.name]) {
58
+ log.warn("duplicate skill name", {
59
+ name: parsed.data.name,
60
+ existing: skills[parsed.data.name].location,
61
+ duplicate: match,
62
+ })
63
+ }
64
+
65
+ skills[parsed.data.name] = {
66
+ name: parsed.data.name,
67
+ description: parsed.data.description,
68
+ location: match,
69
+ }
70
+ }
71
+ }
72
+
73
+ return skills
74
+ })
75
+
76
+ export async function get(name: string) {
77
+ return state().then((x) => x[name])
78
+ }
79
+
80
+ export async function all() {
81
+ return state().then((x) => Object.values(x))
82
+ }
83
+ }
@@ -0,0 +1,197 @@
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
+
10
+ export namespace Snapshot {
11
+ const log = Log.create({ service: "snapshot" })
12
+
13
+ export async function track() {
14
+ if (Instance.project.vcs !== "git") return
15
+ const cfg = await Config.get()
16
+ if (cfg.snapshot === false) return
17
+ const git = gitdir()
18
+ if (await fs.mkdir(git, { recursive: true })) {
19
+ await $`git init`
20
+ .env({
21
+ ...process.env,
22
+ GIT_DIR: git,
23
+ GIT_WORK_TREE: Instance.worktree,
24
+ })
25
+ .quiet()
26
+ .nothrow()
27
+ // Configure git to not convert line endings on Windows
28
+ await $`git --git-dir ${git} config core.autocrlf false`.quiet().nothrow()
29
+ log.info("initialized")
30
+ }
31
+ await $`git --git-dir ${git} --work-tree ${Instance.worktree} add .`.quiet().cwd(Instance.directory).nothrow()
32
+ const hash = await $`git --git-dir ${git} --work-tree ${Instance.worktree} write-tree`
33
+ .quiet()
34
+ .cwd(Instance.directory)
35
+ .nothrow()
36
+ .text()
37
+ log.info("tracking", { hash, cwd: Instance.directory, git })
38
+ return hash.trim()
39
+ }
40
+
41
+ export const Patch = z.object({
42
+ hash: z.string(),
43
+ files: z.string().array(),
44
+ })
45
+ export type Patch = z.infer<typeof Patch>
46
+
47
+ export async function patch(hash: string): Promise<Patch> {
48
+ const git = gitdir()
49
+ await $`git --git-dir ${git} --work-tree ${Instance.worktree} add .`.quiet().cwd(Instance.directory).nothrow()
50
+ const result =
51
+ await $`git -c core.autocrlf=false --git-dir ${git} --work-tree ${Instance.worktree} diff --no-ext-diff --name-only ${hash} -- .`
52
+ .quiet()
53
+ .cwd(Instance.directory)
54
+ .nothrow()
55
+
56
+ // If git diff fails, return empty patch
57
+ if (result.exitCode !== 0) {
58
+ log.warn("failed to get diff", { hash, exitCode: result.exitCode })
59
+ return { hash, files: [] }
60
+ }
61
+
62
+ const files = result.text()
63
+ return {
64
+ hash,
65
+ files: files
66
+ .trim()
67
+ .split("\n")
68
+ .map((x) => x.trim())
69
+ .filter(Boolean)
70
+ .map((x) => path.join(Instance.worktree, x)),
71
+ }
72
+ }
73
+
74
+ export async function restore(snapshot: string) {
75
+ log.info("restore", { commit: snapshot })
76
+ const git = gitdir()
77
+ const result =
78
+ await $`git --git-dir ${git} --work-tree ${Instance.worktree} read-tree ${snapshot} && git --git-dir ${git} --work-tree ${Instance.worktree} checkout-index -a -f`
79
+ .quiet()
80
+ .cwd(Instance.worktree)
81
+ .nothrow()
82
+
83
+ if (result.exitCode !== 0) {
84
+ log.error("failed to restore snapshot", {
85
+ snapshot,
86
+ exitCode: result.exitCode,
87
+ stderr: result.stderr.toString(),
88
+ stdout: result.stdout.toString(),
89
+ })
90
+ }
91
+ }
92
+
93
+ export async function revert(patches: Patch[]) {
94
+ const files = new Set<string>()
95
+ const git = gitdir()
96
+ for (const item of patches) {
97
+ for (const file of item.files) {
98
+ if (files.has(file)) continue
99
+ log.info("reverting", { file, hash: item.hash })
100
+ const result = await $`git --git-dir ${git} --work-tree ${Instance.worktree} checkout ${item.hash} -- ${file}`
101
+ .quiet()
102
+ .cwd(Instance.worktree)
103
+ .nothrow()
104
+ if (result.exitCode !== 0) {
105
+ const relativePath = path.relative(Instance.worktree, file)
106
+ const checkTree =
107
+ await $`git --git-dir ${git} --work-tree ${Instance.worktree} ls-tree ${item.hash} -- ${relativePath}`
108
+ .quiet()
109
+ .cwd(Instance.worktree)
110
+ .nothrow()
111
+ if (checkTree.exitCode === 0 && checkTree.text().trim()) {
112
+ log.info("file existed in snapshot but checkout failed, keeping", {
113
+ file,
114
+ })
115
+ } else {
116
+ log.info("file did not exist in snapshot, deleting", { file })
117
+ await fs.unlink(file).catch(() => {})
118
+ }
119
+ }
120
+ files.add(file)
121
+ }
122
+ }
123
+ }
124
+
125
+ export async function diff(hash: string) {
126
+ const git = gitdir()
127
+ await $`git --git-dir ${git} --work-tree ${Instance.worktree} add .`.quiet().cwd(Instance.directory).nothrow()
128
+ const result =
129
+ await $`git -c core.autocrlf=false --git-dir ${git} --work-tree ${Instance.worktree} diff --no-ext-diff ${hash} -- .`
130
+ .quiet()
131
+ .cwd(Instance.worktree)
132
+ .nothrow()
133
+
134
+ if (result.exitCode !== 0) {
135
+ log.warn("failed to get diff", {
136
+ hash,
137
+ exitCode: result.exitCode,
138
+ stderr: result.stderr.toString(),
139
+ stdout: result.stdout.toString(),
140
+ })
141
+ return ""
142
+ }
143
+
144
+ return result.text().trim()
145
+ }
146
+
147
+ export const FileDiff = z
148
+ .object({
149
+ file: z.string(),
150
+ before: z.string(),
151
+ after: z.string(),
152
+ additions: z.number(),
153
+ deletions: z.number(),
154
+ })
155
+ .meta({
156
+ ref: "FileDiff",
157
+ })
158
+ export type FileDiff = z.infer<typeof FileDiff>
159
+ export async function diffFull(from: string, to: string): Promise<FileDiff[]> {
160
+ const git = gitdir()
161
+ const result: FileDiff[] = []
162
+ for await (const line of $`git -c core.autocrlf=false --git-dir ${git} --work-tree ${Instance.worktree} diff --no-ext-diff --no-renames --numstat ${from} ${to} -- .`
163
+ .quiet()
164
+ .cwd(Instance.directory)
165
+ .nothrow()
166
+ .lines()) {
167
+ if (!line) continue
168
+ const [additions, deletions, file] = line.split("\t")
169
+ const isBinaryFile = additions === "-" && deletions === "-"
170
+ const before = isBinaryFile
171
+ ? ""
172
+ : await $`git -c core.autocrlf=false --git-dir ${git} --work-tree ${Instance.worktree} show ${from}:${file}`
173
+ .quiet()
174
+ .nothrow()
175
+ .text()
176
+ const after = isBinaryFile
177
+ ? ""
178
+ : await $`git -c core.autocrlf=false --git-dir ${git} --work-tree ${Instance.worktree} show ${to}:${file}`
179
+ .quiet()
180
+ .nothrow()
181
+ .text()
182
+ result.push({
183
+ file,
184
+ before,
185
+ after,
186
+ additions: parseInt(additions),
187
+ deletions: parseInt(deletions),
188
+ })
189
+ }
190
+ return result
191
+ }
192
+
193
+ function gitdir() {
194
+ const project = Instance.project
195
+ return path.join(Global.Path.data, "snapshot", project.id)
196
+ }
197
+ }
@@ -0,0 +1,226 @@
1
+ import { Log } from "../util/log"
2
+ import path from "path"
3
+ import fs from "fs/promises"
4
+ import { Global } from "../global"
5
+ import { lazy } from "../util/lazy"
6
+ import { Lock } from "../util/lock"
7
+ import { $ } from "bun"
8
+ import { NamedError } from "@opencode-ai/util/error"
9
+ import z from "zod"
10
+
11
+ export namespace Storage {
12
+ const log = Log.create({ service: "storage" })
13
+
14
+ type Migration = (dir: string) => Promise<void>
15
+
16
+ export const NotFoundError = NamedError.create(
17
+ "NotFoundError",
18
+ z.object({
19
+ message: z.string(),
20
+ }),
21
+ )
22
+
23
+ const MIGRATIONS: Migration[] = [
24
+ async (dir) => {
25
+ const project = path.resolve(dir, "../project")
26
+ if (!fs.exists(project)) return
27
+ for await (const projectDir of new Bun.Glob("*").scan({
28
+ cwd: project,
29
+ onlyFiles: false,
30
+ })) {
31
+ log.info(`migrating project ${projectDir}`)
32
+ let projectID = projectDir
33
+ const fullProjectDir = path.join(project, projectDir)
34
+ let worktree = "/"
35
+
36
+ if (projectID !== "global") {
37
+ for await (const msgFile of new Bun.Glob("storage/session/message/*/*.json").scan({
38
+ cwd: path.join(project, projectDir),
39
+ absolute: true,
40
+ })) {
41
+ const json = await Bun.file(msgFile).json()
42
+ worktree = json.path?.root
43
+ if (worktree) break
44
+ }
45
+ if (!worktree) continue
46
+ if (!(await fs.exists(worktree))) continue
47
+ const [id] = await $`git rev-list --max-parents=0 --all`
48
+ .quiet()
49
+ .nothrow()
50
+ .cwd(worktree)
51
+ .text()
52
+ .then((x) =>
53
+ x
54
+ .split("\n")
55
+ .filter(Boolean)
56
+ .map((x) => x.trim())
57
+ .toSorted(),
58
+ )
59
+ if (!id) continue
60
+ projectID = id
61
+
62
+ await Bun.write(
63
+ path.join(dir, "project", projectID + ".json"),
64
+ JSON.stringify({
65
+ id,
66
+ vcs: "git",
67
+ worktree,
68
+ time: {
69
+ created: Date.now(),
70
+ initialized: Date.now(),
71
+ },
72
+ }),
73
+ )
74
+
75
+ log.info(`migrating sessions for project ${projectID}`)
76
+ for await (const sessionFile of new Bun.Glob("storage/session/info/*.json").scan({
77
+ cwd: fullProjectDir,
78
+ absolute: true,
79
+ })) {
80
+ const dest = path.join(dir, "session", projectID, path.basename(sessionFile))
81
+ log.info("copying", {
82
+ sessionFile,
83
+ dest,
84
+ })
85
+ const session = await Bun.file(sessionFile).json()
86
+ await Bun.write(dest, JSON.stringify(session))
87
+ log.info(`migrating messages for session ${session.id}`)
88
+ for await (const msgFile of new Bun.Glob(`storage/session/message/${session.id}/*.json`).scan({
89
+ cwd: fullProjectDir,
90
+ absolute: true,
91
+ })) {
92
+ const dest = path.join(dir, "message", session.id, path.basename(msgFile))
93
+ log.info("copying", {
94
+ msgFile,
95
+ dest,
96
+ })
97
+ const message = await Bun.file(msgFile).json()
98
+ await Bun.write(dest, JSON.stringify(message))
99
+
100
+ log.info(`migrating parts for message ${message.id}`)
101
+ for await (const partFile of new Bun.Glob(`storage/session/part/${session.id}/${message.id}/*.json`).scan(
102
+ {
103
+ cwd: fullProjectDir,
104
+ absolute: true,
105
+ },
106
+ )) {
107
+ const dest = path.join(dir, "part", message.id, path.basename(partFile))
108
+ const part = await Bun.file(partFile).json()
109
+ log.info("copying", {
110
+ partFile,
111
+ dest,
112
+ })
113
+ await Bun.write(dest, JSON.stringify(part))
114
+ }
115
+ }
116
+ }
117
+ }
118
+ }
119
+ },
120
+ async (dir) => {
121
+ for await (const item of new Bun.Glob("session/*/*.json").scan({
122
+ cwd: dir,
123
+ absolute: true,
124
+ })) {
125
+ const session = await Bun.file(item).json()
126
+ if (!session.projectID) continue
127
+ if (!session.summary?.diffs) continue
128
+ const { diffs } = session.summary
129
+ await Bun.file(path.join(dir, "session_diff", session.id + ".json")).write(JSON.stringify(diffs))
130
+ await Bun.file(path.join(dir, "session", session.projectID, session.id + ".json")).write(
131
+ JSON.stringify({
132
+ ...session,
133
+ summary: {
134
+ additions: diffs.reduce((sum: any, x: any) => sum + x.additions, 0),
135
+ deletions: diffs.reduce((sum: any, x: any) => sum + x.deletions, 0),
136
+ },
137
+ }),
138
+ )
139
+ }
140
+ },
141
+ ]
142
+
143
+ const state = lazy(async () => {
144
+ const dir = path.join(Global.Path.data, "storage")
145
+ const migration = await Bun.file(path.join(dir, "migration"))
146
+ .json()
147
+ .then((x) => parseInt(x))
148
+ .catch(() => 0)
149
+ for (let index = migration; index < MIGRATIONS.length; index++) {
150
+ log.info("running migration", { index })
151
+ const migration = MIGRATIONS[index]
152
+ await migration(dir).catch(() => log.error("failed to run migration", { index }))
153
+ await Bun.write(path.join(dir, "migration"), (index + 1).toString())
154
+ }
155
+ return {
156
+ dir,
157
+ }
158
+ })
159
+
160
+ export async function remove(key: string[]) {
161
+ const dir = await state().then((x) => x.dir)
162
+ const target = path.join(dir, ...key) + ".json"
163
+ return withErrorHandling(async () => {
164
+ await fs.unlink(target).catch(() => {})
165
+ })
166
+ }
167
+
168
+ export async function read<T>(key: string[]) {
169
+ const dir = await state().then((x) => x.dir)
170
+ const target = path.join(dir, ...key) + ".json"
171
+ return withErrorHandling(async () => {
172
+ using _ = await Lock.read(target)
173
+ const result = await Bun.file(target).json()
174
+ return result as T
175
+ })
176
+ }
177
+
178
+ export async function update<T>(key: string[], fn: (draft: T) => void) {
179
+ const dir = await state().then((x) => x.dir)
180
+ const target = path.join(dir, ...key) + ".json"
181
+ return withErrorHandling(async () => {
182
+ using _ = await Lock.write(target)
183
+ const content = await Bun.file(target).json()
184
+ fn(content)
185
+ await Bun.write(target, JSON.stringify(content, null, 2))
186
+ return content as T
187
+ })
188
+ }
189
+
190
+ export async function write<T>(key: string[], content: T) {
191
+ const dir = await state().then((x) => x.dir)
192
+ const target = path.join(dir, ...key) + ".json"
193
+ return withErrorHandling(async () => {
194
+ using _ = await Lock.write(target)
195
+ await Bun.write(target, JSON.stringify(content, null, 2))
196
+ })
197
+ }
198
+
199
+ async function withErrorHandling<T>(body: () => Promise<T>) {
200
+ return body().catch((e) => {
201
+ if (!(e instanceof Error)) throw e
202
+ const errnoException = e as NodeJS.ErrnoException
203
+ if (errnoException.code === "ENOENT") {
204
+ throw new NotFoundError({ message: `Resource not found: ${errnoException.path}` })
205
+ }
206
+ throw e
207
+ })
208
+ }
209
+
210
+ const glob = new Bun.Glob("**/*")
211
+ export async function list(prefix: string[]) {
212
+ const dir = await state().then((x) => x.dir)
213
+ try {
214
+ const result = await Array.fromAsync(
215
+ glob.scan({
216
+ cwd: path.join(dir, ...prefix),
217
+ onlyFiles: true,
218
+ }),
219
+ ).then((results) => results.map((x) => [...prefix, ...x.slice(0, -5).split(path.sep)]))
220
+ result.sort()
221
+ return result
222
+ } catch {
223
+ return []
224
+ }
225
+ }
226
+ }