cerebras-cli 1.0.1 → 1.0.4

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 (314) hide show
  1. package/README.md +3 -5
  2. package/cerebras-cli-1.0.0.tgz +0 -0
  3. package/package.json +7 -88
  4. package/AGENTS.md +0 -27
  5. package/Dockerfile +0 -10
  6. package/bunfig.toml +0 -4
  7. package/parsers-config.ts +0 -239
  8. package/script/build.ts +0 -151
  9. package/script/postinstall.mjs +0 -122
  10. package/script/publish.ts +0 -256
  11. package/script/schema.ts +0 -47
  12. package/src/acp/README.md +0 -164
  13. package/src/acp/agent.ts +0 -812
  14. package/src/acp/session.ts +0 -70
  15. package/src/acp/types.ts +0 -22
  16. package/src/agent/agent.ts +0 -310
  17. package/src/agent/generate.txt +0 -75
  18. package/src/auth/index.ts +0 -70
  19. package/src/bun/index.ts +0 -152
  20. package/src/bus/global.ts +0 -10
  21. package/src/bus/index.ts +0 -142
  22. package/src/cli/bootstrap.ts +0 -17
  23. package/src/cli/cmd/acp.ts +0 -88
  24. package/src/cli/cmd/agent.ts +0 -165
  25. package/src/cli/cmd/auth.ts +0 -369
  26. package/src/cli/cmd/cmd.ts +0 -7
  27. package/src/cli/cmd/debug/config.ts +0 -15
  28. package/src/cli/cmd/debug/file.ts +0 -91
  29. package/src/cli/cmd/debug/index.ts +0 -41
  30. package/src/cli/cmd/debug/lsp.ts +0 -47
  31. package/src/cli/cmd/debug/ripgrep.ts +0 -83
  32. package/src/cli/cmd/debug/scrap.ts +0 -15
  33. package/src/cli/cmd/debug/snapshot.ts +0 -48
  34. package/src/cli/cmd/export.ts +0 -88
  35. package/src/cli/cmd/generate.ts +0 -38
  36. package/src/cli/cmd/github.ts +0 -1200
  37. package/src/cli/cmd/import.ts +0 -98
  38. package/src/cli/cmd/mcp.ts +0 -400
  39. package/src/cli/cmd/models.ts +0 -77
  40. package/src/cli/cmd/pr.ts +0 -112
  41. package/src/cli/cmd/run.ts +0 -342
  42. package/src/cli/cmd/serve.ts +0 -31
  43. package/src/cli/cmd/session.ts +0 -106
  44. package/src/cli/cmd/stats.ts +0 -298
  45. package/src/cli/cmd/tui/app.tsx +0 -732
  46. package/src/cli/cmd/tui/attach.ts +0 -25
  47. package/src/cli/cmd/tui/component/border.tsx +0 -21
  48. package/src/cli/cmd/tui/component/dialog-agent.tsx +0 -31
  49. package/src/cli/cmd/tui/component/dialog-command.tsx +0 -124
  50. package/src/cli/cmd/tui/component/dialog-feedback.tsx +0 -160
  51. package/src/cli/cmd/tui/component/dialog-mcp.tsx +0 -86
  52. package/src/cli/cmd/tui/component/dialog-model.tsx +0 -223
  53. package/src/cli/cmd/tui/component/dialog-notification.tsx +0 -78
  54. package/src/cli/cmd/tui/component/dialog-provider.tsx +0 -222
  55. package/src/cli/cmd/tui/component/dialog-session-list.tsx +0 -97
  56. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +0 -31
  57. package/src/cli/cmd/tui/component/dialog-status.tsx +0 -114
  58. package/src/cli/cmd/tui/component/dialog-tag.tsx +0 -44
  59. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +0 -50
  60. package/src/cli/cmd/tui/component/logo.tsx +0 -37
  61. package/src/cli/cmd/tui/component/notification-banner.tsx +0 -58
  62. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +0 -530
  63. package/src/cli/cmd/tui/component/prompt/history.tsx +0 -107
  64. package/src/cli/cmd/tui/component/prompt/index.tsx +0 -931
  65. package/src/cli/cmd/tui/context/args.tsx +0 -14
  66. package/src/cli/cmd/tui/context/directory.ts +0 -12
  67. package/src/cli/cmd/tui/context/exit.tsx +0 -23
  68. package/src/cli/cmd/tui/context/helper.tsx +0 -25
  69. package/src/cli/cmd/tui/context/keybind.tsx +0 -111
  70. package/src/cli/cmd/tui/context/kv.tsx +0 -49
  71. package/src/cli/cmd/tui/context/local.tsx +0 -339
  72. package/src/cli/cmd/tui/context/prompt.tsx +0 -18
  73. package/src/cli/cmd/tui/context/route.tsx +0 -45
  74. package/src/cli/cmd/tui/context/sdk.tsx +0 -75
  75. package/src/cli/cmd/tui/context/sync.tsx +0 -374
  76. package/src/cli/cmd/tui/context/theme/aura.json +0 -69
  77. package/src/cli/cmd/tui/context/theme/ayu.json +0 -80
  78. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +0 -233
  79. package/src/cli/cmd/tui/context/theme/catppuccin.json +0 -112
  80. package/src/cli/cmd/tui/context/theme/cobalt2.json +0 -228
  81. package/src/cli/cmd/tui/context/theme/dracula.json +0 -219
  82. package/src/cli/cmd/tui/context/theme/everforest.json +0 -241
  83. package/src/cli/cmd/tui/context/theme/flexoki.json +0 -237
  84. package/src/cli/cmd/tui/context/theme/github.json +0 -233
  85. package/src/cli/cmd/tui/context/theme/gruvbox.json +0 -95
  86. package/src/cli/cmd/tui/context/theme/kanagawa.json +0 -77
  87. package/src/cli/cmd/tui/context/theme/material.json +0 -235
  88. package/src/cli/cmd/tui/context/theme/matrix.json +0 -77
  89. package/src/cli/cmd/tui/context/theme/mercury.json +0 -252
  90. package/src/cli/cmd/tui/context/theme/monokai.json +0 -221
  91. package/src/cli/cmd/tui/context/theme/nightowl.json +0 -221
  92. package/src/cli/cmd/tui/context/theme/nord.json +0 -223
  93. package/src/cli/cmd/tui/context/theme/one-dark.json +0 -84
  94. package/src/cli/cmd/tui/context/theme/orng.json +0 -245
  95. package/src/cli/cmd/tui/context/theme/palenight.json +0 -222
  96. package/src/cli/cmd/tui/context/theme/rosepine.json +0 -234
  97. package/src/cli/cmd/tui/context/theme/solarized.json +0 -223
  98. package/src/cli/cmd/tui/context/theme/synthwave84.json +0 -226
  99. package/src/cli/cmd/tui/context/theme/tokyonight.json +0 -243
  100. package/src/cli/cmd/tui/context/theme/vercel.json +0 -245
  101. package/src/cli/cmd/tui/context/theme/vesper.json +0 -218
  102. package/src/cli/cmd/tui/context/theme/zenburn.json +0 -223
  103. package/src/cli/cmd/tui/context/theme.tsx +0 -1077
  104. package/src/cli/cmd/tui/event.ts +0 -39
  105. package/src/cli/cmd/tui/routes/home.tsx +0 -104
  106. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +0 -93
  107. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +0 -37
  108. package/src/cli/cmd/tui/routes/session/footer.tsx +0 -76
  109. package/src/cli/cmd/tui/routes/session/header.tsx +0 -183
  110. package/src/cli/cmd/tui/routes/session/index.tsx +0 -1703
  111. package/src/cli/cmd/tui/routes/session/sidebar.tsx +0 -586
  112. package/src/cli/cmd/tui/spawn.ts +0 -60
  113. package/src/cli/cmd/tui/thread.ts +0 -120
  114. package/src/cli/cmd/tui/ui/dialog-alert.tsx +0 -55
  115. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +0 -81
  116. package/src/cli/cmd/tui/ui/dialog-help.tsx +0 -36
  117. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +0 -75
  118. package/src/cli/cmd/tui/ui/dialog-select.tsx +0 -317
  119. package/src/cli/cmd/tui/ui/dialog.tsx +0 -170
  120. package/src/cli/cmd/tui/ui/spinner.ts +0 -368
  121. package/src/cli/cmd/tui/ui/toast.tsx +0 -100
  122. package/src/cli/cmd/tui/util/clipboard.ts +0 -127
  123. package/src/cli/cmd/tui/util/editor.ts +0 -32
  124. package/src/cli/cmd/tui/util/terminal.ts +0 -114
  125. package/src/cli/cmd/tui/worker.ts +0 -63
  126. package/src/cli/cmd/uninstall.ts +0 -344
  127. package/src/cli/cmd/upgrade.ts +0 -67
  128. package/src/cli/cmd/web.ts +0 -84
  129. package/src/cli/error.ts +0 -55
  130. package/src/cli/ui.ts +0 -84
  131. package/src/cli/upgrade.ts +0 -25
  132. package/src/command/index.ts +0 -79
  133. package/src/command/template/initialize.txt +0 -10
  134. package/src/command/template/review.txt +0 -73
  135. package/src/config/config.ts +0 -886
  136. package/src/config/markdown.ts +0 -41
  137. package/src/env/index.ts +0 -26
  138. package/src/file/fzf.ts +0 -124
  139. package/src/file/ignore.ts +0 -83
  140. package/src/file/index.ts +0 -326
  141. package/src/file/ripgrep.ts +0 -391
  142. package/src/file/time.ts +0 -38
  143. package/src/file/watcher.ts +0 -89
  144. package/src/flag/flag.ts +0 -28
  145. package/src/format/formatter.ts +0 -277
  146. package/src/format/index.ts +0 -137
  147. package/src/global/index.ts +0 -52
  148. package/src/id/id.ts +0 -73
  149. package/src/ide/index.ts +0 -75
  150. package/src/index.ts +0 -158
  151. package/src/installation/index.ts +0 -194
  152. package/src/lsp/client.ts +0 -215
  153. package/src/lsp/index.ts +0 -370
  154. package/src/lsp/language.ts +0 -111
  155. package/src/lsp/server.ts +0 -1327
  156. package/src/mcp/auth.ts +0 -82
  157. package/src/mcp/index.ts +0 -576
  158. package/src/mcp/oauth-callback.ts +0 -203
  159. package/src/mcp/oauth-provider.ts +0 -132
  160. package/src/notification/index.ts +0 -101
  161. package/src/patch/index.ts +0 -622
  162. package/src/permission/index.ts +0 -198
  163. package/src/plugin/index.ts +0 -95
  164. package/src/project/bootstrap.ts +0 -31
  165. package/src/project/instance.ts +0 -68
  166. package/src/project/project.ts +0 -133
  167. package/src/project/state.ts +0 -65
  168. package/src/project/vcs.ts +0 -77
  169. package/src/provider/auth.ts +0 -143
  170. package/src/provider/models-macro.ts +0 -11
  171. package/src/provider/models.ts +0 -93
  172. package/src/provider/provider.ts +0 -996
  173. package/src/provider/sdk/openai-compatible/src/README.md +0 -5
  174. package/src/provider/sdk/openai-compatible/src/index.ts +0 -2
  175. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +0 -100
  176. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +0 -303
  177. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +0 -27
  178. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +0 -18
  179. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +0 -22
  180. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +0 -207
  181. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +0 -1713
  182. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +0 -177
  183. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +0 -1
  184. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +0 -88
  185. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +0 -128
  186. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +0 -115
  187. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +0 -65
  188. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +0 -104
  189. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +0 -103
  190. package/src/provider/transform.ts +0 -406
  191. package/src/pty/index.ts +0 -226
  192. package/src/ratelimit/index.ts +0 -185
  193. package/src/server/error.ts +0 -36
  194. package/src/server/project.ts +0 -50
  195. package/src/server/server.ts +0 -2463
  196. package/src/server/tui.ts +0 -71
  197. package/src/session/compaction.ts +0 -257
  198. package/src/session/index.ts +0 -470
  199. package/src/session/message-v2.ts +0 -641
  200. package/src/session/message.ts +0 -189
  201. package/src/session/processor.ts +0 -443
  202. package/src/session/prompt/anthropic-20250930.txt +0 -166
  203. package/src/session/prompt/anthropic.txt +0 -105
  204. package/src/session/prompt/anthropic_spoof.txt +0 -1
  205. package/src/session/prompt/beast.txt +0 -147
  206. package/src/session/prompt/build-switch.txt +0 -5
  207. package/src/session/prompt/codex.txt +0 -318
  208. package/src/session/prompt/compaction.txt +0 -12
  209. package/src/session/prompt/copilot-gpt-5.txt +0 -143
  210. package/src/session/prompt/gemini.txt +0 -155
  211. package/src/session/prompt/max-steps.txt +0 -16
  212. package/src/session/prompt/plan-reminder-anthropic.txt +0 -67
  213. package/src/session/prompt/plan.txt +0 -26
  214. package/src/session/prompt/polaris.txt +0 -107
  215. package/src/session/prompt/qwen.txt +0 -109
  216. package/src/session/prompt/summarize.txt +0 -4
  217. package/src/session/prompt/title.txt +0 -36
  218. package/src/session/prompt.ts +0 -1541
  219. package/src/session/retry.ts +0 -82
  220. package/src/session/revert.ts +0 -108
  221. package/src/session/status.ts +0 -75
  222. package/src/session/summary.ts +0 -203
  223. package/src/session/system.ts +0 -148
  224. package/src/session/todo.ts +0 -36
  225. package/src/share/share-next.ts +0 -195
  226. package/src/share/share.ts +0 -87
  227. package/src/snapshot/index.ts +0 -197
  228. package/src/storage/storage.ts +0 -226
  229. package/src/telemetry/index.ts +0 -232
  230. package/src/tool/bash.ts +0 -365
  231. package/src/tool/bash.txt +0 -128
  232. package/src/tool/batch.ts +0 -173
  233. package/src/tool/batch.txt +0 -28
  234. package/src/tool/codesearch.ts +0 -138
  235. package/src/tool/codesearch.txt +0 -12
  236. package/src/tool/edit.ts +0 -674
  237. package/src/tool/edit.txt +0 -10
  238. package/src/tool/glob.ts +0 -65
  239. package/src/tool/glob.txt +0 -6
  240. package/src/tool/grep.ts +0 -120
  241. package/src/tool/grep.txt +0 -8
  242. package/src/tool/invalid.ts +0 -17
  243. package/src/tool/ls.ts +0 -110
  244. package/src/tool/ls.txt +0 -1
  245. package/src/tool/lsp-diagnostics.ts +0 -26
  246. package/src/tool/lsp-diagnostics.txt +0 -1
  247. package/src/tool/lsp-hover.ts +0 -31
  248. package/src/tool/lsp-hover.txt +0 -1
  249. package/src/tool/multiedit.ts +0 -46
  250. package/src/tool/multiedit.txt +0 -41
  251. package/src/tool/patch.ts +0 -233
  252. package/src/tool/patch.txt +0 -1
  253. package/src/tool/read.ts +0 -217
  254. package/src/tool/read.txt +0 -12
  255. package/src/tool/registry.ts +0 -148
  256. package/src/tool/task.ts +0 -135
  257. package/src/tool/task.txt +0 -60
  258. package/src/tool/todo.ts +0 -39
  259. package/src/tool/todoread.txt +0 -14
  260. package/src/tool/todowrite.txt +0 -167
  261. package/src/tool/tool.ts +0 -66
  262. package/src/tool/webfetch.ts +0 -187
  263. package/src/tool/webfetch.txt +0 -14
  264. package/src/tool/websearch.ts +0 -150
  265. package/src/tool/websearch.txt +0 -11
  266. package/src/tool/write.ts +0 -99
  267. package/src/tool/write.txt +0 -8
  268. package/src/types/shims.d.ts +0 -3
  269. package/src/util/color.ts +0 -19
  270. package/src/util/context.ts +0 -25
  271. package/src/util/defer.ts +0 -12
  272. package/src/util/eventloop.ts +0 -20
  273. package/src/util/filesystem.ts +0 -69
  274. package/src/util/fn.ts +0 -11
  275. package/src/util/iife.ts +0 -3
  276. package/src/util/keybind.ts +0 -79
  277. package/src/util/lazy.ts +0 -11
  278. package/src/util/locale.ts +0 -81
  279. package/src/util/lock.ts +0 -98
  280. package/src/util/log.ts +0 -177
  281. package/src/util/queue.ts +0 -32
  282. package/src/util/rpc.ts +0 -42
  283. package/src/util/scrap.ts +0 -10
  284. package/src/util/signal.ts +0 -12
  285. package/src/util/timeout.ts +0 -14
  286. package/src/util/token.ts +0 -7
  287. package/src/util/wildcard.ts +0 -54
  288. package/sst-env.d.ts +0 -9
  289. package/test/bun.test.ts +0 -53
  290. package/test/config/agent-color.test.ts +0 -66
  291. package/test/config/config.test.ts +0 -503
  292. package/test/config/markdown.test.ts +0 -89
  293. package/test/file/ignore.test.ts +0 -10
  294. package/test/fixture/fixture.ts +0 -28
  295. package/test/fixture/lsp/fake-lsp-server.js +0 -77
  296. package/test/ide/ide.test.ts +0 -82
  297. package/test/keybind.test.ts +0 -317
  298. package/test/lsp/client.test.ts +0 -95
  299. package/test/patch/patch.test.ts +0 -348
  300. package/test/preload.ts +0 -38
  301. package/test/project/project.test.ts +0 -42
  302. package/test/provider/provider.test.ts +0 -1809
  303. package/test/provider/transform.test.ts +0 -305
  304. package/test/session/retry.test.ts +0 -61
  305. package/test/session/session.test.ts +0 -71
  306. package/test/snapshot/snapshot.test.ts +0 -939
  307. package/test/tool/__snapshots__/tool.test.ts.snap +0 -9
  308. package/test/tool/bash.test.ts +0 -55
  309. package/test/tool/patch.test.ts +0 -259
  310. package/test/util/iife.test.ts +0 -36
  311. package/test/util/lazy.test.ts +0 -50
  312. package/test/util/timeout.test.ts +0 -21
  313. package/test/util/wildcard.test.ts +0 -55
  314. package/tsconfig.json +0 -17
