cerebras-cli 1.0.0

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/AGENTS.md +27 -0
  2. package/Dockerfile +10 -0
  3. package/README.md +15 -0
  4. package/bin/opencode +84 -0
  5. package/bunfig.toml +4 -0
  6. package/package.json +128 -0
  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/src/acp/README.md +164 -0
  13. package/src/acp/agent.ts +812 -0
  14. package/src/acp/session.ts +70 -0
  15. package/src/acp/types.ts +22 -0
  16. package/src/agent/agent.ts +310 -0
  17. package/src/agent/generate.txt +75 -0
  18. package/src/auth/index.ts +70 -0
  19. package/src/bun/index.ts +152 -0
  20. package/src/bus/global.ts +10 -0
  21. package/src/bus/index.ts +142 -0
  22. package/src/cli/bootstrap.ts +17 -0
  23. package/src/cli/cmd/acp.ts +88 -0
  24. package/src/cli/cmd/agent.ts +165 -0
  25. package/src/cli/cmd/auth.ts +369 -0
  26. package/src/cli/cmd/cmd.ts +7 -0
  27. package/src/cli/cmd/debug/config.ts +15 -0
  28. package/src/cli/cmd/debug/file.ts +91 -0
  29. package/src/cli/cmd/debug/index.ts +41 -0
  30. package/src/cli/cmd/debug/lsp.ts +47 -0
  31. package/src/cli/cmd/debug/ripgrep.ts +83 -0
  32. package/src/cli/cmd/debug/scrap.ts +15 -0
  33. package/src/cli/cmd/debug/snapshot.ts +48 -0
  34. package/src/cli/cmd/export.ts +88 -0
  35. package/src/cli/cmd/generate.ts +38 -0
  36. package/src/cli/cmd/github.ts +1200 -0
  37. package/src/cli/cmd/import.ts +98 -0
  38. package/src/cli/cmd/mcp.ts +400 -0
  39. package/src/cli/cmd/models.ts +77 -0
  40. package/src/cli/cmd/pr.ts +112 -0
  41. package/src/cli/cmd/run.ts +342 -0
  42. package/src/cli/cmd/serve.ts +31 -0
  43. package/src/cli/cmd/session.ts +106 -0
  44. package/src/cli/cmd/stats.ts +298 -0
  45. package/src/cli/cmd/tui/app.tsx +732 -0
  46. package/src/cli/cmd/tui/attach.ts +25 -0
  47. package/src/cli/cmd/tui/component/border.tsx +21 -0
  48. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  49. package/src/cli/cmd/tui/component/dialog-command.tsx +124 -0
  50. package/src/cli/cmd/tui/component/dialog-feedback.tsx +160 -0
  51. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  52. package/src/cli/cmd/tui/component/dialog-model.tsx +223 -0
  53. package/src/cli/cmd/tui/component/dialog-notification.tsx +78 -0
  54. package/src/cli/cmd/tui/component/dialog-provider.tsx +222 -0
  55. package/src/cli/cmd/tui/component/dialog-session-list.tsx +97 -0
  56. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  57. package/src/cli/cmd/tui/component/dialog-status.tsx +114 -0
  58. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  59. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  60. package/src/cli/cmd/tui/component/logo.tsx +37 -0
  61. package/src/cli/cmd/tui/component/notification-banner.tsx +58 -0
  62. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +530 -0
  63. package/src/cli/cmd/tui/component/prompt/history.tsx +107 -0
  64. package/src/cli/cmd/tui/component/prompt/index.tsx +931 -0
  65. package/src/cli/cmd/tui/context/args.tsx +14 -0
  66. package/src/cli/cmd/tui/context/directory.ts +12 -0
  67. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  68. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  69. package/src/cli/cmd/tui/context/keybind.tsx +111 -0
  70. package/src/cli/cmd/tui/context/kv.tsx +49 -0
  71. package/src/cli/cmd/tui/context/local.tsx +339 -0
  72. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  73. package/src/cli/cmd/tui/context/route.tsx +45 -0
  74. package/src/cli/cmd/tui/context/sdk.tsx +75 -0
  75. package/src/cli/cmd/tui/context/sync.tsx +374 -0
  76. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  77. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  78. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  79. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  80. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  81. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  82. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  83. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  84. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  85. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  86. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  87. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  88. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  89. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  90. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  91. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  92. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  93. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  94. package/src/cli/cmd/tui/context/theme/orng.json +245 -0
  95. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  96. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  97. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  98. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  99. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  100. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  101. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  102. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  103. package/src/cli/cmd/tui/context/theme.tsx +1077 -0
  104. package/src/cli/cmd/tui/event.ts +39 -0
  105. package/src/cli/cmd/tui/routes/home.tsx +104 -0
  106. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +93 -0
  107. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +37 -0
  108. package/src/cli/cmd/tui/routes/session/footer.tsx +76 -0
  109. package/src/cli/cmd/tui/routes/session/header.tsx +183 -0
  110. package/src/cli/cmd/tui/routes/session/index.tsx +1703 -0
  111. package/src/cli/cmd/tui/routes/session/sidebar.tsx +586 -0
  112. package/src/cli/cmd/tui/spawn.ts +60 -0
  113. package/src/cli/cmd/tui/thread.ts +120 -0
  114. package/src/cli/cmd/tui/ui/dialog-alert.tsx +55 -0
  115. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +81 -0
  116. package/src/cli/cmd/tui/ui/dialog-help.tsx +36 -0
  117. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +75 -0
  118. package/src/cli/cmd/tui/ui/dialog-select.tsx +317 -0
  119. package/src/cli/cmd/tui/ui/dialog.tsx +170 -0
  120. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  121. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  122. package/src/cli/cmd/tui/util/clipboard.ts +127 -0
  123. package/src/cli/cmd/tui/util/editor.ts +32 -0
  124. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  125. package/src/cli/cmd/tui/worker.ts +63 -0
  126. package/src/cli/cmd/uninstall.ts +344 -0
  127. package/src/cli/cmd/upgrade.ts +67 -0
  128. package/src/cli/cmd/web.ts +84 -0
  129. package/src/cli/error.ts +55 -0
  130. package/src/cli/ui.ts +84 -0
  131. package/src/cli/upgrade.ts +25 -0
  132. package/src/command/index.ts +79 -0
  133. package/src/command/template/initialize.txt +10 -0
  134. package/src/command/template/review.txt +73 -0
  135. package/src/config/config.ts +886 -0
  136. package/src/config/markdown.ts +41 -0
  137. package/src/env/index.ts +26 -0
  138. package/src/file/fzf.ts +124 -0
  139. package/src/file/ignore.ts +83 -0
  140. package/src/file/index.ts +326 -0
  141. package/src/file/ripgrep.ts +391 -0
  142. package/src/file/time.ts +38 -0
  143. package/src/file/watcher.ts +89 -0
  144. package/src/flag/flag.ts +28 -0
  145. package/src/format/formatter.ts +277 -0
  146. package/src/format/index.ts +137 -0
  147. package/src/global/index.ts +52 -0
  148. package/src/id/id.ts +73 -0
  149. package/src/ide/index.ts +75 -0
  150. package/src/index.ts +158 -0
  151. package/src/installation/index.ts +194 -0
  152. package/src/lsp/client.ts +215 -0
  153. package/src/lsp/index.ts +370 -0
  154. package/src/lsp/language.ts +111 -0
  155. package/src/lsp/server.ts +1327 -0
  156. package/src/mcp/auth.ts +82 -0
  157. package/src/mcp/index.ts +576 -0
  158. package/src/mcp/oauth-callback.ts +203 -0
  159. package/src/mcp/oauth-provider.ts +132 -0
  160. package/src/notification/index.ts +101 -0
  161. package/src/patch/index.ts +622 -0
  162. package/src/permission/index.ts +198 -0
  163. package/src/plugin/index.ts +95 -0
  164. package/src/project/bootstrap.ts +31 -0
  165. package/src/project/instance.ts +68 -0
  166. package/src/project/project.ts +133 -0
  167. package/src/project/state.ts +65 -0
  168. package/src/project/vcs.ts +77 -0
  169. package/src/provider/auth.ts +143 -0
  170. package/src/provider/models-macro.ts +11 -0
  171. package/src/provider/models.ts +93 -0
  172. package/src/provider/provider.ts +996 -0
  173. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  174. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  175. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  176. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  177. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +27 -0
  178. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  179. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  180. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  181. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1713 -0
  182. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  183. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  184. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  185. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  186. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  187. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  188. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  189. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  190. package/src/provider/transform.ts +406 -0
  191. package/src/pty/index.ts +226 -0
  192. package/src/ratelimit/index.ts +185 -0
  193. package/src/server/error.ts +36 -0
  194. package/src/server/project.ts +50 -0
  195. package/src/server/server.ts +2463 -0
  196. package/src/server/tui.ts +71 -0
  197. package/src/session/compaction.ts +257 -0
  198. package/src/session/index.ts +470 -0
  199. package/src/session/message-v2.ts +641 -0
  200. package/src/session/message.ts +189 -0
  201. package/src/session/processor.ts +443 -0
  202. package/src/session/prompt/anthropic-20250930.txt +166 -0
  203. package/src/session/prompt/anthropic.txt +105 -0
  204. package/src/session/prompt/anthropic_spoof.txt +1 -0
  205. package/src/session/prompt/beast.txt +147 -0
  206. package/src/session/prompt/build-switch.txt +5 -0
  207. package/src/session/prompt/codex.txt +318 -0
  208. package/src/session/prompt/compaction.txt +12 -0
  209. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  210. package/src/session/prompt/gemini.txt +155 -0
  211. package/src/session/prompt/max-steps.txt +16 -0
  212. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  213. package/src/session/prompt/plan.txt +26 -0
  214. package/src/session/prompt/polaris.txt +107 -0
  215. package/src/session/prompt/qwen.txt +109 -0
  216. package/src/session/prompt/summarize.txt +4 -0
  217. package/src/session/prompt/title.txt +36 -0
  218. package/src/session/prompt.ts +1541 -0
  219. package/src/session/retry.ts +82 -0
  220. package/src/session/revert.ts +108 -0
  221. package/src/session/status.ts +75 -0
  222. package/src/session/summary.ts +203 -0
  223. package/src/session/system.ts +148 -0
  224. package/src/session/todo.ts +36 -0
  225. package/src/share/share-next.ts +195 -0
  226. package/src/share/share.ts +87 -0
  227. package/src/snapshot/index.ts +197 -0
  228. package/src/storage/storage.ts +226 -0
  229. package/src/telemetry/index.ts +232 -0
  230. package/src/tool/bash.ts +365 -0
  231. package/src/tool/bash.txt +128 -0
  232. package/src/tool/batch.ts +173 -0
  233. package/src/tool/batch.txt +28 -0
  234. package/src/tool/codesearch.ts +138 -0
  235. package/src/tool/codesearch.txt +12 -0
  236. package/src/tool/edit.ts +674 -0
  237. package/src/tool/edit.txt +10 -0
  238. package/src/tool/glob.ts +65 -0
  239. package/src/tool/glob.txt +6 -0
  240. package/src/tool/grep.ts +120 -0
  241. package/src/tool/grep.txt +8 -0
  242. package/src/tool/invalid.ts +17 -0
  243. package/src/tool/ls.ts +110 -0
  244. package/src/tool/ls.txt +1 -0
  245. package/src/tool/lsp-diagnostics.ts +26 -0
  246. package/src/tool/lsp-diagnostics.txt +1 -0
  247. package/src/tool/lsp-hover.ts +31 -0
  248. package/src/tool/lsp-hover.txt +1 -0
  249. package/src/tool/multiedit.ts +46 -0
  250. package/src/tool/multiedit.txt +41 -0
  251. package/src/tool/patch.ts +233 -0
  252. package/src/tool/patch.txt +1 -0
  253. package/src/tool/read.ts +217 -0
  254. package/src/tool/read.txt +12 -0
  255. package/src/tool/registry.ts +148 -0
  256. package/src/tool/task.ts +135 -0
  257. package/src/tool/task.txt +60 -0
  258. package/src/tool/todo.ts +39 -0
  259. package/src/tool/todoread.txt +14 -0
  260. package/src/tool/todowrite.txt +167 -0
  261. package/src/tool/tool.ts +66 -0
  262. package/src/tool/webfetch.ts +187 -0
  263. package/src/tool/webfetch.txt +14 -0
  264. package/src/tool/websearch.ts +150 -0
  265. package/src/tool/websearch.txt +11 -0
  266. package/src/tool/write.ts +99 -0
  267. package/src/tool/write.txt +8 -0
  268. package/src/types/shims.d.ts +3 -0
  269. package/src/util/color.ts +19 -0
  270. package/src/util/context.ts +25 -0
  271. package/src/util/defer.ts +12 -0
  272. package/src/util/eventloop.ts +20 -0
  273. package/src/util/filesystem.ts +69 -0
  274. package/src/util/fn.ts +11 -0
  275. package/src/util/iife.ts +3 -0
  276. package/src/util/keybind.ts +79 -0
  277. package/src/util/lazy.ts +11 -0
  278. package/src/util/locale.ts +81 -0
  279. package/src/util/lock.ts +98 -0
  280. package/src/util/log.ts +177 -0
  281. package/src/util/queue.ts +32 -0
  282. package/src/util/rpc.ts +42 -0
  283. package/src/util/scrap.ts +10 -0
  284. package/src/util/signal.ts +12 -0
  285. package/src/util/timeout.ts +14 -0
  286. package/src/util/token.ts +7 -0
  287. package/src/util/wildcard.ts +54 -0
  288. package/sst-env.d.ts +9 -0
  289. package/test/bun.test.ts +53 -0
  290. package/test/config/agent-color.test.ts +66 -0
  291. package/test/config/config.test.ts +503 -0
  292. package/test/config/markdown.test.ts +89 -0
  293. package/test/file/ignore.test.ts +10 -0
  294. package/test/fixture/fixture.ts +28 -0
  295. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  296. package/test/ide/ide.test.ts +82 -0
  297. package/test/keybind.test.ts +317 -0
  298. package/test/lsp/client.test.ts +95 -0
  299. package/test/patch/patch.test.ts +348 -0
  300. package/test/preload.ts +38 -0
  301. package/test/project/project.test.ts +42 -0
  302. package/test/provider/provider.test.ts +1809 -0
  303. package/test/provider/transform.test.ts +305 -0
  304. package/test/session/retry.test.ts +61 -0
  305. package/test/session/session.test.ts +71 -0
  306. package/test/snapshot/snapshot.test.ts +939 -0
  307. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  308. package/test/tool/bash.test.ts +55 -0
  309. package/test/tool/patch.test.ts +259 -0
  310. package/test/util/iife.test.ts +36 -0
  311. package/test/util/lazy.test.ts +50 -0
  312. package/test/util/timeout.test.ts +21 -0
  313. package/test/util/wildcard.test.ts +55 -0
  314. package/tsconfig.json +17 -0
