cerebras-cli 1.0.5 → 1.0.138

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 (322) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +10 -0
  3. package/README.md +5 -3
  4. package/bin/{opencode.cjs → opencode} +4 -4
  5. package/bunfig.toml +4 -0
  6. package/package.json +89 -32
  7. package/parsers-config.ts +239 -0
  8. package/script/build.ts +151 -0
  9. package/script/postinstall.mjs +122 -0
  10. package/script/publish.ts +256 -0
  11. package/script/schema.ts +47 -0
  12. package/snake_game.py +111 -0
  13. package/src/acp/README.md +164 -0
  14. package/src/acp/agent.ts +812 -0
  15. package/src/acp/session.ts +70 -0
  16. package/src/acp/types.ts +22 -0
  17. package/src/agent/agent.ts +310 -0
  18. package/src/agent/generate.txt +75 -0
  19. package/src/auth/index.ts +70 -0
  20. package/src/bun/index.ts +152 -0
  21. package/src/bus/global.ts +10 -0
  22. package/src/bus/index.ts +142 -0
  23. package/src/cli/bootstrap.ts +17 -0
  24. package/src/cli/cmd/acp.ts +88 -0
  25. package/src/cli/cmd/agent.ts +165 -0
  26. package/src/cli/cmd/auth.ts +369 -0
  27. package/src/cli/cmd/cmd.ts +7 -0
  28. package/src/cli/cmd/debug/config.ts +15 -0
  29. package/src/cli/cmd/debug/file.ts +91 -0
  30. package/src/cli/cmd/debug/index.ts +43 -0
  31. package/src/cli/cmd/debug/lsp.ts +47 -0
  32. package/src/cli/cmd/debug/ripgrep.ts +83 -0
  33. package/src/cli/cmd/debug/scrap.ts +15 -0
  34. package/src/cli/cmd/debug/skill.ts +36 -0
  35. package/src/cli/cmd/debug/snapshot.ts +48 -0
  36. package/src/cli/cmd/export.ts +88 -0
  37. package/src/cli/cmd/generate.ts +38 -0
  38. package/src/cli/cmd/github.ts +1200 -0
  39. package/src/cli/cmd/import.ts +98 -0
  40. package/src/cli/cmd/mcp.ts +400 -0
  41. package/src/cli/cmd/models.ts +77 -0
  42. package/src/cli/cmd/pr.ts +112 -0
  43. package/src/cli/cmd/run.ts +342 -0
  44. package/src/cli/cmd/serve.ts +31 -0
  45. package/src/cli/cmd/session.ts +106 -0
  46. package/src/cli/cmd/stats.ts +298 -0
  47. package/src/cli/cmd/tui/app.tsx +833 -0
  48. package/src/cli/cmd/tui/attach.ts +25 -0
  49. package/src/cli/cmd/tui/component/border.tsx +21 -0
  50. package/src/cli/cmd/tui/component/cerebras-onboarding.tsx +225 -0
  51. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  52. package/src/cli/cmd/tui/component/dialog-command.tsx +124 -0
  53. package/src/cli/cmd/tui/component/dialog-feedback.tsx +160 -0
  54. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  55. package/src/cli/cmd/tui/component/dialog-model.tsx +223 -0
  56. package/src/cli/cmd/tui/component/dialog-notification.tsx +78 -0
  57. package/src/cli/cmd/tui/component/dialog-provider.tsx +222 -0
  58. package/src/cli/cmd/tui/component/dialog-session-list.tsx +97 -0
  59. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  60. package/src/cli/cmd/tui/component/dialog-status.tsx +114 -0
  61. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  62. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  63. package/src/cli/cmd/tui/component/logo.tsx +43 -0
  64. package/src/cli/cmd/tui/component/notification-banner.tsx +58 -0
  65. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +530 -0
  66. package/src/cli/cmd/tui/component/prompt/history.tsx +107 -0
  67. package/src/cli/cmd/tui/component/prompt/index.tsx +931 -0
  68. package/src/cli/cmd/tui/component/quickstart-onboarding.tsx +116 -0
  69. package/src/cli/cmd/tui/context/args.tsx +14 -0
  70. package/src/cli/cmd/tui/context/directory.ts +12 -0
  71. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  72. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  73. package/src/cli/cmd/tui/context/keybind.tsx +111 -0
  74. package/src/cli/cmd/tui/context/kv.tsx +49 -0
  75. package/src/cli/cmd/tui/context/local.tsx +338 -0
  76. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  77. package/src/cli/cmd/tui/context/route.tsx +45 -0
  78. package/src/cli/cmd/tui/context/sdk.tsx +75 -0
  79. package/src/cli/cmd/tui/context/sync.tsx +374 -0
  80. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  81. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  82. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  83. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  84. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  85. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  86. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  87. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  88. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  89. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  90. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  91. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  92. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  93. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  94. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  95. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  96. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  97. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  98. package/src/cli/cmd/tui/context/theme/orng.json +245 -0
  99. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  100. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  101. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  102. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  103. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  104. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  105. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  106. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  107. package/src/cli/cmd/tui/context/theme.tsx +1077 -0
  108. package/src/cli/cmd/tui/event.ts +39 -0
  109. package/src/cli/cmd/tui/routes/home.tsx +150 -0
  110. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +93 -0
  111. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +37 -0
  112. package/src/cli/cmd/tui/routes/session/footer.tsx +76 -0
  113. package/src/cli/cmd/tui/routes/session/header.tsx +181 -0
  114. package/src/cli/cmd/tui/routes/session/index.tsx +1695 -0
  115. package/src/cli/cmd/tui/routes/session/sidebar.tsx +686 -0
  116. package/src/cli/cmd/tui/spawn.ts +60 -0
  117. package/src/cli/cmd/tui/thread.ts +120 -0
  118. package/src/cli/cmd/tui/ui/dialog-alert.tsx +55 -0
  119. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +81 -0
  120. package/src/cli/cmd/tui/ui/dialog-help.tsx +36 -0
  121. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +75 -0
  122. package/src/cli/cmd/tui/ui/dialog-select.tsx +317 -0
  123. package/src/cli/cmd/tui/ui/dialog.tsx +170 -0
  124. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  125. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  126. package/src/cli/cmd/tui/util/clipboard.ts +127 -0
  127. package/src/cli/cmd/tui/util/editor.ts +32 -0
  128. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  129. package/src/cli/cmd/tui/worker.ts +63 -0
  130. package/src/cli/cmd/uninstall.ts +344 -0
  131. package/src/cli/cmd/upgrade.ts +67 -0
  132. package/src/cli/cmd/web.ts +84 -0
  133. package/src/cli/error.ts +55 -0
  134. package/src/cli/ui.ts +84 -0
  135. package/src/cli/upgrade.ts +25 -0
  136. package/src/command/index.ts +79 -0
  137. package/src/command/template/initialize.txt +10 -0
  138. package/src/command/template/review.txt +73 -0
  139. package/src/config/config.ts +886 -0
  140. package/src/config/markdown.ts +41 -0
  141. package/src/env/index.ts +26 -0
  142. package/src/file/fzf.ts +124 -0
  143. package/src/file/ignore.ts +83 -0
  144. package/src/file/index.ts +326 -0
  145. package/src/file/ripgrep.ts +391 -0
  146. package/src/file/time.ts +38 -0
  147. package/src/file/watcher.ts +89 -0
  148. package/src/flag/flag.ts +29 -0
  149. package/src/format/formatter.ts +277 -0
  150. package/src/format/index.ts +137 -0
  151. package/src/global/index.ts +52 -0
  152. package/src/id/id.ts +73 -0
  153. package/src/ide/index.ts +75 -0
  154. package/src/index.ts +158 -0
  155. package/src/installation/index.ts +194 -0
  156. package/src/lsp/client.ts +215 -0
  157. package/src/lsp/index.ts +370 -0
  158. package/src/lsp/language.ts +111 -0
  159. package/src/lsp/server.ts +1327 -0
  160. package/src/mcp/auth.ts +82 -0
  161. package/src/mcp/index.ts +576 -0
  162. package/src/mcp/oauth-callback.ts +203 -0
  163. package/src/mcp/oauth-provider.ts +132 -0
  164. package/src/notification/index.ts +101 -0
  165. package/src/patch/index.ts +622 -0
  166. package/src/permission/index.ts +198 -0
  167. package/src/plugin/index.ts +95 -0
  168. package/src/project/bootstrap.ts +31 -0
  169. package/src/project/instance.ts +68 -0
  170. package/src/project/project.ts +133 -0
  171. package/src/project/state.ts +65 -0
  172. package/src/project/vcs.ts +77 -0
  173. package/src/provider/auth.ts +143 -0
  174. package/src/provider/models-macro.ts +11 -0
  175. package/src/provider/models.ts +93 -0
  176. package/src/provider/provider.ts +1005 -0
  177. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  178. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  179. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  180. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  181. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +27 -0
  182. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  183. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  184. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  185. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1713 -0
  186. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  187. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  188. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  189. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  190. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  191. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  192. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  193. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  194. package/src/provider/transform.ts +406 -0
  195. package/src/pty/index.ts +226 -0
  196. package/src/ratelimit/index.ts +185 -0
  197. package/src/server/error.ts +36 -0
  198. package/src/server/project.ts +50 -0
  199. package/src/server/server.ts +2463 -0
  200. package/src/server/tui.ts +71 -0
  201. package/src/session/compaction.ts +257 -0
  202. package/src/session/index.ts +470 -0
  203. package/src/session/message-v2.ts +641 -0
  204. package/src/session/message.ts +189 -0
  205. package/src/session/processor.ts +448 -0
  206. package/src/session/prompt/anthropic-20250930.txt +166 -0
  207. package/src/session/prompt/anthropic.txt +105 -0
  208. package/src/session/prompt/anthropic_spoof.txt +1 -0
  209. package/src/session/prompt/beast.txt +147 -0
  210. package/src/session/prompt/build-switch.txt +5 -0
  211. package/src/session/prompt/codex.txt +318 -0
  212. package/src/session/prompt/compaction.txt +12 -0
  213. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  214. package/src/session/prompt/gemini.txt +155 -0
  215. package/src/session/prompt/max-steps.txt +16 -0
  216. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  217. package/src/session/prompt/plan.txt +26 -0
  218. package/src/session/prompt/polaris.txt +107 -0
  219. package/src/session/prompt/qwen.txt +109 -0
  220. package/src/session/prompt/summarize.txt +4 -0
  221. package/src/session/prompt/title.txt +36 -0
  222. package/src/session/prompt.ts +1541 -0
  223. package/src/session/retry.ts +82 -0
  224. package/src/session/revert.ts +108 -0
  225. package/src/session/status.ts +75 -0
  226. package/src/session/summary.ts +203 -0
  227. package/src/session/system.ts +148 -0
  228. package/src/session/todo.ts +36 -0
  229. package/src/share/share-next.ts +195 -0
  230. package/src/share/share.ts +87 -0
  231. package/src/skill/index.ts +2 -0
  232. package/src/skill/skill.ts +138 -0
  233. package/src/snapshot/index.ts +197 -0
  234. package/src/storage/storage.ts +226 -0
  235. package/src/telemetry/index.ts +247 -0
  236. package/src/tool/bash.ts +365 -0
  237. package/src/tool/bash.txt +128 -0
  238. package/src/tool/batch.ts +173 -0
  239. package/src/tool/batch.txt +28 -0
  240. package/src/tool/codesearch.ts +138 -0
  241. package/src/tool/codesearch.txt +12 -0
  242. package/src/tool/edit.ts +674 -0
  243. package/src/tool/edit.txt +10 -0
  244. package/src/tool/glob.ts +65 -0
  245. package/src/tool/glob.txt +6 -0
  246. package/src/tool/grep.ts +120 -0
  247. package/src/tool/grep.txt +8 -0
  248. package/src/tool/invalid.ts +17 -0
  249. package/src/tool/ls.ts +110 -0
  250. package/src/tool/ls.txt +1 -0
  251. package/src/tool/lsp-diagnostics.ts +26 -0
  252. package/src/tool/lsp-diagnostics.txt +1 -0
  253. package/src/tool/lsp-hover.ts +31 -0
  254. package/src/tool/lsp-hover.txt +1 -0
  255. package/src/tool/multiedit.ts +46 -0
  256. package/src/tool/multiedit.txt +41 -0
  257. package/src/tool/patch.ts +233 -0
  258. package/src/tool/patch.txt +1 -0
  259. package/src/tool/read.ts +217 -0
  260. package/src/tool/read.txt +12 -0
  261. package/src/tool/registry.ts +150 -0
  262. package/src/tool/skill.ts +85 -0
  263. package/src/tool/task.ts +135 -0
  264. package/src/tool/task.txt +60 -0
  265. package/src/tool/todo.ts +39 -0
  266. package/src/tool/todoread.txt +14 -0
  267. package/src/tool/todowrite.txt +167 -0
  268. package/src/tool/tool.ts +66 -0
  269. package/src/tool/webfetch.ts +187 -0
  270. package/src/tool/webfetch.txt +14 -0
  271. package/src/tool/websearch.ts +150 -0
  272. package/src/tool/websearch.txt +11 -0
  273. package/src/tool/write.ts +99 -0
  274. package/src/tool/write.txt +8 -0
  275. package/src/types/shims.d.ts +3 -0
  276. package/src/util/color.ts +19 -0
  277. package/src/util/context.ts +25 -0
  278. package/src/util/defer.ts +12 -0
  279. package/src/util/eventloop.ts +20 -0
  280. package/src/util/filesystem.ts +69 -0
  281. package/src/util/fn.ts +11 -0
  282. package/src/util/iife.ts +3 -0
  283. package/src/util/keybind.ts +79 -0
  284. package/src/util/lazy.ts +11 -0
  285. package/src/util/locale.ts +81 -0
  286. package/src/util/lock.ts +98 -0
  287. package/src/util/log.ts +177 -0
  288. package/src/util/queue.ts +32 -0
  289. package/src/util/rpc.ts +42 -0
  290. package/src/util/scrap.ts +10 -0
  291. package/src/util/signal.ts +12 -0
  292. package/src/util/timeout.ts +14 -0
  293. package/src/util/token.ts +7 -0
  294. package/src/util/wildcard.ts +54 -0
  295. package/sst-env.d.ts +9 -0
  296. package/test/bun.test.ts +53 -0
  297. package/test/config/agent-color.test.ts +66 -0
  298. package/test/config/config.test.ts +503 -0
  299. package/test/config/markdown.test.ts +89 -0
  300. package/test/file/ignore.test.ts +10 -0
  301. package/test/fixture/fixture.ts +28 -0
  302. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  303. package/test/ide/ide.test.ts +82 -0
  304. package/test/keybind.test.ts +317 -0
  305. package/test/lsp/client.test.ts +95 -0
  306. package/test/patch/patch.test.ts +348 -0
  307. package/test/preload.ts +38 -0
  308. package/test/project/project.test.ts +42 -0
  309. package/test/provider/provider.test.ts +1809 -0
  310. package/test/provider/transform.test.ts +305 -0
  311. package/test/session/retry.test.ts +61 -0
  312. package/test/session/session.test.ts +71 -0
  313. package/test/snapshot/snapshot.test.ts +939 -0
  314. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  315. package/test/tool/bash.test.ts +55 -0
  316. package/test/tool/patch.test.ts +259 -0
  317. package/test/util/iife.test.ts +36 -0
  318. package/test/util/lazy.test.ts +50 -0
  319. package/test/util/timeout.test.ts +21 -0
  320. package/test/util/wildcard.test.ts +55 -0
  321. package/tsconfig.json +17 -0
  322. package/cerebras-cli-1.0.0.tgz +0 -0
