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,82 +0,0 @@
1
- import type { NamedError } from "@opencode-ai/util/error"
2
- import { MessageV2 } from "./message-v2"
3
-
4
- export namespace SessionRetry {
5
- // Hand-tuned backoff schedule (ms): 10s, 10s, 10s, 15s, 15s; after that, stop retrying
6
- const BACKOFF_SCHEDULE = [10_000, 10_000, 10_000, 15_000, 15_000]
7
- export const RETRY_MAX_DELAY = 60_000 // absolute cap
8
-
9
- export async function sleep(ms: number, signal: AbortSignal): Promise<void> {
10
- return new Promise((resolve, reject) => {
11
- const timeout = setTimeout(resolve, ms)
12
- signal.addEventListener(
13
- "abort",
14
- () => {
15
- clearTimeout(timeout)
16
- reject(new DOMException("Aborted", "AbortError"))
17
- },
18
- { once: true },
19
- )
20
- })
21
- }
22
-
23
- export function delay(attempt: number, error?: MessageV2.APIError) {
24
- if (attempt > BACKOFF_SCHEDULE.length) {
25
- return undefined
26
- }
27
- const idx = Math.min(attempt - 1, BACKOFF_SCHEDULE.length - 1)
28
- const baseDelay = BACKOFF_SCHEDULE[idx]
29
-
30
- if (error) {
31
- const headers = error.data.responseHeaders
32
- if (headers) {
33
- const retryAfterMs = headers["retry-after-ms"]
34
- if (retryAfterMs) {
35
- const parsedMs = Number.parseFloat(retryAfterMs)
36
- if (!Number.isNaN(parsedMs)) {
37
- return Math.min(parsedMs, baseDelay, RETRY_MAX_DELAY)
38
- }
39
- }
40
-
41
- const retryAfter = headers["retry-after"]
42
- if (retryAfter) {
43
- const parsedSeconds = Number.parseFloat(retryAfter)
44
- if (!Number.isNaN(parsedSeconds)) {
45
- // convert seconds to milliseconds
46
- return Math.min(Math.ceil(parsedSeconds * 1000), baseDelay, RETRY_MAX_DELAY)
47
- }
48
- // Try parsing as HTTP date format
49
- const parsed = Date.parse(retryAfter) - Date.now()
50
- if (!Number.isNaN(parsed) && parsed > 0) {
51
- return Math.min(Math.ceil(parsed), baseDelay, RETRY_MAX_DELAY)
52
- }
53
- }
54
-
55
- return Math.min(baseDelay, RETRY_MAX_DELAY)
56
- }
57
- }
58
-
59
- return baseDelay
60
- }
61
-
62
- export function retryable(error: ReturnType<NamedError["toObject"]>) {
63
- if (MessageV2.APIError.isInstance(error)) {
64
- if (!error.data.isRetryable) return undefined
65
- return error.data.message.includes("Overloaded") ? "Provider is overloaded" : error.data.message
66
- }
67
-
68
- if (typeof error.data?.message === "string") {
69
- try {
70
- const json = JSON.parse(error.data.message)
71
- if (json.type === "error" && json.error?.type === "too_many_requests") {
72
- return "Too Many Requests"
73
- }
74
- if (json.code === "Some resource has been exhausted") {
75
- return "Provider is overloaded"
76
- }
77
- } catch {}
78
- }
79
-
80
- return undefined
81
- }
82
- }
@@ -1,108 +0,0 @@
1
- import z from "zod"
2
- import { Identifier } from "../id/id"
3
- import { Snapshot } from "../snapshot"
4
- import { MessageV2 } from "./message-v2"
5
- import { Session } from "."
6
- import { Log } from "../util/log"
7
- import { splitWhen } from "remeda"
8
- import { Storage } from "../storage/storage"
9
- import { Bus } from "../bus"
10
- import { SessionPrompt } from "./prompt"
11
-
12
- export namespace SessionRevert {
13
- const log = Log.create({ service: "session.revert" })
14
-
15
- export const RevertInput = z.object({
16
- sessionID: Identifier.schema("session"),
17
- messageID: Identifier.schema("message"),
18
- partID: Identifier.schema("part").optional(),
19
- })
20
- export type RevertInput = z.infer<typeof RevertInput>
21
-
22
- export async function revert(input: RevertInput) {
23
- SessionPrompt.assertNotBusy(input.sessionID)
24
- const all = await Session.messages({ sessionID: input.sessionID })
25
- let lastUser: MessageV2.User | undefined
26
- const session = await Session.get(input.sessionID)
27
-
28
- let revert: Session.Info["revert"]
29
- const patches: Snapshot.Patch[] = []
30
- for (const msg of all) {
31
- if (msg.info.role === "user") lastUser = msg.info
32
- const remaining = []
33
- for (const part of msg.parts) {
34
- if (revert) {
35
- if (part.type === "patch") {
36
- patches.push(part)
37
- }
38
- continue
39
- }
40
-
41
- if (!revert) {
42
- if ((msg.info.id === input.messageID && !input.partID) || part.id === input.partID) {
43
- // if no useful parts left in message, same as reverting whole message
44
- const partID = remaining.some((item) => ["text", "tool"].includes(item.type)) ? input.partID : undefined
45
- revert = {
46
- messageID: !partID && lastUser ? lastUser.id : msg.info.id,
47
- partID,
48
- }
49
- }
50
- remaining.push(part)
51
- }
52
- }
53
- }
54
-
55
- if (revert) {
56
- const session = await Session.get(input.sessionID)
57
- revert.snapshot = session.revert?.snapshot ?? (await Snapshot.track())
58
- await Snapshot.revert(patches)
59
- if (revert.snapshot) revert.diff = await Snapshot.diff(revert.snapshot)
60
- return Session.update(input.sessionID, (draft) => {
61
- draft.revert = revert
62
- })
63
- }
64
- return session
65
- }
66
-
67
- export async function unrevert(input: { sessionID: string }) {
68
- log.info("unreverting", input)
69
- SessionPrompt.assertNotBusy(input.sessionID)
70
- const session = await Session.get(input.sessionID)
71
- if (!session.revert) return session
72
- if (session.revert.snapshot) await Snapshot.restore(session.revert.snapshot)
73
- const next = await Session.update(input.sessionID, (draft) => {
74
- draft.revert = undefined
75
- })
76
- return next
77
- }
78
-
79
- export async function cleanup(session: Session.Info) {
80
- if (!session.revert) return
81
- const sessionID = session.id
82
- let msgs = await Session.messages({ sessionID })
83
- const messageID = session.revert.messageID
84
- const [preserve, remove] = splitWhen(msgs, (x) => x.info.id === messageID)
85
- msgs = preserve
86
- for (const msg of remove) {
87
- await Storage.remove(["message", sessionID, msg.info.id])
88
- await Bus.publish(MessageV2.Event.Removed, { sessionID: sessionID, messageID: msg.info.id })
89
- }
90
- const last = preserve.at(-1)
91
- if (session.revert.partID && last) {
92
- const partID = session.revert.partID
93
- const [preserveParts, removeParts] = splitWhen(last.parts, (x) => x.id === partID)
94
- last.parts = preserveParts
95
- for (const part of removeParts) {
96
- await Storage.remove(["part", last.info.id, part.id])
97
- await Bus.publish(MessageV2.Event.PartRemoved, {
98
- sessionID: sessionID,
99
- messageID: last.info.id,
100
- partID: part.id,
101
- })
102
- }
103
- }
104
- await Session.update(sessionID, (draft) => {
105
- draft.revert = undefined
106
- })
107
- }
108
- }
@@ -1,75 +0,0 @@
1
- import { Bus } from "@/bus"
2
- import { Instance } from "@/project/instance"
3
- import z from "zod"
4
-
5
- export namespace SessionStatus {
6
- export const Info = z
7
- .union([
8
- z.object({
9
- type: z.literal("idle"),
10
- }),
11
- z.object({
12
- type: z.literal("retry"),
13
- attempt: z.number(),
14
- message: z.string(),
15
- next: z.number(),
16
- }),
17
- z.object({
18
- type: z.literal("busy"),
19
- }),
20
- ])
21
- .meta({
22
- ref: "SessionStatus",
23
- })
24
- export type Info = z.infer<typeof Info>
25
-
26
- export const Event = {
27
- Status: Bus.event(
28
- "session.status",
29
- z.object({
30
- sessionID: z.string(),
31
- status: Info,
32
- }),
33
- ),
34
- // deprecated
35
- Idle: Bus.event(
36
- "session.idle",
37
- z.object({
38
- sessionID: z.string(),
39
- }),
40
- ),
41
- }
42
-
43
- const state = Instance.state(() => {
44
- const data: Record<string, Info> = {}
45
- return data
46
- })
47
-
48
- export function get(sessionID: string) {
49
- return (
50
- state()[sessionID] ?? {
51
- type: "idle",
52
- }
53
- )
54
- }
55
-
56
- export function list() {
57
- return Object.values(state())
58
- }
59
-
60
- export function set(sessionID: string, status: Info) {
61
- Bus.publish(Event.Status, {
62
- sessionID,
63
- status,
64
- })
65
- if (status.type === "idle") {
66
- // deprecated
67
- Bus.publish(Event.Idle, {
68
- sessionID,
69
- })
70
- delete state()[sessionID]
71
- return
72
- }
73
- state()[sessionID] = status
74
- }
75
- }
@@ -1,203 +0,0 @@
1
- import { Provider } from "@/provider/provider"
2
- import { Config } from "@/config/config"
3
- import { fn } from "@/util/fn"
4
- import z from "zod"
5
- import { Session } from "."
6
- import { generateText, type ModelMessage } from "ai"
7
- import { MessageV2 } from "./message-v2"
8
- import { Identifier } from "@/id/id"
9
- import { Snapshot } from "@/snapshot"
10
- import { ProviderTransform } from "@/provider/transform"
11
- import { SystemPrompt } from "./system"
12
- import { Log } from "@/util/log"
13
- import path from "path"
14
- import { Instance } from "@/project/instance"
15
- import { Storage } from "@/storage/storage"
16
- import { Bus } from "@/bus"
17
- import { mergeDeep, pipe } from "remeda"
18
-
19
- export namespace SessionSummary {
20
- const log = Log.create({ service: "session.summary" })
21
-
22
- export const summarize = fn(
23
- z.object({
24
- sessionID: z.string(),
25
- messageID: z.string(),
26
- }),
27
- async (input) => {
28
- const all = await Session.messages({ sessionID: input.sessionID })
29
- await Promise.all([
30
- summarizeSession({ sessionID: input.sessionID, messages: all }),
31
- summarizeMessage({ messageID: input.messageID, messages: all }),
32
- ])
33
- },
34
- )
35
-
36
- async function summarizeSession(input: { sessionID: string; messages: MessageV2.WithParts[] }) {
37
- const files = new Set(
38
- input.messages
39
- .flatMap((x) => x.parts)
40
- .filter((x) => x.type === "patch")
41
- .flatMap((x) => x.files)
42
- .map((x) => path.relative(Instance.worktree, x)),
43
- )
44
- const diffs = await computeDiff({ messages: input.messages }).then((x) =>
45
- x.filter((x) => {
46
- return files.has(x.file)
47
- }),
48
- )
49
- await Session.update(input.sessionID, (draft) => {
50
- draft.summary = {
51
- additions: diffs.reduce((sum, x) => sum + x.additions, 0),
52
- deletions: diffs.reduce((sum, x) => sum + x.deletions, 0),
53
- files: diffs.length,
54
- }
55
- })
56
- await Storage.write(["session_diff", input.sessionID], diffs)
57
- Bus.publish(Session.Event.Diff, {
58
- sessionID: input.sessionID,
59
- diff: diffs,
60
- })
61
- }
62
-
63
- async function summarizeMessage(input: { messageID: string; messages: MessageV2.WithParts[] }) {
64
- const cfg = await Config.get()
65
- const messages = input.messages.filter(
66
- (m) => m.info.id === input.messageID || (m.info.role === "assistant" && m.info.parentID === input.messageID),
67
- )
68
- const msgWithParts = messages.find((m) => m.info.id === input.messageID)!
69
- const userMsg = msgWithParts.info as MessageV2.User
70
- const diffs = await computeDiff({ messages })
71
- userMsg.summary = {
72
- ...userMsg.summary,
73
- diffs,
74
- }
75
- await Session.updateMessage(userMsg)
76
-
77
- const assistantMsg = messages.find((m) => m.info.role === "assistant")!.info as MessageV2.Assistant
78
- const small =
79
- (await Provider.getSmallModel(assistantMsg.providerID)) ??
80
- (await Provider.getModel(assistantMsg.providerID, assistantMsg.modelID))
81
- const language = await Provider.getLanguage(small)
82
-
83
- const options = pipe(
84
- {},
85
- mergeDeep(ProviderTransform.options(small, assistantMsg.sessionID)),
86
- mergeDeep(ProviderTransform.smallOptions(small)),
87
- mergeDeep(small.options),
88
- )
89
-
90
- const textPart = msgWithParts.parts.find((p) => p.type === "text" && !p.synthetic) as MessageV2.TextPart
91
- if (textPart && !userMsg.summary?.title) {
92
- const result = await generateText({
93
- maxOutputTokens: small.capabilities.reasoning ? 1500 : 20,
94
- providerOptions: ProviderTransform.providerOptions(small.api.npm, small.providerID, options),
95
- messages: [
96
- ...SystemPrompt.title(small.providerID).map(
97
- (x): ModelMessage => ({
98
- role: "system",
99
- content: x,
100
- }),
101
- ),
102
- {
103
- role: "user" as const,
104
- content: `
105
- The following is the text to summarize:
106
- <text>
107
- ${textPart?.text ?? ""}
108
- </text>
109
- `,
110
- },
111
- ],
112
- headers: small.headers,
113
- model: language,
114
- experimental_telemetry: { isEnabled: cfg.experimental?.openTelemetry },
115
- })
116
- log.info("title", { title: result.text })
117
- userMsg.summary.title = result.text
118
- await Session.updateMessage(userMsg)
119
- }
120
-
121
- if (
122
- messages.some(
123
- (m) =>
124
- m.info.role === "assistant" && m.parts.some((p) => p.type === "step-finish" && p.reason !== "tool-calls"),
125
- )
126
- ) {
127
- let summary = messages
128
- .findLast((m) => m.info.role === "assistant")
129
- ?.parts.findLast((p) => p.type === "text")?.text
130
- if (!summary || diffs.length > 0) {
131
- for (const msg of messages) {
132
- for (const part of msg.parts) {
133
- if (part.type === "tool" && part.state.status === "completed") {
134
- part.state.output = "[TOOL OUTPUT PRUNED]"
135
- }
136
- }
137
- }
138
- const result = await generateText({
139
- model: language,
140
- maxOutputTokens: 100,
141
- providerOptions: ProviderTransform.providerOptions(small.api.npm, small.providerID, options),
142
- messages: [
143
- ...SystemPrompt.summarize(small.providerID).map(
144
- (x): ModelMessage => ({
145
- role: "system",
146
- content: x,
147
- }),
148
- ),
149
- ...MessageV2.toModelMessage(messages),
150
- {
151
- role: "user",
152
- content: `Summarize the above conversation according to your system prompts.`,
153
- },
154
- ],
155
- headers: small.headers,
156
- experimental_telemetry: { isEnabled: cfg.experimental?.openTelemetry },
157
- }).catch(() => {})
158
- if (result) summary = result.text
159
- }
160
- userMsg.summary.body = summary
161
- log.info("body", { body: summary })
162
- await Session.updateMessage(userMsg)
163
- }
164
- }
165
-
166
- export const diff = fn(
167
- z.object({
168
- sessionID: Identifier.schema("session"),
169
- messageID: Identifier.schema("message").optional(),
170
- }),
171
- async (input) => {
172
- return Storage.read<Snapshot.FileDiff[]>(["session_diff", input.sessionID]).catch(() => [])
173
- },
174
- )
175
-
176
- async function computeDiff(input: { messages: MessageV2.WithParts[] }) {
177
- let from: string | undefined
178
- let to: string | undefined
179
-
180
- // scan assistant messages to find earliest from and latest to
181
- // snapshot
182
- for (const item of input.messages) {
183
- if (!from) {
184
- for (const part of item.parts) {
185
- if (part.type === "step-start" && part.snapshot) {
186
- from = part.snapshot
187
- break
188
- }
189
- }
190
- }
191
-
192
- for (const part of item.parts) {
193
- if (part.type === "step-finish" && part.snapshot) {
194
- to = part.snapshot
195
- break
196
- }
197
- }
198
- }
199
-
200
- if (from && to) return Snapshot.diffFull(from, to)
201
- return []
202
- }
203
- }
@@ -1,148 +0,0 @@
1
- import { Ripgrep } from "../file/ripgrep"
2
- import { Global } from "../global"
3
- import { Filesystem } from "../util/filesystem"
4
- import { Config } from "../config/config"
5
-
6
- import { Instance } from "../project/instance"
7
- import path from "path"
8
- import os from "os"
9
-
10
- import PROMPT_ANTHROPIC from "./prompt/anthropic.txt"
11
- import PROMPT_ANTHROPIC_WITHOUT_TODO from "./prompt/qwen.txt"
12
- import PROMPT_POLARIS from "./prompt/polaris.txt"
13
- import PROMPT_BEAST from "./prompt/beast.txt"
14
- import PROMPT_GEMINI from "./prompt/gemini.txt"
15
- import PROMPT_ANTHROPIC_SPOOF from "./prompt/anthropic_spoof.txt"
16
- import PROMPT_COMPACTION from "./prompt/compaction.txt"
17
- import PROMPT_SUMMARIZE from "./prompt/summarize.txt"
18
- import PROMPT_TITLE from "./prompt/title.txt"
19
- import PROMPT_CODEX from "./prompt/codex.txt"
20
- import type { Provider } from "@/provider/provider"
21
-
22
- export namespace SystemPrompt {
23
- export function header(providerID: string) {
24
- if (providerID.includes("anthropic")) return [PROMPT_ANTHROPIC_SPOOF.trim()]
25
- return []
26
- }
27
-
28
- export function provider(model: Provider.Model) {
29
- if (model.api.id.includes("gpt-5")) return [PROMPT_CODEX]
30
- if (model.api.id.includes("gpt-") || model.api.id.includes("o1") || model.api.id.includes("o3"))
31
- return [PROMPT_BEAST]
32
- if (model.api.id.includes("gemini-")) return [PROMPT_GEMINI]
33
- if (model.api.id.includes("claude")) return [PROMPT_ANTHROPIC]
34
- if (model.api.id.includes("polaris-alpha")) return [PROMPT_POLARIS]
35
- return [PROMPT_ANTHROPIC_WITHOUT_TODO]
36
- }
37
-
38
- export async function environment() {
39
- const project = Instance.project
40
- return [
41
- [
42
- `Here is some useful information about the environment you are running in:`,
43
- `<env>`,
44
- ` Working directory: ${Instance.directory}`,
45
- ` Is directory a git repo: ${project.vcs === "git" ? "yes" : "no"}`,
46
- ` Platform: ${process.platform}`,
47
- ` Today's date: ${new Date().toDateString()}`,
48
- `</env>`,
49
- `<files>`,
50
- ` ${
51
- project.vcs === "git"
52
- ? await Ripgrep.tree({
53
- cwd: Instance.directory,
54
- limit: 200,
55
- })
56
- : ""
57
- }`,
58
- `</files>`,
59
- ].join("\n"),
60
- ]
61
- }
62
-
63
- const LOCAL_RULE_FILES = [
64
- "AGENTS.md",
65
- "CLAUDE.md",
66
- "CONTEXT.md", // deprecated
67
- ]
68
- const GLOBAL_RULE_FILES = [
69
- path.join(Global.Path.config, "AGENTS.md"),
70
- path.join(os.homedir(), ".claude", "CLAUDE.md"),
71
- ]
72
-
73
- export async function custom() {
74
- const config = await Config.get()
75
- const paths = new Set<string>()
76
-
77
- for (const localRuleFile of LOCAL_RULE_FILES) {
78
- const matches = await Filesystem.findUp(localRuleFile, Instance.directory, Instance.worktree)
79
- if (matches.length > 0) {
80
- matches.forEach((path) => paths.add(path))
81
- break
82
- }
83
- }
84
-
85
- for (const globalRuleFile of GLOBAL_RULE_FILES) {
86
- if (await Bun.file(globalRuleFile).exists()) {
87
- paths.add(globalRuleFile)
88
- break
89
- }
90
- }
91
-
92
- if (config.instructions) {
93
- for (let instruction of config.instructions) {
94
- if (instruction.startsWith("~/")) {
95
- instruction = path.join(os.homedir(), instruction.slice(2))
96
- }
97
- let matches: string[] = []
98
- if (path.isAbsolute(instruction)) {
99
- matches = await Array.fromAsync(
100
- new Bun.Glob(path.basename(instruction)).scan({
101
- cwd: path.dirname(instruction),
102
- absolute: true,
103
- onlyFiles: true,
104
- }),
105
- ).catch(() => [])
106
- } else {
107
- matches = await Filesystem.globUp(instruction, Instance.directory, Instance.worktree).catch(() => [])
108
- }
109
- matches.forEach((path) => paths.add(path))
110
- }
111
- }
112
-
113
- const found = Array.from(paths).map((p) =>
114
- Bun.file(p)
115
- .text()
116
- .catch(() => "")
117
- .then((x) => "Instructions from: " + p + "\n" + x),
118
- )
119
- return Promise.all(found).then((result) => result.filter(Boolean))
120
- }
121
-
122
- export function compaction(providerID: string) {
123
- switch (providerID) {
124
- case "anthropic":
125
- return [PROMPT_ANTHROPIC_SPOOF.trim(), PROMPT_COMPACTION]
126
- default:
127
- return [PROMPT_COMPACTION]
128
- }
129
- }
130
-
131
- export function summarize(providerID: string) {
132
- switch (providerID) {
133
- case "anthropic":
134
- return [PROMPT_ANTHROPIC_SPOOF.trim(), PROMPT_SUMMARIZE]
135
- default:
136
- return [PROMPT_SUMMARIZE]
137
- }
138
- }
139
-
140
- export function title(providerID: string) {
141
- switch (providerID) {
142
- case "anthropic":
143
- return [PROMPT_ANTHROPIC_SPOOF.trim(), PROMPT_TITLE]
144
- default:
145
- return [PROMPT_TITLE]
146
- }
147
- }
148
- }
@@ -1,36 +0,0 @@
1
- import z from "zod"
2
- import { Bus } from "../bus"
3
- import { Storage } from "../storage/storage"
4
-
5
- export namespace Todo {
6
- export const Info = z
7
- .object({
8
- content: z.string().describe("Brief description of the task"),
9
- status: z.string().describe("Current status of the task: pending, in_progress, completed, cancelled"),
10
- priority: z.string().describe("Priority level of the task: high, medium, low"),
11
- id: z.string().describe("Unique identifier for the todo item"),
12
- })
13
- .meta({ ref: "Todo" })
14
- export type Info = z.infer<typeof Info>
15
-
16
- export const Event = {
17
- Updated: Bus.event(
18
- "todo.updated",
19
- z.object({
20
- sessionID: z.string(),
21
- todos: z.array(Info),
22
- }),
23
- ),
24
- }
25
-
26
- export async function update(input: { sessionID: string; todos: Info[] }) {
27
- await Storage.write(["todo", input.sessionID], input.todos)
28
- Bus.publish(Event.Updated, input)
29
- }
30
-
31
- export async function get(sessionID: string) {
32
- return Storage.read<Info[]>(["todo", sessionID])
33
- .then((x) => x || [])
34
- .catch(() => [])
35
- }
36
- }