@@ -0,0 +1,58 @@
1
+ import { TextAttributes } from "@opentui/core"
2
+ import { useTheme } from "../context/theme"
3
+ import { createSignal, Show } from "solid-js"
4
+ import { useKeyboard } from "@opentui/solid"
5
+ import type { Notification } from "@/notification"
6
+
7
+ export function NotificationBanner(props: { notification: Notification; onDismiss: () => void }) {
8
+ const { theme } = useTheme()
9
+ const [visible, setVisible] = createSignal(true)
10
+
11
+ const bgColor = () => {
12
+ switch (props.notification.type) {
13
+ case "critical":
14
+ return theme.error
15
+ case "warning":
16
+ return theme.warning
17
+ default:
18
+ return theme.primary
19
+ }
20
+ }
21
+
22
+ const dismiss = () => {
23
+ setVisible(false)
24
+ props.onDismiss()
25
+ }
26
+
27
+ useKeyboard((evt) => {
28
+ if (evt.name === "escape" && visible()) {
29
+ dismiss()
30
+ }
31
+ })
32
+
33
+ return (
34
+ <Show when={visible()}>
35
+ <box
36
+ width="100%"
37
+ backgroundColor={bgColor()}
38
+ paddingLeft={2}
39
+ paddingRight={2}
40
+ paddingTop={1}
41
+ paddingBottom={1}
42
+ flexDirection="column"
43
+ gap={0}
44
+ >
45
+ <box flexDirection="row" justifyContent="space-between" alignItems="center">
46
+ <text attributes={TextAttributes.BOLD} fg={theme.backgroundPanel}>
47
+ {props.notification.title}
48
+ </text>
49
+ <text fg={theme.backgroundPanel} onMouseUp={dismiss}>
50
+ ✕ esc to dismiss
51
+ </text>
52
+ </box>
53
+ <text fg={theme.backgroundPanel}>{props.notification.message}</text>
54
+ </box>
55
+ </Show>
56
+ )
57
+ }
58
+
@@ -0,0 +1,530 @@
1
+ import type { BoxRenderable, TextareaRenderable, KeyEvent } from "@opentui/core"
2
+ import fuzzysort from "fuzzysort"
3
+ import { firstBy } from "remeda"
4
+ import { createMemo, createResource, createEffect, onMount, onCleanup, For, Show, createSignal } from "solid-js"
5
+ import { createStore } from "solid-js/store"
6
+ import { useSDK } from "@tui/context/sdk"
7
+ import { useSync } from "@tui/context/sync"
8
+ import { useTheme, selectedForeground } from "@tui/context/theme"
9
+ import { SplitBorder } from "@tui/component/border"
10
+ import { useCommandDialog } from "@tui/component/dialog-command"
11
+ import { useTerminalDimensions } from "@opentui/solid"
12
+ import { Locale } from "@/util/locale"
13
+ import type { PromptInfo } from "./history"
14
+
15
+ export type AutocompleteRef = {
16
+ onInput: (value: string) => void
17
+ onKeyDown: (e: KeyEvent) => void
18
+ visible: false | "@" | "/"
19
+ }
20
+
21
+ export type AutocompleteOption = {
22
+ display: string
23
+ aliases?: string[]
24
+ disabled?: boolean
25
+ description?: string
26
+ onSelect?: () => void
27
+ }
28
+
29
+ export function Autocomplete(props: {
30
+ value: string
31
+ sessionID?: string
32
+ setPrompt: (input: (prompt: PromptInfo) => void) => void
33
+ setExtmark: (partIndex: number, extmarkId: number) => void
34
+ anchor: () => BoxRenderable
35
+ input: () => TextareaRenderable
36
+ ref: (ref: AutocompleteRef) => void
37
+ fileStyleId: number
38
+ agentStyleId: number
39
+ promptPartTypeId: () => number
40
+ }) {
41
+ const sdk = useSDK()
42
+ const sync = useSync()
43
+ const command = useCommandDialog()
44
+ const { theme } = useTheme()
45
+ const dimensions = useTerminalDimensions()
46
+
47
+ const [store, setStore] = createStore({
48
+ index: 0,
49
+ selected: 0,
50
+ visible: false as AutocompleteRef["visible"],
51
+ })
52
+
53
+ const [positionTick, setPositionTick] = createSignal(0)
54
+
55
+ createEffect(() => {
56
+ if (store.visible) {
57
+ let lastPos = { x: 0, y: 0, width: 0 }
58
+ const interval = setInterval(() => {
59
+ const anchor = props.anchor()
60
+ if (anchor.x !== lastPos.x || anchor.y !== lastPos.y || anchor.width !== lastPos.width) {
61
+ lastPos = { x: anchor.x, y: anchor.y, width: anchor.width }
62
+ setPositionTick((t) => t + 1)
63
+ }
64
+ }, 50)
65
+
66
+ onCleanup(() => clearInterval(interval))
67
+ }
68
+ })
69
+
70
+ const position = createMemo(() => {
71
+ if (!store.visible) return { x: 0, y: 0, width: 0 }
72
+ const dims = dimensions()
73
+ positionTick()
74
+ const anchor = props.anchor()
75
+ return {
76
+ x: anchor.x,
77
+ y: anchor.y,
78
+ width: anchor.width,
79
+ }
80
+ })
81
+
82
+ const filter = createMemo(() => {
83
+ if (!store.visible) return
84
+ // Track props.value to make memo reactive to text changes
85
+ props.value // <- there surely is a better way to do this, like making .input() reactive
86
+
87
+ return props.input().getTextRange(store.index + 1, props.input().cursorOffset)
88
+ })
89
+
90
+ function insertPart(text: string, part: PromptInfo["parts"][number]) {
91
+ const input = props.input()
92
+ const currentCursorOffset = input.cursorOffset
93
+
94
+ const charAfterCursor = props.value.at(currentCursorOffset)
95
+ const needsSpace = charAfterCursor !== " "
96
+ const append = "@" + text + (needsSpace ? " " : "")
97
+
98
+ input.cursorOffset = store.index
99
+ const startCursor = input.logicalCursor
100
+ input.cursorOffset = currentCursorOffset
101
+ const endCursor = input.logicalCursor
102
+
103
+ input.deleteRange(startCursor.row, startCursor.col, endCursor.row, endCursor.col)
104
+ input.insertText(append)
105
+
106
+ const virtualText = "@" + text
107
+ const extmarkStart = store.index
108
+ const extmarkEnd = extmarkStart + Bun.stringWidth(virtualText)
109
+
110
+ const styleId = part.type === "file" ? props.fileStyleId : part.type === "agent" ? props.agentStyleId : undefined
111
+
112
+ const extmarkId = input.extmarks.create({
113
+ start: extmarkStart,
114
+ end: extmarkEnd,
115
+ virtual: true,
116
+ styleId,
117
+ typeId: props.promptPartTypeId(),
118
+ })
119
+
120
+ props.setPrompt((draft) => {
121
+ if (part.type === "file" && part.source?.text) {
122
+ part.source.text.start = extmarkStart
123
+ part.source.text.end = extmarkEnd
124
+ part.source.text.value = virtualText
125
+ } else if (part.type === "agent" && part.source) {
126
+ part.source.start = extmarkStart
127
+ part.source.end = extmarkEnd
128
+ part.source.value = virtualText
129
+ }
130
+ const partIndex = draft.parts.length
131
+ draft.parts.push(part)
132
+ props.setExtmark(partIndex, extmarkId)
133
+ })
134
+ }
135
+
136
+ const [files] = createResource(
137
+ () => filter(),
138
+ async (query) => {
139
+ if (!store.visible || store.visible === "/") return []
140
+
141
+ // Get files from SDK
142
+ const result = await sdk.client.find.files({
143
+ query: query ?? "",
144
+ })
145
+
146
+ const options: AutocompleteOption[] = []
147
+
148
+ // Add file options
149
+ if (!result.error && result.data) {
150
+ const width = props.anchor().width - 4
151
+ options.push(
152
+ ...result.data.map(
153
+ (item): AutocompleteOption => ({
154
+ display: Locale.truncateMiddle(item, width),
155
+ onSelect: () => {
156
+ insertPart(item, {
157
+ type: "file",
158
+ mime: "text/plain",
159
+ filename: item,
160
+ url: `file://${process.cwd()}/${item}`,
161
+ source: {
162
+ type: "file",
163
+ text: {
164
+ start: 0,
165
+ end: 0,
166
+ value: "",
167
+ },
168
+ path: item,
169
+ },
170
+ })
171
+ },
172
+ }),
173
+ ),
174
+ )
175
+ }
176
+
177
+ return options
178
+ },
179
+ {
180
+ initialValue: [],
181
+ },
182
+ )
183
+
184
+ const agents = createMemo(() => {
185
+ const agents = sync.data.agent
186
+ return agents
187
+ .filter((agent) => !agent.builtIn && agent.mode !== "primary")
188
+ .map(
189
+ (agent): AutocompleteOption => ({
190
+ display: "@" + agent.name,
191
+ onSelect: () => {
192
+ insertPart(agent.name, {
193
+ type: "agent",
194
+ name: agent.name,
195
+ source: {
196
+ start: 0,
197
+ end: 0,
198
+ value: "",
199
+ },
200
+ })
201
+ },
202
+ }),
203
+ )
204
+ })
205
+
206
+ const session = createMemo(() => (props.sessionID ? sync.session.get(props.sessionID) : undefined))
207
+ const commands = createMemo((): AutocompleteOption[] => {
208
+ const results: AutocompleteOption[] = []
209
+ const s = session()
210
+ for (const command of sync.data.command) {
211
+ results.push({
212
+ display: "/" + command.name,
213
+ description: command.description,
214
+ onSelect: () => {
215
+ const newText = "/" + command.name + " "
216
+ const cursor = props.input().logicalCursor
217
+ props.input().deleteRange(0, 0, cursor.row, cursor.col)
218
+ props.input().insertText(newText)
219
+ props.input().cursorOffset = Bun.stringWidth(newText)
220
+ },
221
+ })
222
+ }
223
+ if (s) {
224
+ results.push(
225
+ {
226
+ display: "/undo",
227
+ description: "undo the last message",
228
+ onSelect: () => {
229
+ command.trigger("session.undo")
230
+ },
231
+ },
232
+ {
233
+ display: "/redo",
234
+ description: "redo the last message",
235
+ onSelect: () => command.trigger("session.redo"),
236
+ },
237
+ {
238
+ display: "/compact",
239
+ aliases: ["/summarize"],
240
+ description: "compact the session",
241
+ onSelect: () => command.trigger("session.compact"),
242
+ },
243
+ {
244
+ display: "/unshare",
245
+ disabled: !s.share,
246
+ description: "unshare a session",
247
+ onSelect: () => command.trigger("session.unshare"),
248
+ },
249
+ {
250
+ display: "/rename",
251
+ description: "rename session",
252
+ onSelect: () => command.trigger("session.rename"),
253
+ },
254
+ {
255
+ display: "/copy",
256
+ description: "copy session transcript to clipboard",
257
+ onSelect: () => command.trigger("session.copy"),
258
+ },
259
+ {
260
+ display: "/export",
261
+ description: "export session transcript to file",
262
+ onSelect: () => command.trigger("session.export"),
263
+ },
264
+ {
265
+ display: "/timeline",
266
+ description: "jump to message",
267
+ onSelect: () => command.trigger("session.timeline"),
268
+ },
269
+ {
270
+ display: "/thinking",
271
+ description: "toggle thinking visibility",
272
+ onSelect: () => command.trigger("session.toggle.thinking"),
273
+ },
274
+ )
275
+ if (sync.data.config.share !== "disabled") {
276
+ results.push({
277
+ display: "/share",
278
+ disabled: !!s.share?.url,
279
+ description: "share a session",
280
+ onSelect: () => command.trigger("session.share"),
281
+ })
282
+ }
283
+ }
284
+
285
+ results.push(
286
+ {
287
+ display: "/new",
288
+ aliases: ["/clear"],
289
+ description: "create a new session",
290
+ onSelect: () => command.trigger("session.new"),
291
+ },
292
+ {
293
+ display: "/models",
294
+ description: "list models",
295
+ onSelect: () => command.trigger("model.list"),
296
+ },
297
+ {
298
+ display: "/agents",
299
+ description: "list agents",
300
+ onSelect: () => command.trigger("agent.list"),
301
+ },
302
+ {
303
+ display: "/session",
304
+ aliases: ["/resume", "/continue"],
305
+ description: "list sessions",
306
+ onSelect: () => command.trigger("session.list"),
307
+ },
308
+ {
309
+ display: "/status",
310
+ description: "show status",
311
+ onSelect: () => command.trigger("opencode.status"),
312
+ },
313
+ {
314
+ display: "/mcp",
315
+ description: "toggle MCPs",
316
+ onSelect: () => command.trigger("mcp.list"),
317
+ },
318
+ {
319
+ display: "/theme",
320
+ description: "toggle theme",
321
+ onSelect: () => command.trigger("theme.switch"),
322
+ },
323
+ {
324
+ display: "/editor",
325
+ description: "open editor",
326
+ onSelect: () => command.trigger("prompt.editor", "prompt"),
327
+ },
328
+ {
329
+ display: "/connect",
330
+ description: "connect to a provider",
331
+ onSelect: () => command.trigger("provider.connect"),
332
+ },
333
+ {
334
+ display: "/help",
335
+ description: "show help",
336
+ onSelect: () => command.trigger("help.show"),
337
+ },
338
+ {
339
+ display: "/feedback",
340
+ description: "send feedback or report a bug",
341
+ onSelect: () => command.trigger("feedback.send"),
342
+ },
343
+ {
344
+ display: "/commands",
345
+ description: "show all commands",
346
+ onSelect: () => command.show(),
347
+ },
348
+ {
349
+ display: "/exit",
350
+ aliases: ["/quit", "/q"],
351
+ description: "exit the app",
352
+ onSelect: () => command.trigger("app.exit"),
353
+ },
354
+ )
355
+ const max = firstBy(results, [(x) => x.display.length, "desc"])?.display.length
356
+ if (!max) return results
357
+ return results.map((item) => ({
358
+ ...item,
359
+ display: item.display.padEnd(max + 2),
360
+ }))
361
+ })
362
+
363
+ const options = createMemo(() => {
364
+ const mixed: AutocompleteOption[] = (
365
+ store.visible === "@" ? [...agents(), ...(files.loading ? files.latest || [] : files())] : [...commands()]
366
+ ).filter((x) => x.disabled !== true)
367
+ const currentFilter = filter()
368
+ if (!currentFilter) return mixed.slice(0, 10)
369
+ const result = fuzzysort.go(currentFilter, mixed, {
370
+ keys: [(obj) => obj.display.trimEnd(), "description", (obj) => obj.aliases?.join(" ") ?? ""],
371
+ limit: 10,
372
+ })
373
+ return result.map((arr) => arr.obj)
374
+ })
375
+
376
+ createEffect(() => {
377
+ filter()
378
+ setStore("selected", 0)
379
+ })
380
+
381
+ function move(direction: -1 | 1) {
382
+ if (!store.visible) return
383
+ if (!options().length) return
384
+ let next = store.selected + direction
385
+ if (next < 0) next = options().length - 1
386
+ if (next >= options().length) next = 0
387
+ setStore("selected", next)
388
+ }
389
+
390
+ function select() {
391
+ const selected = options()[store.selected]
392
+ if (!selected) return
393
+ hide()
394
+ selected.onSelect?.()
395
+ }
396
+
397
+ function show(mode: "@" | "/") {
398
+ command.keybinds(false)
399
+ setStore({
400
+ visible: mode,
401
+ index: props.input().cursorOffset,
402
+ })
403
+ }
404
+
405
+ function hide() {
406
+ const text = props.input().plainText
407
+ if (store.visible === "/" && !text.endsWith(" ") && text.startsWith("/")) {
408
+ const cursor = props.input().logicalCursor
409
+ props.input().deleteRange(0, 0, cursor.row, cursor.col)
410
+ // Sync the prompt store immediately since onContentChange is async
411
+ props.setPrompt((draft) => {
412
+ draft.input = props.input().plainText
413
+ })
414
+ }
415
+ command.keybinds(true)
416
+ setStore("visible", false)
417
+ }
418
+
419
+ onMount(() => {
420
+ props.ref({
421
+ get visible() {
422
+ return store.visible
423
+ },
424
+ onInput(value) {
425
+ if (store.visible) {
426
+ if (
427
+ // Typed text before the trigger
428
+ props.input().cursorOffset <= store.index ||
429
+ // There is a space between the trigger and the cursor
430
+ props.input().getTextRange(store.index, props.input().cursorOffset).match(/\s/) ||
431
+ // "/<command>" is not the sole content
432
+ (store.visible === "/" && value.match(/^\S+\s+\S+\s*$/))
433
+ ) {
434
+ hide()
435
+ return
436
+ }
437
+ }
438
+ },
439
+ onKeyDown(e: KeyEvent) {
440
+ if (store.visible) {
441
+ const name = e.name?.toLowerCase()
442
+ const ctrlOnly = e.ctrl && !e.meta && !e.shift
443
+ const isNavUp = name === "up" || (ctrlOnly && name === "p")
444
+ const isNavDown = name === "down" || (ctrlOnly && name === "n")
445
+
446
+ if (isNavUp) {
447
+ move(-1)
448
+ e.preventDefault()
449
+ return
450
+ }
451
+ if (isNavDown) {
452
+ move(1)
453
+ e.preventDefault()
454
+ return
455
+ }
456
+ if (name === "escape") {
457
+ hide()
458
+ e.preventDefault()
459
+ return
460
+ }
461
+ if (name === "return" || name === "tab") {
462
+ select()
463
+ e.preventDefault()
464
+ return
465
+ }
466
+ }
467
+ if (!store.visible) {
468
+ if (e.name === "@") {
469
+ const cursorOffset = props.input().cursorOffset
470
+ const charBeforeCursor =
471
+ cursorOffset === 0 ? undefined : props.input().getTextRange(cursorOffset - 1, cursorOffset)
472
+ const canTrigger = charBeforeCursor === undefined || charBeforeCursor === "" || /\s/.test(charBeforeCursor)
473
+ if (canTrigger) show("@")
474
+ }
475
+
476
+ if (e.name === "/") {
477
+ if (props.input().cursorOffset === 0) show("/")
478
+ }
479
+ }
480
+ },
481
+ })
482
+ })
483
+
484
+ const height = createMemo(() => {
485
+ if (options().length) return Math.min(10, options().length)
486
+ return 1
487
+ })
488
+
489
+ return (
490
+ <box
491
+ visible={store.visible !== false}
492
+ position="absolute"
493
+ top={position().y - height()}
494
+ left={position().x}
495
+ width={position().width}
496
+ zIndex={100}
497
+ {...SplitBorder}
498
+ borderColor={theme.border}
499
+ >
500
+ <box backgroundColor={theme.backgroundMenu} height={height()}>
501
+ <For
502
+ each={options()}
503
+ fallback={
504
+ <box paddingLeft={1} paddingRight={1}>
505
+ <text fg={theme.textMuted}>No matching items</text>
506
+ </box>
507
+ }
508
+ >
509
+ {(option, index) => (
510
+ <box
511
+ paddingLeft={1}
512
+ paddingRight={1}
513
+ backgroundColor={index() === store.selected ? theme.primary : undefined}
514
+ flexDirection="row"
515
+ >
516
+ <text fg={index() === store.selected ? selectedForeground(theme) : theme.text} flexShrink={0}>
517
+ {option.display}
518
+ </text>
519
+ <Show when={option.description}>
520
+ <text fg={index() === store.selected ? selectedForeground(theme) : theme.textMuted} wrapMode="none">
521
+ {option.description}
522
+ </text>
523
+ </Show>
524
+ </box>
525
+ )}
526
+ </For>
527
+ </box>
528
+ </box>
529
+ )
530
+ }
@@ -0,0 +1,107 @@
1
+ import path from "path"
2
+ import { Global } from "@/global"
3
+ import { onMount } from "solid-js"
4
+ import { createStore, produce } from "solid-js/store"
5
+ import { clone } from "remeda"
6
+ import { createSimpleContext } from "../../context/helper"
7
+ import { appendFile, writeFile } from "fs/promises"
8
+ import type { AgentPart, FilePart, TextPart } from "@opencode-ai/sdk/v2"
9
+
10
+ export type PromptInfo = {
11
+ input: string
12
+ parts: (
13
+ | Omit<FilePart, "id" | "messageID" | "sessionID">
14
+ | Omit<AgentPart, "id" | "messageID" | "sessionID">
15
+ | (Omit<TextPart, "id" | "messageID" | "sessionID"> & {
16
+ source?: {
17
+ text: {
18
+ start: number
19
+ end: number
20
+ value: string
21
+ }
22
+ }
23
+ })
24
+ )[]
25
+ }
26
+
27
+ const MAX_HISTORY_ENTRIES = 50
28
+
29
+ export const { use: usePromptHistory, provider: PromptHistoryProvider } = createSimpleContext({
30
+ name: "PromptHistory",
31
+ init: () => {
32
+ const historyFile = Bun.file(path.join(Global.Path.state, "prompt-history.jsonl"))
33
+ onMount(async () => {
34
+ const text = await historyFile.text().catch(() => "")
35
+ const lines = text
36
+ .split("\n")
37
+ .filter(Boolean)
38
+ .map((line) => {
39
+ try {
40
+ return JSON.parse(line)
41
+ } catch {
42
+ return null
43
+ }
44
+ })
45
+ .filter((line): line is PromptInfo => line !== null)
46
+ .slice(-MAX_HISTORY_ENTRIES)
47
+
48
+ setStore("history", lines)
49
+
50
+ // Rewrite file with only valid entries to self-heal corruption
51
+ if (lines.length > 0) {
52
+ const content = lines.map((line) => JSON.stringify(line)).join("\n") + "\n"
53
+ writeFile(historyFile.name!, content).catch(() => {})
54
+ }
55
+ })
56
+
57
+ const [store, setStore] = createStore({
58
+ index: 0,
59
+ history: [] as PromptInfo[],
60
+ })
61
+
62
+ return {
63
+ move(direction: 1 | -1, input: string) {
64
+ if (!store.history.length) return undefined
65
+ const current = store.history.at(store.index)
66
+ if (!current) return undefined
67
+ if (current.input !== input && input.length) return
68
+ setStore(
69
+ produce((draft) => {
70
+ const next = store.index + direction
71
+ if (Math.abs(next) > store.history.length) return
72
+ if (next > 0) return
73
+ draft.index = next
74
+ }),
75
+ )
76
+ if (store.index === 0)
77
+ return {
78
+ input: "",
79
+ parts: [],
80
+ }
81
+ return store.history.at(store.index)
82
+ },
83
+ append(item: PromptInfo) {
84
+ const entry = clone(item)
85
+ let trimmed = false
86
+ setStore(
87
+ produce((draft) => {
88
+ draft.history.push(entry)
89
+ if (draft.history.length > MAX_HISTORY_ENTRIES) {
90
+ draft.history = draft.history.slice(-MAX_HISTORY_ENTRIES)
91
+ trimmed = true
92
+ }
93
+ draft.index = 0
94
+ }),
95
+ )
96
+
97
+ if (trimmed) {
98
+ const content = store.history.map((line) => JSON.stringify(line)).join("\n") + "\n"
99
+ writeFile(historyFile.name!, content).catch(() => {})
100
+ return
101
+ }
102
+
103
+ appendFile(historyFile.name!, JSON.stringify(entry) + "\n").catch(() => {})
104
+ },
105
+ }
106
+ },
107
+ })