@@ -0,0 +1,195 @@
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
+ }
@@ -0,0 +1,87 @@
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
+ }
@@ -0,0 +1,2 @@
1
+ export * from "./skill"
2
+
@@ -0,0 +1,138 @@
1
+ import z from "zod"
2
+ import path from "path"
3
+ import { stat } from "fs/promises"
4
+ import { Config } from "../config/config"
5
+ import { Instance } from "../project/instance"
6
+ import { NamedError } from "@opencode-ai/util/error"
7
+ import { ConfigMarkdown } from "../config/markdown"
8
+ import { Log } from "../util/log"
9
+ import { Global } from "@/global"
10
+ import { Filesystem } from "@/util/filesystem"
11
+ import { Flag } from "@/flag/flag"
12
+ import { Bus } from "@/bus"
13
+ import { Session } from "@/session"
14
+
15
+ export namespace Skill {
16
+ const log = Log.create({ service: "skill" })
17
+ export const Info = z.object({
18
+ name: z.string(),
19
+ description: z.string(),
20
+ location: z.string(),
21
+ })
22
+ export type Info = z.infer<typeof Info>
23
+
24
+ export const InvalidError = NamedError.create(
25
+ "SkillInvalidError",
26
+ z.object({
27
+ path: z.string(),
28
+ message: z.string().optional(),
29
+ issues: z.custom<z.core.$ZodIssue[]>().optional(),
30
+ }),
31
+ )
32
+
33
+ export const NameMismatchError = NamedError.create(
34
+ "SkillNameMismatchError",
35
+ z.object({
36
+ path: z.string(),
37
+ expected: z.string(),
38
+ actual: z.string(),
39
+ }),
40
+ )
41
+
42
+ const OPENCODE_SKILL_GLOB = new Bun.Glob("{skill,skills}/**/SKILL.md")
43
+ const CLAUDE_SKILL_GLOB = new Bun.Glob("skills/**/SKILL.md")
44
+
45
+ export const state = Instance.state(async () => {
46
+ const skills: Record<string, Info> = {}
47
+
48
+ const addSkill = async (match: string) => {
49
+ const md = await ConfigMarkdown.parse(match).catch((err) => {
50
+ const message = ConfigMarkdown.FrontmatterError.isInstance(err)
51
+ ? err.data.message
52
+ : `Failed to parse skill ${match}`
53
+ Bus.publish(Session.Event.Error, { error: new NamedError.Unknown({ message }).toObject() })
54
+ log.error("failed to load skill", { skill: match, err })
55
+ return undefined
56
+ })
57
+
58
+ if (!md) return
59
+
60
+ const parsed = Info.pick({ name: true, description: true }).safeParse(md.data)
61
+ if (!parsed.success) return
62
+
63
+ // Warn on duplicate skill names
64
+ if (skills[parsed.data.name]) {
65
+ log.warn("duplicate skill name", {
66
+ name: parsed.data.name,
67
+ existing: skills[parsed.data.name].location,
68
+ duplicate: match,
69
+ })
70
+ }
71
+
72
+ skills[parsed.data.name] = {
73
+ name: parsed.data.name,
74
+ description: parsed.data.description,
75
+ location: match,
76
+ }
77
+ }
78
+
79
+ // Scan .claude/skills/ directories (project-level)
80
+ const claudeDirs = await Array.fromAsync(
81
+ Filesystem.up({
82
+ targets: [".claude"],
83
+ start: Instance.directory,
84
+ stop: Instance.worktree,
85
+ }),
86
+ )
87
+ // Also include global ~/.claude/skills/
88
+ const globalClaude = `${Global.Path.home}/.claude`
89
+ const globalClaudeStats = await stat(globalClaude).catch(() => null)
90
+ if (globalClaudeStats?.isDirectory()) {
91
+ claudeDirs.push(globalClaude)
92
+ }
93
+
94
+ if (!Flag.OPENCODE_DISABLE_CLAUDE_CODE_SKILLS) {
95
+ for (const dir of claudeDirs) {
96
+ const matches = await Array.fromAsync(
97
+ CLAUDE_SKILL_GLOB.scan({
98
+ cwd: dir,
99
+ absolute: true,
100
+ onlyFiles: true,
101
+ followSymlinks: true,
102
+ dot: true,
103
+ }),
104
+ ).catch((error) => {
105
+ log.error("failed .claude directory scan for skills", { dir, error })
106
+ return []
107
+ })
108
+
109
+ for (const match of matches) {
110
+ await addSkill(match)
111
+ }
112
+ }
113
+ }
114
+
115
+ // Scan .opencode/skill/ directories
116
+ for (const dir of await Config.directories()) {
117
+ for await (const match of OPENCODE_SKILL_GLOB.scan({
118
+ cwd: dir,
119
+ absolute: true,
120
+ onlyFiles: true,
121
+ followSymlinks: true,
122
+ })) {
123
+ await addSkill(match)
124
+ }
125
+ }
126
+
127
+ return skills
128
+ })
129
+
130
+ export async function get(name: string) {
131
+ return state().then((x) => x[name])
132
+ }
133
+
134
+ export async function all() {
135
+ return state().then((x) => Object.values(x))
136
+ }
137
+ }
138
+
@@ -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
+ }