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,116 @@
1
+ import { createSignal, onMount, For } from "solid-js"
2
+ import { useTheme } from "../context/theme"
3
+ import { TextAttributes } from "@opentui/core"
4
+ import { useKeyboard, useTerminalDimensions } from "@opentui/solid"
5
+ import { useKV } from "../context/kv"
6
+
7
+ const STARTER_PROMPTS = [
8
+ { prompt: "Make a snake game in Python", icon: "" },
9
+ { prompt: "Organize my downloads folder", icon: "" },
10
+ { prompt: "Recreate the Google landing page UI", icon: "" },
11
+ ] as const
12
+
13
+ export function QuickStartOnboarding(props: { onSelect: (prompt: string) => void; onSkip: () => void }) {
14
+ const { theme } = useTheme()
15
+ const kv = useKV()
16
+ const dimensions = useTerminalDimensions()
17
+ const [selected, setSelected] = createSignal(0)
18
+ const [ready, setReady] = createSignal(false)
19
+
20
+ onMount(() => {
21
+ setTimeout(() => setReady(true), 300)
22
+ })
23
+
24
+ const markSeen = () => {
25
+ kv.set("hasSeenQuickStart", true)
26
+ }
27
+
28
+ const selectPrompt = (prompt: string) => {
29
+ markSeen()
30
+ props.onSelect(prompt)
31
+ }
32
+
33
+ const skip = () => {
34
+ markSeen()
35
+ props.onSkip()
36
+ }
37
+
38
+ useKeyboard((evt) => {
39
+ // Always prevent arrow keys from bubbling
40
+ if (evt.name === "up" || evt.name === "down") {
41
+ evt.preventDefault?.()
42
+ }
43
+
44
+ if (!ready()) return
45
+
46
+ if (evt.name === "up") {
47
+ setSelected((s) => (s > 0 ? s - 1 : STARTER_PROMPTS.length - 1))
48
+ } else if (evt.name === "down") {
49
+ setSelected((s) => (s < STARTER_PROMPTS.length - 1 ? s + 1 : 0))
50
+ } else if (evt.name === "return") {
51
+ evt.preventDefault?.()
52
+ selectPrompt(STARTER_PROMPTS[selected()].prompt)
53
+ } else if (evt.name === "escape") {
54
+ evt.preventDefault?.()
55
+ skip()
56
+ }
57
+ })
58
+
59
+ return (
60
+ <box
61
+ position="absolute"
62
+ top={0}
63
+ left={0}
64
+ width={dimensions().width}
65
+ height={dimensions().height}
66
+ backgroundColor={theme.background}
67
+ justifyContent="center"
68
+ alignItems="center"
69
+ flexDirection="column"
70
+ >
71
+ <box
72
+ width={Math.min(55, dimensions().width - 4)}
73
+ backgroundColor={theme.backgroundPanel}
74
+ paddingTop={2}
75
+ paddingBottom={2}
76
+ paddingLeft={3}
77
+ paddingRight={3}
78
+ gap={1}
79
+ >
80
+ <box gap={1}>
81
+ <text fg={theme.primary} attributes={TextAttributes.BOLD}>
82
+ ◆ Quick Start
83
+ </text>
84
+ <text fg={theme.text}>
85
+ Try one of these prompts to get started:
86
+ </text>
87
+ <box marginTop={1} gap={0}>
88
+ <For each={STARTER_PROMPTS}>
89
+ {(item, index) => {
90
+ const isSelected = () => selected() === index()
91
+ return (
92
+ <box
93
+ flexDirection="row"
94
+ backgroundColor={isSelected() ? theme.primary : undefined}
95
+ paddingLeft={1}
96
+ paddingRight={1}
97
+ >
98
+ <text fg={isSelected() ? theme.backgroundPanel : theme.text}>
99
+ {isSelected() ? "▸ " : " "}{item.icon} {item.prompt}
100
+ </text>
101
+ </box>
102
+ )
103
+ }}
104
+ </For>
105
+ </box>
106
+ <box marginTop={2}>
107
+ <text fg={theme.textMuted}>
108
+ ↑↓ navigate • enter select • esc skip
109
+ </text>
110
+ </box>
111
+ </box>
112
+ </box>
113
+ </box>
114
+ )
115
+ }
116
+
@@ -0,0 +1,14 @@
1
+ import { createSimpleContext } from "./helper"
2
+
3
+ export interface Args {
4
+ model?: string
5
+ agent?: string
6
+ prompt?: string
7
+ continue?: boolean
8
+ sessionID?: string
9
+ }
10
+
11
+ export const { use: useArgs, provider: ArgsProvider } = createSimpleContext({
12
+ name: "Args",
13
+ init: (props: Args) => props,
14
+ })
@@ -0,0 +1,12 @@
1
+ import { createMemo } from "solid-js"
2
+ import { useSync } from "./sync"
3
+ import { Global } from "@/global"
4
+
5
+ export function useDirectory() {
6
+ const sync = useSync()
7
+ return createMemo(() => {
8
+ const result = process.cwd().replace(Global.Path.home, "~")
9
+ if (sync.data.vcs?.branch) return result + ":" + sync.data.vcs.branch
10
+ return result
11
+ })
12
+ }
@@ -0,0 +1,23 @@
1
+ import { useRenderer } from "@opentui/solid"
2
+ import { createSimpleContext } from "./helper"
3
+ import { FormatError, FormatUnknownError } from "@/cli/error"
4
+
5
+ export const { use: useExit, provider: ExitProvider } = createSimpleContext({
6
+ name: "Exit",
7
+ init: (input: { onExit?: () => Promise<void> }) => {
8
+ const renderer = useRenderer()
9
+ return async (reason?: any) => {
10
+ // Reset window title before destroying renderer
11
+ renderer.setTerminalTitle("")
12
+ renderer.destroy()
13
+ await input.onExit?.()
14
+ if (reason) {
15
+ const formatted = FormatError(reason) ?? FormatUnknownError(reason)
16
+ if (formatted) {
17
+ process.stderr.write(formatted + "\n")
18
+ }
19
+ }
20
+ process.exit(0)
21
+ }
22
+ },
23
+ })
@@ -0,0 +1,25 @@
1
+ import { createContext, Show, useContext, type ParentProps } from "solid-js"
2
+
3
+ export function createSimpleContext<T, Props extends Record<string, any>>(input: {
4
+ name: string
5
+ init: ((input: Props) => T) | (() => T)
6
+ }) {
7
+ const ctx = createContext<T>()
8
+
9
+ return {
10
+ provider: (props: ParentProps<Props>) => {
11
+ const init = input.init(props)
12
+ return (
13
+ // @ts-expect-error
14
+ <Show when={init.ready === undefined || init.ready === true}>
15
+ <ctx.Provider value={init}>{props.children}</ctx.Provider>
16
+ </Show>
17
+ )
18
+ },
19
+ use() {
20
+ const value = useContext(ctx)
21
+ if (!value) throw new Error(`${input.name} context must be used within a context provider`)
22
+ return value
23
+ },
24
+ }
25
+ }
@@ -0,0 +1,111 @@
1
+ import { createMemo } from "solid-js"
2
+ import { useSync } from "@tui/context/sync"
3
+ import { Keybind } from "@/util/keybind"
4
+ import { pipe, mapValues } from "remeda"
5
+ import type { KeybindsConfig } from "@opencode-ai/sdk/v2"
6
+ import type { ParsedKey, Renderable } from "@opentui/core"
7
+ import { createStore } from "solid-js/store"
8
+ import { useKeyboard, useRenderer } from "@opentui/solid"
9
+ import { createSimpleContext } from "./helper"
10
+
11
+ export const { use: useKeybind, provider: KeybindProvider } = createSimpleContext({
12
+ name: "Keybind",
13
+ init: () => {
14
+ const sync = useSync()
15
+ const keybinds = createMemo(() => {
16
+ return pipe(
17
+ sync.data.config.keybinds ?? {},
18
+ mapValues((value) => Keybind.parse(value)),
19
+ )
20
+ })
21
+ const [store, setStore] = createStore({
22
+ leader: false,
23
+ })
24
+ const renderer = useRenderer()
25
+
26
+ let focus: Renderable | null
27
+ let timeout: NodeJS.Timeout
28
+ function leader(active: boolean) {
29
+ if (active) {
30
+ setStore("leader", true)
31
+ focus = renderer.currentFocusedRenderable
32
+ focus?.blur()
33
+ if (timeout) clearTimeout(timeout)
34
+ timeout = setTimeout(() => {
35
+ if (!store.leader) return
36
+ leader(false)
37
+ if (focus) {
38
+ focus.focus()
39
+ }
40
+ }, 2000)
41
+ return
42
+ }
43
+
44
+ if (!active) {
45
+ if (focus && !renderer.currentFocusedRenderable) {
46
+ focus.focus()
47
+ }
48
+ setStore("leader", false)
49
+ }
50
+ }
51
+
52
+ useKeyboard(async (evt) => {
53
+ if (!store.leader && result.match("leader", evt)) {
54
+ leader(true)
55
+ return
56
+ }
57
+
58
+ if (store.leader && evt.name) {
59
+ setImmediate(() => {
60
+ if (focus && renderer.currentFocusedRenderable === focus) {
61
+ focus.focus()
62
+ }
63
+ leader(false)
64
+ })
65
+ }
66
+ })
67
+
68
+ const result = {
69
+ get all() {
70
+ return keybinds()
71
+ },
72
+ get leader() {
73
+ return store.leader
74
+ },
75
+ parse(evt: ParsedKey): Keybind.Info {
76
+ if (evt.name === "\x1F")
77
+ return {
78
+ ctrl: true,
79
+ name: "_",
80
+ shift: false,
81
+ leader: false,
82
+ meta: false,
83
+ }
84
+ return {
85
+ ctrl: evt.ctrl,
86
+ name: evt.name,
87
+ shift: evt.shift,
88
+ leader: store.leader,
89
+ meta: evt.meta,
90
+ }
91
+ },
92
+ match(key: keyof KeybindsConfig, evt: ParsedKey) {
93
+ const keybind = keybinds()[key]
94
+ if (!keybind) return false
95
+ const parsed: Keybind.Info = result.parse(evt)
96
+ for (const key of keybind) {
97
+ if (Keybind.match(key, parsed)) {
98
+ return true
99
+ }
100
+ }
101
+ },
102
+ print(key: keyof KeybindsConfig) {
103
+ const first = keybinds()[key]?.at(0)
104
+ if (!first) return ""
105
+ const result = Keybind.toString(first)
106
+ return result.replace("<leader>", Keybind.toString(keybinds().leader![0]!))
107
+ },
108
+ }
109
+ return result
110
+ },
111
+ })
@@ -0,0 +1,49 @@
1
+ import { Global } from "@/global"
2
+ import { createSignal, type Setter } from "solid-js"
3
+ import { createStore } from "solid-js/store"
4
+ import { createSimpleContext } from "./helper"
5
+ import path from "path"
6
+
7
+ export const { use: useKV, provider: KVProvider } = createSimpleContext({
8
+ name: "KV",
9
+ init: () => {
10
+ const [ready, setReady] = createSignal(false)
11
+ const [kvStore, setKvStore] = createStore<Record<string, any>>()
12
+ const file = Bun.file(path.join(Global.Path.state, "kv.json"))
13
+
14
+ file
15
+ .json()
16
+ .then((x) => {
17
+ setKvStore(x)
18
+ })
19
+ .catch(() => {})
20
+ .finally(() => {
21
+ setReady(true)
22
+ })
23
+
24
+ const result = {
25
+ get ready() {
26
+ return ready()
27
+ },
28
+ signal<T>(name: string, defaultValue: T) {
29
+ if (!kvStore[name]) setKvStore(name, defaultValue)
30
+ return [
31
+ function () {
32
+ return result.get(name)
33
+ },
34
+ function setter(next: Setter<T>) {
35
+ result.set(name, next)
36
+ },
37
+ ] as const
38
+ },
39
+ get(key: string, defaultValue?: any) {
40
+ return kvStore[key] ?? defaultValue
41
+ },
42
+ set(key: string, value: any) {
43
+ setKvStore(key, value)
44
+ Bun.write(file, JSON.stringify(kvStore, null, 2))
45
+ },
46
+ }
47
+ return result
48
+ },
49
+ })
@@ -0,0 +1,338 @@
1
+ import { createStore } from "solid-js/store"
2
+ import { batch, createEffect, createMemo } from "solid-js"
3
+ import { useSync } from "@tui/context/sync"
4
+ import { useTheme } from "@tui/context/theme"
5
+ import { uniqueBy } from "remeda"
6
+ import path from "path"
7
+ import { Global } from "@/global"
8
+ import { iife } from "@/util/iife"
9
+ import { createSimpleContext } from "./helper"
10
+ import { useToast } from "../ui/toast"
11
+ import { Provider } from "@/provider/provider"
12
+ import { useArgs } from "./args"
13
+ import { useSDK } from "./sdk"
14
+ import { RGBA } from "@opentui/core"
15
+
16
+ export const { use: useLocal, provider: LocalProvider } = createSimpleContext({
17
+ name: "Local",
18
+ init: () => {
19
+ const sync = useSync()
20
+ const sdk = useSDK()
21
+ const toast = useToast()
22
+
23
+ function isModelValid(model: { providerID: string; modelID: string }) {
24
+ const provider = sync.data.provider.find((x) => x.id === model.providerID)
25
+ return !!provider?.models[model.modelID]
26
+ }
27
+
28
+ function getFirstValidModel(...modelFns: (() => { providerID: string; modelID: string } | undefined)[]) {
29
+ for (const modelFn of modelFns) {
30
+ const model = modelFn()
31
+ if (!model) continue
32
+ if (isModelValid(model)) return model
33
+ }
34
+ }
35
+
36
+ // Automatically update model when agent changes
37
+ createEffect(() => {
38
+ const value = agent.current()
39
+ if (value.model) {
40
+ if (isModelValid(value.model))
41
+ model.set({
42
+ providerID: value.model.providerID,
43
+ modelID: value.model.modelID,
44
+ })
45
+ else
46
+ toast.show({
47
+ variant: "warning",
48
+ message: `Agent ${value.name}'s configured model ${value.model.providerID}/${value.model.modelID} is not valid`,
49
+ duration: 3000,
50
+ })
51
+ }
52
+ })
53
+
54
+ const agent = iife(() => {
55
+ const agents = createMemo(() => sync.data.agent.filter((x) => x.mode !== "subagent"))
56
+ const [agentStore, setAgentStore] = createStore<{
57
+ current: string
58
+ }>({
59
+ current: agents()[0].name,
60
+ })
61
+ const { theme } = useTheme()
62
+ const colors = createMemo(() => [
63
+ theme.primary, // Cerebras orange
64
+ theme.success, // Green
65
+ theme.info, // Cyan
66
+ theme.error, // Red
67
+ theme.syntaxType, // Yellow
68
+ theme.syntaxKeyword, // Orange variant
69
+ ])
70
+ return {
71
+ list() {
72
+ return agents()
73
+ },
74
+ current() {
75
+ return agents().find((x) => x.name === agentStore.current)!
76
+ },
77
+ set(name: string) {
78
+ if (!agents().some((x) => x.name === name))
79
+ return toast.show({
80
+ variant: "warning",
81
+ message: `Agent not found: ${name}`,
82
+ duration: 3000,
83
+ })
84
+ setAgentStore("current", name)
85
+ },
86
+ move(direction: 1 | -1) {
87
+ batch(() => {
88
+ let next = agents().findIndex((x) => x.name === agentStore.current) + direction
89
+ if (next < 0) next = agents().length - 1
90
+ if (next >= agents().length) next = 0
91
+ const value = agents()[next]
92
+ setAgentStore("current", value.name)
93
+ })
94
+ },
95
+ color(name: string) {
96
+ const agent = agents().find((x) => x.name === name)
97
+ if (agent?.color) return RGBA.fromHex(agent.color)
98
+ const index = agents().findIndex((x) => x.name === name)
99
+ if (index === -1) return colors()[0]
100
+ return colors()[index % colors().length]
101
+ },
102
+ }
103
+ })
104
+
105
+ const model = iife(() => {
106
+ const [modelStore, setModelStore] = createStore<{
107
+ ready: boolean
108
+ model: Record<
109
+ string,
110
+ {
111
+ providerID: string
112
+ modelID: string
113
+ }
114
+ >
115
+ recent: {
116
+ providerID: string
117
+ modelID: string
118
+ }[]
119
+ favorite: {
120
+ providerID: string
121
+ modelID: string
122
+ }[]
123
+ }>({
124
+ ready: false,
125
+ model: {},
126
+ recent: [],
127
+ favorite: [],
128
+ })
129
+
130
+ const file = Bun.file(path.join(Global.Path.state, "model.json"))
131
+
132
+ function save() {
133
+ Bun.write(
134
+ file,
135
+ JSON.stringify({
136
+ recent: modelStore.recent,
137
+ favorite: modelStore.favorite,
138
+ }),
139
+ )
140
+ }
141
+
142
+ file
143
+ .json()
144
+ .then((x) => {
145
+ if (Array.isArray(x.recent)) setModelStore("recent", x.recent)
146
+ if (Array.isArray(x.favorite)) setModelStore("favorite", x.favorite)
147
+ })
148
+ .catch(() => {})
149
+ .finally(() => {
150
+ setModelStore("ready", true)
151
+ })
152
+
153
+ const args = useArgs()
154
+ const fallbackModel = createMemo(() => {
155
+ if (args.model) {
156
+ const { providerID, modelID } = Provider.parseModel(args.model)
157
+ if (isModelValid({ providerID, modelID })) {
158
+ return {
159
+ providerID,
160
+ modelID,
161
+ }
162
+ }
163
+ }
164
+
165
+ if (sync.data.config.model) {
166
+ const { providerID, modelID } = Provider.parseModel(sync.data.config.model)
167
+ if (isModelValid({ providerID, modelID })) {
168
+ return {
169
+ providerID,
170
+ modelID,
171
+ }
172
+ }
173
+ }
174
+
175
+ for (const item of modelStore.recent) {
176
+ if (isModelValid(item)) {
177
+ return item
178
+ }
179
+ }
180
+
181
+ // Default to Cerebras ZAI-GLM-4.7 for Cerebras Code CLI
182
+ const cerebrasDefault = { providerID: "cerebras", modelID: "zai-glm-4.7" }
183
+ if (isModelValid(cerebrasDefault)) {
184
+ return cerebrasDefault
185
+ }
186
+
187
+ // For Cerebras Code CLI, don't fall back to other providers - show nothing until Cerebras is connected
188
+ // This prevents showing "big pickle" or "gemini 3 pro" on first load
189
+ return undefined
190
+ })
191
+
192
+ const currentModel = createMemo(() => {
193
+ const a = agent.current()
194
+ return (
195
+ getFirstValidModel(
196
+ () => modelStore.model[a.name],
197
+ () => a.model,
198
+ fallbackModel,
199
+ ) ?? undefined
200
+ )
201
+ })
202
+
203
+ return {
204
+ current: currentModel,
205
+ get ready() {
206
+ return modelStore.ready
207
+ },
208
+ recent() {
209
+ return modelStore.recent
210
+ },
211
+ favorite() {
212
+ return modelStore.favorite
213
+ },
214
+ parsed: createMemo(() => {
215
+ const value = currentModel()
216
+ if (!value) {
217
+ return {
218
+ provider: "Cerebras",
219
+ model: "Loading...",
220
+ }
221
+ }
222
+ const provider = sync.data.provider.find((x) => x.id === value.providerID)
223
+ const info = provider?.models[value.modelID]
224
+ return {
225
+ provider: provider?.name ?? value.providerID,
226
+ model: info?.name ?? value.modelID,
227
+ }
228
+ }),
229
+ cycle(direction: 1 | -1) {
230
+ const current = currentModel()
231
+ if (!current) return
232
+ const recent = modelStore.recent
233
+ const index = recent.findIndex((x) => x.providerID === current.providerID && x.modelID === current.modelID)
234
+ if (index === -1) return
235
+ let next = index + direction
236
+ if (next < 0) next = recent.length - 1
237
+ if (next >= recent.length) next = 0
238
+ const val = recent[next]
239
+ if (!val) return
240
+ setModelStore("model", agent.current().name, { ...val })
241
+ },
242
+ cycleFavorite(direction: 1 | -1) {
243
+ const favorites = modelStore.favorite.filter((item) => isModelValid(item))
244
+ if (!favorites.length) {
245
+ toast.show({
246
+ variant: "info",
247
+ message: "Add a favorite model to use this shortcut",
248
+ duration: 3000,
249
+ })
250
+ return
251
+ }
252
+ const current = currentModel()
253
+ let index = -1
254
+ if (current) {
255
+ index = favorites.findIndex((x) => x.providerID === current.providerID && x.modelID === current.modelID)
256
+ }
257
+ if (index === -1) {
258
+ index = direction === 1 ? 0 : favorites.length - 1
259
+ } else {
260
+ index += direction
261
+ if (index < 0) index = favorites.length - 1
262
+ if (index >= favorites.length) index = 0
263
+ }
264
+ const next = favorites[index]
265
+ if (!next) return
266
+ setModelStore("model", agent.current().name, { ...next })
267
+ const uniq = uniqueBy([next, ...modelStore.recent], (x) => x.providerID + x.modelID)
268
+ if (uniq.length > 10) uniq.pop()
269
+ setModelStore("recent", uniq)
270
+ save()
271
+ },
272
+ set(model: { providerID: string; modelID: string }, options?: { recent?: boolean }) {
273
+ batch(() => {
274
+ if (!isModelValid(model)) {
275
+ toast.show({
276
+ message: `Model ${model.providerID}/${model.modelID} is not valid`,
277
+ variant: "warning",
278
+ duration: 3000,
279
+ })
280
+ return
281
+ }
282
+ setModelStore("model", agent.current().name, model)
283
+ if (options?.recent) {
284
+ const uniq = uniqueBy([model, ...modelStore.recent], (x) => x.providerID + x.modelID)
285
+ if (uniq.length > 10) uniq.pop()
286
+ setModelStore("recent", uniq)
287
+ save()
288
+ }
289
+ })
290
+ },
291
+ toggleFavorite(model: { providerID: string; modelID: string }) {
292
+ batch(() => {
293
+ if (!isModelValid(model)) {
294
+ toast.show({
295
+ message: `Model ${model.providerID}/${model.modelID} is not valid`,
296
+ variant: "warning",
297
+ duration: 3000,
298
+ })
299
+ return
300
+ }
301
+ const exists = modelStore.favorite.some(
302
+ (x) => x.providerID === model.providerID && x.modelID === model.modelID,
303
+ )
304
+ const next = exists
305
+ ? modelStore.favorite.filter((x) => x.providerID !== model.providerID || x.modelID !== model.modelID)
306
+ : [model, ...modelStore.favorite]
307
+ setModelStore("favorite", next)
308
+ save()
309
+ })
310
+ },
311
+ }
312
+ })
313
+
314
+ const mcp = {
315
+ isEnabled(name: string) {
316
+ const status = sync.data.mcp[name]
317
+ return status?.status === "connected"
318
+ },
319
+ async toggle(name: string) {
320
+ const status = sync.data.mcp[name]
321
+ if (status?.status === "connected") {
322
+ // Disable: disconnect the MCP
323
+ await sdk.client.mcp.disconnect({ name })
324
+ } else {
325
+ // Enable/Retry: connect the MCP (handles disabled, failed, and other states)
326
+ await sdk.client.mcp.connect({ name })
327
+ }
328
+ },
329
+ }
330
+
331
+ const result = {
332
+ model,
333
+ agent,
334
+ mcp,
335
+ }
336
+ return result
337
+ },
338
+ })