@@ -1,195 +0,0 @@
1
- import { Bus } from "@/bus"
2
- import { Config } from "@/config/config"
3
- import { ulid } from "ulid"
4
- import { Provider } from "@/provider/provider"
5
- import { Session } from "@/session"
6
- import { MessageV2 } from "@/session/message-v2"
7
- import { Storage } from "@/storage/storage"
8
- import { Log } from "@/util/log"
9
- import type * as SDK from "@opencode-ai/sdk/v2"
10
-
11
- export namespace ShareNext {
12
- const log = Log.create({ service: "share-next" })
13
-
14
- export async function init() {
15
- const config = await Config.get()
16
- if (!config.enterprise) return
17
- Bus.subscribe(Session.Event.Updated, async (evt) => {
18
- await sync(evt.properties.info.id, [
19
- {
20
- type: "session",
21
- data: evt.properties.info,
22
- },
23
- ])
24
- })
25
- Bus.subscribe(MessageV2.Event.Updated, async (evt) => {
26
- await sync(evt.properties.info.sessionID, [
27
- {
28
- type: "message",
29
- data: evt.properties.info,
30
- },
31
- ])
32
- if (evt.properties.info.role === "user") {
33
- await sync(evt.properties.info.sessionID, [
34
- {
35
- type: "model",
36
- data: [
37
- await Provider.getModel(evt.properties.info.model.providerID, evt.properties.info.model.modelID).then(
38
- (m) => m,
39
- ),
40
- ],
41
- },
42
- ])
43
- }
44
- })
45
- Bus.subscribe(MessageV2.Event.PartUpdated, async (evt) => {
46
- await sync(evt.properties.part.sessionID, [
47
- {
48
- type: "part",
49
- data: evt.properties.part,
50
- },
51
- ])
52
- })
53
- Bus.subscribe(Session.Event.Diff, async (evt) => {
54
- await sync(evt.properties.sessionID, [
55
- {
56
- type: "session_diff",
57
- data: evt.properties.diff,
58
- },
59
- ])
60
- })
61
- }
62
-
63
- export async function create(sessionID: string) {
64
- log.info("creating share", { sessionID })
65
- const url = await Config.get().then((x) => x.enterprise!.url)
66
- const result = await fetch(`${url}/api/share`, {
67
- method: "POST",
68
- headers: {
69
- "Content-Type": "application/json",
70
- },
71
- body: JSON.stringify({ sessionID: sessionID }),
72
- })
73
- .then((x) => x.json())
74
- .then((x) => x as { id: string; url: string; secret: string })
75
- await Storage.write(["session_share", sessionID], result)
76
- fullSync(sessionID)
77
- return result
78
- }
79
-
80
- function get(sessionID: string) {
81
- return Storage.read<{
82
- id: string
83
- secret: string
84
- url: string
85
- }>(["session_share", sessionID])
86
- }
87
-
88
- type Data =
89
- | {
90
- type: "session"
91
- data: SDK.Session
92
- }
93
- | {
94
- type: "message"
95
- data: SDK.Message
96
- }
97
- | {
98
- type: "part"
99
- data: SDK.Part
100
- }
101
- | {
102
- type: "session_diff"
103
- data: SDK.FileDiff[]
104
- }
105
- | {
106
- type: "model"
107
- data: SDK.Model[]
108
- }
109
-
110
- const queue = new Map<string, { timeout: NodeJS.Timeout; data: Map<string, Data> }>()
111
- async function sync(sessionID: string, data: Data[]) {
112
- const existing = queue.get(sessionID)
113
- if (existing) {
114
- for (const item of data) {
115
- existing.data.set("id" in item ? (item.id as string) : ulid(), item)
116
- }
117
- return
118
- }
119
-
120
- const dataMap = new Map<string, Data>()
121
- for (const item of data) {
122
- dataMap.set("id" in item ? (item.id as string) : ulid(), item)
123
- }
124
-
125
- const timeout = setTimeout(async () => {
126
- const queued = queue.get(sessionID)
127
- if (!queued) return
128
- queue.delete(sessionID)
129
- const url = await Config.get().then((x) => x.enterprise!.url)
130
- const share = await get(sessionID)
131
- if (!share) return
132
-
133
- await fetch(`${url}/api/share/${share.id}/sync`, {
134
- method: "POST",
135
- headers: {
136
- "Content-Type": "application/json",
137
- },
138
- body: JSON.stringify({
139
- secret: share.secret,
140
- data: Array.from(queued.data.values()),
141
- }),
142
- })
143
- }, 1000)
144
- queue.set(sessionID, { timeout, data: dataMap })
145
- }
146
-
147
- export async function remove(sessionID: string) {
148
- log.info("removing share", { sessionID })
149
- const url = await Config.get().then((x) => x.enterprise!.url)
150
- const share = await get(sessionID)
151
- if (!share) return
152
- await fetch(`${url}/api/share/${share.id}`, {
153
- method: "DELETE",
154
- headers: {
155
- "Content-Type": "application/json",
156
- },
157
- body: JSON.stringify({
158
- secret: share.secret,
159
- }),
160
- })
161
- await Storage.remove(["session_share", share.id])
162
- }
163
-
164
- async function fullSync(sessionID: string) {
165
- log.info("full sync", { sessionID })
166
- const session = await Session.get(sessionID)
167
- const diffs = await Session.diff(sessionID)
168
- const messages = await Array.fromAsync(MessageV2.stream(sessionID))
169
- const models = await Promise.all(
170
- messages
171
- .filter((m) => m.info.role === "user")
172
- .map((m) => (m.info as SDK.UserMessage).model)
173
- .map((m) => Provider.getModel(m.providerID, m.modelID).then((m) => m)),
174
- )
175
- await sync(sessionID, [
176
- {
177
- type: "session",
178
- data: session,
179
- },
180
- ...messages.map((x) => ({
181
- type: "message" as const,
182
- data: x.info,
183
- })),
184
- ...messages.flatMap((x) => x.parts.map((y) => ({ type: "part" as const, data: y }))),
185
- {
186
- type: "session_diff",
187
- data: diffs,
188
- },
189
- {
190
- type: "model",
191
- data: models,
192
- },
193
- ])
194
- }
195
- }
@@ -1,87 +0,0 @@
1
- import { Bus } from "../bus"
2
- import { Installation } from "../installation"
3
- import { Session } from "../session"
4
- import { MessageV2 } from "../session/message-v2"
5
- import { Log } from "../util/log"
6
-
7
- export namespace Share {
8
- const log = Log.create({ service: "share" })
9
-
10
- let queue: Promise<void> = Promise.resolve()
11
- const pending = new Map<string, any>()
12
-
13
- export async function sync(key: string, content: any) {
14
- const [root, ...splits] = key.split("/")
15
- if (root !== "session") return
16
- const [sub, sessionID] = splits
17
- if (sub === "share") return
18
- const share = await Session.getShare(sessionID).catch(() => {})
19
- if (!share) return
20
- const { secret } = share
21
- pending.set(key, content)
22
- queue = queue
23
- .then(async () => {
24
- const content = pending.get(key)
25
- if (content === undefined) return
26
- pending.delete(key)
27
-
28
- return fetch(`${URL}/share_sync`, {
29
- method: "POST",
30
- body: JSON.stringify({
31
- sessionID: sessionID,
32
- secret,
33
- key: key,
34
- content,
35
- }),
36
- })
37
- })
38
- .then((x) => {
39
- if (x) {
40
- log.info("synced", {
41
- key: key,
42
- status: x.status,
43
- })
44
- }
45
- })
46
- }
47
-
48
- export function init() {
49
- Bus.subscribe(Session.Event.Updated, async (evt) => {
50
- await sync("session/info/" + evt.properties.info.id, evt.properties.info)
51
- })
52
- Bus.subscribe(MessageV2.Event.Updated, async (evt) => {
53
- await sync("session/message/" + evt.properties.info.sessionID + "/" + evt.properties.info.id, evt.properties.info)
54
- })
55
- Bus.subscribe(MessageV2.Event.PartUpdated, async (evt) => {
56
- await sync(
57
- "session/part/" +
58
- evt.properties.part.sessionID +
59
- "/" +
60
- evt.properties.part.messageID +
61
- "/" +
62
- evt.properties.part.id,
63
- evt.properties.part,
64
- )
65
- })
66
- }
67
-
68
- export const URL =
69
- process.env["OPENCODE_API"] ??
70
- (Installation.isPreview() || Installation.isLocal() ? "https://api.dev.opencode.ai" : "https://api.opencode.ai")
71
-
72
- export async function create(sessionID: string) {
73
- return fetch(`${URL}/share_create`, {
74
- method: "POST",
75
- body: JSON.stringify({ sessionID: sessionID }),
76
- })
77
- .then((x) => x.json())
78
- .then((x) => x as { url: string; secret: string })
79
- }
80
-
81
- export async function remove(sessionID: string, secret: string) {
82
- return fetch(`${URL}/share_delete`, {
83
- method: "POST",
84
- body: JSON.stringify({ sessionID, secret }),
85
- }).then((x) => x.json())
86
- }
87
- }
@@ -1,197 +0,0 @@
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
- }
@@ -1,226 +0,0 @@
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
- }