bincode-cli 1.0.1

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 (300) hide show
  1. package/AGENTS.md +27 -0
  2. package/README.md +15 -0
  3. package/bin/bincode +98 -0
  4. package/bunfig.toml +4 -0
  5. package/package.json +124 -0
  6. package/parsers-config.ts +239 -0
  7. package/script/build.ts +167 -0
  8. package/script/postinstall.mjs +206 -0
  9. package/script/publish.ts +99 -0
  10. package/script/schema.ts +47 -0
  11. package/src/acp/README.md +164 -0
  12. package/src/acp/agent.ts +1051 -0
  13. package/src/acp/session.ts +101 -0
  14. package/src/acp/types.ts +22 -0
  15. package/src/agent/agent.ts +398 -0
  16. package/src/agent/generate.txt +75 -0
  17. package/src/agent/prompt/compaction.txt +12 -0
  18. package/src/agent/prompt/explore.txt +18 -0
  19. package/src/agent/prompt/summary.txt +10 -0
  20. package/src/agent/prompt/title.txt +36 -0
  21. package/src/auth/bineric-login.ts +506 -0
  22. package/src/auth/index.ts +70 -0
  23. package/src/bun/index.ts +114 -0
  24. package/src/bus/bus-event.ts +43 -0
  25. package/src/bus/global.ts +10 -0
  26. package/src/bus/index.ts +105 -0
  27. package/src/cli/auth-check.ts +61 -0
  28. package/src/cli/bootstrap.ts +21 -0
  29. package/src/cli/cmd/acp.ts +88 -0
  30. package/src/cli/cmd/agent.ts +256 -0
  31. package/src/cli/cmd/auth.ts +436 -0
  32. package/src/cli/cmd/cmd.ts +7 -0
  33. package/src/cli/cmd/debug/config.ts +15 -0
  34. package/src/cli/cmd/debug/file.ts +91 -0
  35. package/src/cli/cmd/debug/index.ts +43 -0
  36. package/src/cli/cmd/debug/lsp.ts +48 -0
  37. package/src/cli/cmd/debug/ripgrep.ts +83 -0
  38. package/src/cli/cmd/debug/scrap.ts +15 -0
  39. package/src/cli/cmd/debug/skill.ts +15 -0
  40. package/src/cli/cmd/debug/snapshot.ts +48 -0
  41. package/src/cli/cmd/export.ts +88 -0
  42. package/src/cli/cmd/generate.ts +38 -0
  43. package/src/cli/cmd/github.ts +1399 -0
  44. package/src/cli/cmd/import.ts +98 -0
  45. package/src/cli/cmd/login.ts +112 -0
  46. package/src/cli/cmd/logout.ts +38 -0
  47. package/src/cli/cmd/mcp.ts +654 -0
  48. package/src/cli/cmd/models.ts +77 -0
  49. package/src/cli/cmd/pr.ts +112 -0
  50. package/src/cli/cmd/run.ts +368 -0
  51. package/src/cli/cmd/serve.ts +31 -0
  52. package/src/cli/cmd/session.ts +106 -0
  53. package/src/cli/cmd/stats.ts +298 -0
  54. package/src/cli/cmd/tui/app.tsx +669 -0
  55. package/src/cli/cmd/tui/attach.ts +30 -0
  56. package/src/cli/cmd/tui/component/border.tsx +21 -0
  57. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  58. package/src/cli/cmd/tui/component/dialog-command.tsx +123 -0
  59. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  60. package/src/cli/cmd/tui/component/dialog-model.tsx +223 -0
  61. package/src/cli/cmd/tui/component/dialog-provider.tsx +224 -0
  62. package/src/cli/cmd/tui/component/dialog-session-list.tsx +102 -0
  63. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  64. package/src/cli/cmd/tui/component/dialog-status.tsx +162 -0
  65. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  66. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  67. package/src/cli/cmd/tui/component/logo.tsx +32 -0
  68. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +560 -0
  69. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  70. package/src/cli/cmd/tui/component/prompt/index.tsx +1052 -0
  71. package/src/cli/cmd/tui/context/args.tsx +14 -0
  72. package/src/cli/cmd/tui/context/directory.ts +13 -0
  73. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  74. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  75. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  76. package/src/cli/cmd/tui/context/kv.tsx +49 -0
  77. package/src/cli/cmd/tui/context/local.tsx +339 -0
  78. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  79. package/src/cli/cmd/tui/context/route.tsx +46 -0
  80. package/src/cli/cmd/tui/context/sdk.tsx +74 -0
  81. package/src/cli/cmd/tui/context/sync.tsx +372 -0
  82. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  83. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  84. package/src/cli/cmd/tui/context/theme/bincode.json +245 -0
  85. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  86. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  87. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  88. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  89. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  90. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  91. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  92. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  93. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  94. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  95. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  96. package/src/cli/cmd/tui/context/theme/lucent-orng.json +227 -0
  97. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  98. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  99. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  100. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  101. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  102. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  103. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  104. package/src/cli/cmd/tui/context/theme/orng.json +245 -0
  105. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  106. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  107. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  108. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  109. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  110. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  111. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  112. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  113. package/src/cli/cmd/tui/context/theme.tsx +1109 -0
  114. package/src/cli/cmd/tui/event.ts +40 -0
  115. package/src/cli/cmd/tui/routes/home.tsx +105 -0
  116. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  117. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  118. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  119. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  120. package/src/cli/cmd/tui/routes/session/footer.tsx +88 -0
  121. package/src/cli/cmd/tui/routes/session/header.tsx +141 -0
  122. package/src/cli/cmd/tui/routes/session/index.tsx +1888 -0
  123. package/src/cli/cmd/tui/routes/session/sidebar.tsx +321 -0
  124. package/src/cli/cmd/tui/spawn.ts +60 -0
  125. package/src/cli/cmd/tui/thread.ts +120 -0
  126. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  127. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  128. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  129. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  130. package/src/cli/cmd/tui/ui/dialog-select.tsx +330 -0
  131. package/src/cli/cmd/tui/ui/dialog.tsx +170 -0
  132. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  133. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  134. package/src/cli/cmd/tui/util/clipboard.ts +127 -0
  135. package/src/cli/cmd/tui/util/editor.ts +32 -0
  136. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  137. package/src/cli/cmd/tui/worker.ts +63 -0
  138. package/src/cli/cmd/uninstall.ts +344 -0
  139. package/src/cli/cmd/upgrade.ts +67 -0
  140. package/src/cli/cmd/web.ts +84 -0
  141. package/src/cli/error.ts +55 -0
  142. package/src/cli/ui.ts +84 -0
  143. package/src/cli/upgrade.ts +25 -0
  144. package/src/command/index.ts +80 -0
  145. package/src/command/template/initialize.txt +10 -0
  146. package/src/command/template/review.txt +97 -0
  147. package/src/config/config.ts +995 -0
  148. package/src/config/markdown.ts +41 -0
  149. package/src/env/index.ts +26 -0
  150. package/src/file/ignore.ts +83 -0
  151. package/src/file/index.ts +328 -0
  152. package/src/file/ripgrep.ts +393 -0
  153. package/src/file/time.ts +64 -0
  154. package/src/file/watcher.ts +103 -0
  155. package/src/flag/flag.ts +46 -0
  156. package/src/format/formatter.ts +315 -0
  157. package/src/format/index.ts +137 -0
  158. package/src/global/index.ts +52 -0
  159. package/src/id/id.ts +73 -0
  160. package/src/ide/index.ts +76 -0
  161. package/src/index.ts +217 -0
  162. package/src/installation/index.ts +196 -0
  163. package/src/lsp/client.ts +229 -0
  164. package/src/lsp/index.ts +485 -0
  165. package/src/lsp/language.ts +116 -0
  166. package/src/lsp/server.ts +1895 -0
  167. package/src/mcp/auth.ts +135 -0
  168. package/src/mcp/index.ts +654 -0
  169. package/src/mcp/oauth-callback.ts +200 -0
  170. package/src/mcp/oauth-provider.ts +154 -0
  171. package/src/patch/index.ts +622 -0
  172. package/src/permission/index.ts +199 -0
  173. package/src/plugin/index.ts +101 -0
  174. package/src/project/bootstrap.ts +31 -0
  175. package/src/project/instance.ts +78 -0
  176. package/src/project/project.ts +221 -0
  177. package/src/project/state.ts +65 -0
  178. package/src/project/vcs.ts +76 -0
  179. package/src/provider/auth.ts +143 -0
  180. package/src/provider/models-macro.ts +11 -0
  181. package/src/provider/models.ts +106 -0
  182. package/src/provider/provider.ts +1071 -0
  183. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  184. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  185. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +101 -0
  186. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  187. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
  188. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  189. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  190. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  191. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1713 -0
  192. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  193. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  194. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  195. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  196. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  197. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  198. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  199. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  200. package/src/provider/transform.ts +455 -0
  201. package/src/pty/index.ts +231 -0
  202. package/src/server/error.ts +36 -0
  203. package/src/server/project.ts +79 -0
  204. package/src/server/server.ts +2642 -0
  205. package/src/server/tui.ts +71 -0
  206. package/src/session/compaction.ts +223 -0
  207. package/src/session/index.ts +458 -0
  208. package/src/session/llm.ts +201 -0
  209. package/src/session/message-v2.ts +659 -0
  210. package/src/session/message.ts +189 -0
  211. package/src/session/processor.ts +409 -0
  212. package/src/session/prompt/anthropic-20250930.txt +166 -0
  213. package/src/session/prompt/anthropic.txt +104 -0
  214. package/src/session/prompt/anthropic_spoof.txt +1 -0
  215. package/src/session/prompt/beast.txt +147 -0
  216. package/src/session/prompt/build-switch.txt +5 -0
  217. package/src/session/prompt/codex.txt +318 -0
  218. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  219. package/src/session/prompt/gemini.txt +155 -0
  220. package/src/session/prompt/max-steps.txt +16 -0
  221. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  222. package/src/session/prompt/plan.txt +26 -0
  223. package/src/session/prompt/polaris.txt +106 -0
  224. package/src/session/prompt/qwen.txt +109 -0
  225. package/src/session/prompt.ts +1446 -0
  226. package/src/session/retry.ts +86 -0
  227. package/src/session/revert.ts +108 -0
  228. package/src/session/status.ts +76 -0
  229. package/src/session/summary.ts +194 -0
  230. package/src/session/system.ts +120 -0
  231. package/src/session/todo.ts +37 -0
  232. package/src/share/share-next.ts +194 -0
  233. package/src/share/share.ts +87 -0
  234. package/src/shell/shell.ts +67 -0
  235. package/src/skill/index.ts +1 -0
  236. package/src/skill/skill.ts +83 -0
  237. package/src/snapshot/index.ts +197 -0
  238. package/src/storage/storage.ts +226 -0
  239. package/src/tool/bash.ts +306 -0
  240. package/src/tool/bash.txt +158 -0
  241. package/src/tool/batch.ts +175 -0
  242. package/src/tool/batch.txt +24 -0
  243. package/src/tool/codesearch.ts +138 -0
  244. package/src/tool/codesearch.txt +12 -0
  245. package/src/tool/edit.ts +675 -0
  246. package/src/tool/edit.txt +10 -0
  247. package/src/tool/glob.ts +65 -0
  248. package/src/tool/glob.txt +6 -0
  249. package/src/tool/grep.ts +121 -0
  250. package/src/tool/grep.txt +8 -0
  251. package/src/tool/invalid.ts +17 -0
  252. package/src/tool/ls.ts +110 -0
  253. package/src/tool/ls.txt +1 -0
  254. package/src/tool/lsp-diagnostics.ts +26 -0
  255. package/src/tool/lsp-diagnostics.txt +1 -0
  256. package/src/tool/lsp-hover.ts +31 -0
  257. package/src/tool/lsp-hover.txt +1 -0
  258. package/src/tool/lsp.ts +87 -0
  259. package/src/tool/lsp.txt +19 -0
  260. package/src/tool/multiedit.ts +46 -0
  261. package/src/tool/multiedit.txt +41 -0
  262. package/src/tool/patch.ts +233 -0
  263. package/src/tool/patch.txt +1 -0
  264. package/src/tool/read.ts +219 -0
  265. package/src/tool/read.txt +12 -0
  266. package/src/tool/registry.ts +162 -0
  267. package/src/tool/skill.ts +100 -0
  268. package/src/tool/task.ts +136 -0
  269. package/src/tool/task.txt +60 -0
  270. package/src/tool/todo.ts +39 -0
  271. package/src/tool/todoread.txt +14 -0
  272. package/src/tool/todowrite.txt +167 -0
  273. package/src/tool/tool.ts +71 -0
  274. package/src/tool/webfetch.ts +187 -0
  275. package/src/tool/webfetch.txt +13 -0
  276. package/src/tool/websearch.ts +150 -0
  277. package/src/tool/websearch.txt +11 -0
  278. package/src/tool/write.ts +110 -0
  279. package/src/tool/write.txt +8 -0
  280. package/src/util/archive.ts +16 -0
  281. package/src/util/color.ts +19 -0
  282. package/src/util/context.ts +25 -0
  283. package/src/util/defer.ts +12 -0
  284. package/src/util/eventloop.ts +20 -0
  285. package/src/util/filesystem.ts +83 -0
  286. package/src/util/fn.ts +11 -0
  287. package/src/util/iife.ts +3 -0
  288. package/src/util/keybind.ts +102 -0
  289. package/src/util/lazy.ts +11 -0
  290. package/src/util/locale.ts +81 -0
  291. package/src/util/lock.ts +98 -0
  292. package/src/util/log.ts +180 -0
  293. package/src/util/queue.ts +32 -0
  294. package/src/util/rpc.ts +42 -0
  295. package/src/util/scrap.ts +10 -0
  296. package/src/util/signal.ts +12 -0
  297. package/src/util/timeout.ts +14 -0
  298. package/src/util/token.ts +7 -0
  299. package/src/util/wildcard.ts +54 -0
  300. package/tsconfig.json +16 -0
@@ -0,0 +1,321 @@
1
+ import { useSync } from "@tui/context/sync"
2
+ import { createMemo, For, Show, Switch, Match } from "solid-js"
3
+ import { createStore } from "solid-js/store"
4
+ import { useTheme } from "../../context/theme"
5
+ import { Locale } from "@/util/locale"
6
+ import path from "path"
7
+ import type { AssistantMessage } from "@bincode-ai/sdk/v2"
8
+ import { Global } from "@/global"
9
+ import { Installation } from "@/installation"
10
+ import { useKeybind } from "../../context/keybind"
11
+ import { useDirectory } from "../../context/directory"
12
+ import { useKV } from "../../context/kv"
13
+
14
+ export function Sidebar(props: { sessionID: string }) {
15
+ const sync = useSync()
16
+ const { theme } = useTheme()
17
+ const session = createMemo(() => sync.session.get(props.sessionID)!)
18
+ const diff = createMemo(() => sync.data.session_diff[props.sessionID] ?? [])
19
+ const todo = createMemo(() => sync.data.todo[props.sessionID] ?? [])
20
+ const messages = createMemo(() => sync.data.message[props.sessionID] ?? [])
21
+
22
+ const [expanded, setExpanded] = createStore({
23
+ mcp: true,
24
+ diff: true,
25
+ todo: true,
26
+ lsp: true,
27
+ })
28
+
29
+ // Sort MCP servers alphabetically for consistent display order
30
+ const mcpEntries = createMemo(() => Object.entries(sync.data.mcp).sort(([a], [b]) => a.localeCompare(b)))
31
+
32
+ // Count connected and error MCP servers for collapsed header display
33
+ const connectedMcpCount = createMemo(() => mcpEntries().filter(([_, item]) => item.status === "connected").length)
34
+ const errorMcpCount = createMemo(
35
+ () =>
36
+ mcpEntries().filter(
37
+ ([_, item]) =>
38
+ item.status === "failed" || item.status === "needs_auth" || item.status === "needs_client_registration",
39
+ ).length,
40
+ )
41
+
42
+ const cost = createMemo(() => {
43
+ const total = messages().reduce((sum, x) => sum + (x.role === "assistant" ? x.cost : 0), 0)
44
+ return new Intl.NumberFormat("en-US", {
45
+ style: "currency",
46
+ currency: "USD",
47
+ }).format(total)
48
+ })
49
+
50
+ const context = createMemo(() => {
51
+ const last = messages().findLast((x) => x.role === "assistant" && x.tokens.output > 0) as AssistantMessage
52
+ if (!last) return
53
+ const total =
54
+ last.tokens.input + last.tokens.output + last.tokens.reasoning + last.tokens.cache.read + last.tokens.cache.write
55
+ const model = sync.data.provider.find((x) => x.id === last.providerID)?.models[last.modelID]
56
+ return {
57
+ tokens: total.toLocaleString(),
58
+ percentage: model?.limit.context ? Math.round((total / model.limit.context) * 100) : null,
59
+ }
60
+ })
61
+
62
+ const directory = useDirectory()
63
+ const kv = useKV()
64
+
65
+ const hasProviders = createMemo(() =>
66
+ sync.data.provider.some((x) => x.id !== "bincode" || Object.values(x.models).some((y) => y.cost?.input !== 0)),
67
+ )
68
+ const gettingStartedDismissed = createMemo(() => kv.get("dismissed_getting_started", false))
69
+
70
+ return (
71
+ <Show when={session()}>
72
+ <box
73
+ backgroundColor={theme.backgroundPanel}
74
+ width={42}
75
+ height="100%"
76
+ paddingTop={1}
77
+ paddingBottom={1}
78
+ paddingLeft={2}
79
+ paddingRight={2}
80
+ >
81
+ <scrollbox flexGrow={1}>
82
+ <box flexShrink={0} gap={1} paddingRight={1}>
83
+ <box>
84
+ <text fg={theme.text}>
85
+ <b>{session().title}</b>
86
+ </text>
87
+ <Show when={session().share?.url}>
88
+ <text fg={theme.textMuted}>{session().share!.url}</text>
89
+ </Show>
90
+ </box>
91
+ <box>
92
+ <text fg={theme.text}>
93
+ <b>Context</b>
94
+ </text>
95
+ <text fg={theme.textMuted}>{context()?.tokens ?? 0} tokens</text>
96
+ <text fg={theme.textMuted}>{context()?.percentage ?? 0}% used</text>
97
+ <text fg={theme.textMuted}>{cost()} spent</text>
98
+ </box>
99
+ <Show when={mcpEntries().length > 0}>
100
+ <box>
101
+ <box
102
+ flexDirection="row"
103
+ gap={1}
104
+ onMouseDown={() => mcpEntries().length > 2 && setExpanded("mcp", !expanded.mcp)}
105
+ >
106
+ <Show when={mcpEntries().length > 2}>
107
+ <text fg={theme.text}>{expanded.mcp ? "▼" : "▶"}</text>
108
+ </Show>
109
+ <text fg={theme.text}>
110
+ <b>MCP</b>
111
+ <Show when={!expanded.mcp}>
112
+ <span style={{ fg: theme.textMuted }}>
113
+ {" "}
114
+ ({connectedMcpCount()} active
115
+ {errorMcpCount() > 0 ? `, ${errorMcpCount()} error${errorMcpCount() > 1 ? "s" : ""}` : ""})
116
+ </span>
117
+ </Show>
118
+ </text>
119
+ </box>
120
+ <Show when={mcpEntries().length <= 2 || expanded.mcp}>
121
+ <For each={mcpEntries()}>
122
+ {([key, item]) => (
123
+ <box flexDirection="row" gap={1}>
124
+ <text
125
+ flexShrink={0}
126
+ style={{
127
+ fg: (
128
+ {
129
+ connected: theme.success,
130
+ failed: theme.error,
131
+ disabled: theme.textMuted,
132
+ needs_auth: theme.warning,
133
+ needs_client_registration: theme.error,
134
+ } as Record<string, typeof theme.success>
135
+ )[item.status],
136
+ }}
137
+ >
138
+
139
+ </text>
140
+ <text fg={theme.text} wrapMode="word">
141
+ {key}{" "}
142
+ <span style={{ fg: theme.textMuted }}>
143
+ <Switch fallback={item.status}>
144
+ <Match when={item.status === "connected"}>Connected</Match>
145
+ <Match when={item.status === "failed" && item}>{(val) => <i>{val().error}</i>}</Match>
146
+ <Match when={item.status === "disabled"}>Disabled</Match>
147
+ <Match when={(item.status as string) === "needs_auth"}>Needs auth</Match>
148
+ <Match when={(item.status as string) === "needs_client_registration"}>
149
+ Needs client ID
150
+ </Match>
151
+ </Switch>
152
+ </span>
153
+ </text>
154
+ </box>
155
+ )}
156
+ </For>
157
+ </Show>
158
+ </box>
159
+ </Show>
160
+ <box>
161
+ <box
162
+ flexDirection="row"
163
+ gap={1}
164
+ onMouseDown={() => sync.data.lsp.length > 2 && setExpanded("lsp", !expanded.lsp)}
165
+ >
166
+ <Show when={sync.data.lsp.length > 2}>
167
+ <text fg={theme.text}>{expanded.lsp ? "▼" : "▶"}</text>
168
+ </Show>
169
+ <text fg={theme.text}>
170
+ <b>LSP</b>
171
+ </text>
172
+ </box>
173
+ <Show when={sync.data.lsp.length <= 2 || expanded.lsp}>
174
+ <Show when={sync.data.lsp.length === 0}>
175
+ <text fg={theme.textMuted}>
176
+ {sync.data.config.lsp === false
177
+ ? "LSPs have been disabled in settings"
178
+ : "LSPs will activate as files are read"}
179
+ </text>
180
+ </Show>
181
+ <For each={sync.data.lsp}>
182
+ {(item) => (
183
+ <box flexDirection="row" gap={1}>
184
+ <text
185
+ flexShrink={0}
186
+ style={{
187
+ fg: {
188
+ connected: theme.success,
189
+ error: theme.error,
190
+ }[item.status],
191
+ }}
192
+ >
193
+
194
+ </text>
195
+ <text fg={theme.textMuted}>
196
+ {item.id} {item.root}
197
+ </text>
198
+ </box>
199
+ )}
200
+ </For>
201
+ </Show>
202
+ </box>
203
+ <Show when={todo().length > 0 && todo().some((t) => t.status !== "completed")}>
204
+ <box>
205
+ <box
206
+ flexDirection="row"
207
+ gap={1}
208
+ onMouseDown={() => todo().length > 2 && setExpanded("todo", !expanded.todo)}
209
+ >
210
+ <Show when={todo().length > 2}>
211
+ <text fg={theme.text}>{expanded.todo ? "▼" : "▶"}</text>
212
+ </Show>
213
+ <text fg={theme.text}>
214
+ <b>Todo</b>
215
+ </text>
216
+ </box>
217
+ <Show when={todo().length <= 2 || expanded.todo}>
218
+ <For each={todo()}>
219
+ {(todo) => (
220
+ <text style={{ fg: todo.status === "in_progress" ? theme.success : theme.textMuted }}>
221
+ [{todo.status === "completed" ? "✓" : " "}] {todo.content}
222
+ </text>
223
+ )}
224
+ </For>
225
+ </Show>
226
+ </box>
227
+ </Show>
228
+ <Show when={diff().length > 0}>
229
+ <box>
230
+ <box
231
+ flexDirection="row"
232
+ gap={1}
233
+ onMouseDown={() => diff().length > 2 && setExpanded("diff", !expanded.diff)}
234
+ >
235
+ <Show when={diff().length > 2}>
236
+ <text fg={theme.text}>{expanded.diff ? "▼" : "▶"}</text>
237
+ </Show>
238
+ <text fg={theme.text}>
239
+ <b>Modified Files</b>
240
+ </text>
241
+ </box>
242
+ <Show when={diff().length <= 2 || expanded.diff}>
243
+ <For each={diff() || []}>
244
+ {(item) => {
245
+ const file = createMemo(() => {
246
+ const splits = item.file.split(path.sep).filter(Boolean)
247
+ const last = splits.at(-1)!
248
+ const rest = splits.slice(0, -1).join(path.sep)
249
+ if (!rest) return last
250
+ return Locale.truncateMiddle(rest, 30 - last.length) + "/" + last
251
+ })
252
+ return (
253
+ <box flexDirection="row" gap={1} justifyContent="space-between">
254
+ <text fg={theme.textMuted} wrapMode="char">
255
+ {file()}
256
+ </text>
257
+ <box flexDirection="row" gap={1} flexShrink={0}>
258
+ <Show when={item.additions}>
259
+ <text fg={theme.diffAdded}>+{item.additions}</text>
260
+ </Show>
261
+ <Show when={item.deletions}>
262
+ <text fg={theme.diffRemoved}>-{item.deletions}</text>
263
+ </Show>
264
+ </box>
265
+ </box>
266
+ )
267
+ }}
268
+ </For>
269
+ </Show>
270
+ </box>
271
+ </Show>
272
+ </box>
273
+ </scrollbox>
274
+
275
+ <box flexShrink={0} gap={1} paddingTop={1}>
276
+ <Show when={!hasProviders() && !gettingStartedDismissed()}>
277
+ <box
278
+ backgroundColor={theme.backgroundElement}
279
+ paddingTop={1}
280
+ paddingBottom={1}
281
+ paddingLeft={2}
282
+ paddingRight={2}
283
+ flexDirection="row"
284
+ gap={1}
285
+ >
286
+ <text flexShrink={0} fg={theme.text}>
287
+
288
+ </text>
289
+ <box flexGrow={1} gap={1}>
290
+ <box flexDirection="row" justifyContent="space-between">
291
+ <text fg={theme.text}>
292
+ <b>Getting started</b>
293
+ </text>
294
+ <text fg={theme.textMuted} onMouseDown={() => kv.set("dismissed_getting_started", true)}>
295
+
296
+ </text>
297
+ </box>
298
+ <text fg={theme.textMuted}>Bincode includes free models so you can start immediately.</text>
299
+ <text fg={theme.textMuted}>
300
+ Connect from 75+ providers to use other models, including Claude, GPT, Gemini etc
301
+ </text>
302
+ <box flexDirection="row" gap={1} justifyContent="space-between">
303
+ <text fg={theme.text}>Connect provider</text>
304
+ <text fg={theme.textMuted}>/connect</text>
305
+ </box>
306
+ </box>
307
+ </box>
308
+ </Show>
309
+ <text>
310
+ <span style={{ fg: theme.textMuted }}>{directory().split("/").slice(0, -1).join("/")}/</span>
311
+ <span style={{ fg: theme.text }}>{directory().split("/").at(-1)}</span>
312
+ </text>
313
+ <text fg={theme.textMuted}>
314
+ <span style={{ fg: theme.success }}>•</span> <b>Bincode</b>{" "}
315
+ <span>{Installation.VERSION}</span>
316
+ </text>
317
+ </box>
318
+ </box>
319
+ </Show>
320
+ )
321
+ }
@@ -0,0 +1,60 @@
1
+ import { cmd } from "@/cli/cmd/cmd"
2
+ import { Instance } from "@/project/instance"
3
+ import path from "path"
4
+ import { Server } from "@/server/server"
5
+ import { upgrade } from "@/cli/upgrade"
6
+
7
+ export const TuiSpawnCommand = cmd({
8
+ command: "spawn [project]",
9
+ builder: (yargs) =>
10
+ yargs
11
+ .positional("project", {
12
+ type: "string",
13
+ describe: "path to start bincode in",
14
+ })
15
+ .option("port", {
16
+ type: "number",
17
+ describe: "port to listen on",
18
+ default: 0,
19
+ })
20
+ .option("hostname", {
21
+ type: "string",
22
+ describe: "hostname to listen on",
23
+ default: "127.0.0.1",
24
+ }),
25
+ handler: async (args) => {
26
+ upgrade()
27
+ const server = Server.listen({
28
+ port: args.port,
29
+ hostname: "127.0.0.1",
30
+ })
31
+ const bin = process.execPath
32
+ const cmd = []
33
+ let cwd = process.cwd()
34
+ if (bin.endsWith("bun")) {
35
+ cmd.push(
36
+ process.execPath,
37
+ "run",
38
+ "--conditions",
39
+ "browser",
40
+ new URL("../../../index.ts", import.meta.url).pathname,
41
+ )
42
+ cwd = new URL("../../../../", import.meta.url).pathname
43
+ } else cmd.push(process.execPath)
44
+ cmd.push("attach", server.url.toString(), "--dir", args.project ? path.resolve(args.project) : process.cwd())
45
+ const proc = Bun.spawn({
46
+ cmd,
47
+ cwd,
48
+ stdout: "inherit",
49
+ stderr: "inherit",
50
+ stdin: "inherit",
51
+ env: {
52
+ ...process.env,
53
+ BUN_OPTIONS: "",
54
+ },
55
+ })
56
+ await proc.exited
57
+ await Instance.disposeAll()
58
+ await server.stop(true)
59
+ },
60
+ })
@@ -0,0 +1,120 @@
1
+ import { cmd } from "@/cli/cmd/cmd"
2
+ import { tui } from "./app"
3
+ import { Rpc } from "@/util/rpc"
4
+ import { type rpc } from "./worker"
5
+ import path from "path"
6
+ import { UI } from "@/cli/ui"
7
+ import { iife } from "@/util/iife"
8
+ import { Log } from "@/util/log"
9
+
10
+ declare global {
11
+ const BINCODE_WORKER_PATH: string
12
+ }
13
+
14
+ export const TuiThreadCommand = cmd({
15
+ command: "$0 [project]",
16
+ describe: "start bincode tui",
17
+ builder: (yargs) =>
18
+ yargs
19
+ .positional("project", {
20
+ type: "string",
21
+ describe: "path to start bincode in",
22
+ })
23
+ .option("model", {
24
+ type: "string",
25
+ alias: ["m"],
26
+ describe: "model to use in the format of provider/model",
27
+ })
28
+ .option("continue", {
29
+ alias: ["c"],
30
+ describe: "continue the last session",
31
+ type: "boolean",
32
+ })
33
+ .option("session", {
34
+ alias: ["s"],
35
+ type: "string",
36
+ describe: "session id to continue",
37
+ })
38
+ .option("prompt", {
39
+ alias: ["p"],
40
+ type: "string",
41
+ describe: "prompt to use",
42
+ })
43
+ .option("agent", {
44
+ type: "string",
45
+ describe: "agent to use",
46
+ })
47
+ .option("port", {
48
+ type: "number",
49
+ describe: "port to listen on",
50
+ default: 0,
51
+ })
52
+ .option("hostname", {
53
+ type: "string",
54
+ describe: "hostname to listen on",
55
+ default: "127.0.0.1",
56
+ }),
57
+ handler: async (args) => {
58
+ // Resolve relative paths against PWD to preserve behavior when using --cwd flag
59
+ const baseCwd = process.env.PWD ?? process.cwd()
60
+ const cwd = args.project ? path.resolve(baseCwd, args.project) : process.cwd()
61
+ const localWorker = new URL("./worker.ts", import.meta.url)
62
+ const distWorker = new URL("./cli/cmd/tui/worker.js", import.meta.url)
63
+ const workerPath = await iife(async () => {
64
+ if (typeof BINCODE_WORKER_PATH !== "undefined") return BINCODE_WORKER_PATH
65
+ if (await Bun.file(distWorker).exists()) return distWorker
66
+ return localWorker
67
+ })
68
+ try {
69
+ process.chdir(cwd)
70
+ } catch (e) {
71
+ UI.error("Failed to change directory to " + cwd)
72
+ return
73
+ }
74
+
75
+ const worker = new Worker(workerPath, {
76
+ env: Object.fromEntries(
77
+ Object.entries(process.env).filter((entry): entry is [string, string] => entry[1] !== undefined),
78
+ ),
79
+ })
80
+ worker.onerror = (e) => {
81
+ Log.Default.error(e)
82
+ }
83
+ const client = Rpc.client<typeof rpc>(worker)
84
+ process.on("uncaughtException", (e) => {
85
+ Log.Default.error(e)
86
+ })
87
+ process.on("unhandledRejection", (e) => {
88
+ Log.Default.error(e)
89
+ })
90
+ const server = await client.call("server", {
91
+ port: args.port,
92
+ hostname: args.hostname,
93
+ })
94
+ const prompt = await iife(async () => {
95
+ const piped = !process.stdin.isTTY ? await Bun.stdin.text() : undefined
96
+ if (!args.prompt) return piped
97
+ return piped ? piped + "\n" + args.prompt : args.prompt
98
+ })
99
+
100
+ const tuiPromise = tui({
101
+ url: server.url,
102
+ args: {
103
+ continue: args.continue,
104
+ sessionID: args.session,
105
+ agent: args.agent,
106
+ model: args.model,
107
+ prompt,
108
+ },
109
+ onExit: async () => {
110
+ await client.call("shutdown", undefined)
111
+ },
112
+ })
113
+
114
+ setTimeout(() => {
115
+ client.call("checkUpgrade", { directory: cwd }).catch(() => {})
116
+ }, 1000)
117
+
118
+ await tuiPromise
119
+ },
120
+ })
@@ -0,0 +1,57 @@
1
+ import { TextAttributes } from "@opentui/core"
2
+ import { useTheme } from "../context/theme"
3
+ import { useDialog, type DialogContext } from "./dialog"
4
+ import { useKeyboard } from "@opentui/solid"
5
+
6
+ export type DialogAlertProps = {
7
+ title: string
8
+ message: string
9
+ onConfirm?: () => void
10
+ }
11
+
12
+ export function DialogAlert(props: DialogAlertProps) {
13
+ const dialog = useDialog()
14
+ const { theme } = useTheme()
15
+
16
+ useKeyboard((evt) => {
17
+ if (evt.name === "return") {
18
+ props.onConfirm?.()
19
+ dialog.clear()
20
+ }
21
+ })
22
+ return (
23
+ <box paddingLeft={2} paddingRight={2} gap={1}>
24
+ <box flexDirection="row" justifyContent="space-between">
25
+ <text attributes={TextAttributes.BOLD} fg={theme.text}>
26
+ {props.title}
27
+ </text>
28
+ <text fg={theme.textMuted}>esc</text>
29
+ </box>
30
+ <box paddingBottom={1}>
31
+ <text fg={theme.textMuted}>{props.message}</text>
32
+ </box>
33
+ <box flexDirection="row" justifyContent="flex-end" paddingBottom={1}>
34
+ <box
35
+ paddingLeft={3}
36
+ paddingRight={3}
37
+ backgroundColor={theme.primary}
38
+ onMouseUp={() => {
39
+ props.onConfirm?.()
40
+ dialog.clear()
41
+ }}
42
+ >
43
+ <text fg={theme.selectedListItemText}>ok</text>
44
+ </box>
45
+ </box>
46
+ </box>
47
+ )
48
+ }
49
+
50
+ DialogAlert.show = (dialog: DialogContext, title: string, message: string) => {
51
+ return new Promise<void>((resolve) => {
52
+ dialog.replace(
53
+ () => <DialogAlert title={title} message={message} onConfirm={() => resolve()} />,
54
+ () => resolve(),
55
+ )
56
+ })
57
+ }
@@ -0,0 +1,83 @@
1
+ import { TextAttributes } from "@opentui/core"
2
+ import { useTheme } from "../context/theme"
3
+ import { useDialog, type DialogContext } from "./dialog"
4
+ import { createStore } from "solid-js/store"
5
+ import { For } from "solid-js"
6
+ import { useKeyboard } from "@opentui/solid"
7
+ import { Locale } from "@/util/locale"
8
+
9
+ export type DialogConfirmProps = {
10
+ title: string
11
+ message: string
12
+ onConfirm?: () => void
13
+ onCancel?: () => void
14
+ }
15
+
16
+ export function DialogConfirm(props: DialogConfirmProps) {
17
+ const dialog = useDialog()
18
+ const { theme } = useTheme()
19
+ const [store, setStore] = createStore({
20
+ active: "confirm" as "confirm" | "cancel",
21
+ })
22
+
23
+ useKeyboard((evt) => {
24
+ if (evt.name === "return") {
25
+ if (store.active === "confirm") props.onConfirm?.()
26
+ if (store.active === "cancel") props.onCancel?.()
27
+ dialog.clear()
28
+ }
29
+
30
+ if (evt.name === "left" || evt.name === "right") {
31
+ setStore("active", store.active === "confirm" ? "cancel" : "confirm")
32
+ }
33
+ })
34
+ return (
35
+ <box paddingLeft={2} paddingRight={2} gap={1}>
36
+ <box flexDirection="row" justifyContent="space-between">
37
+ <text attributes={TextAttributes.BOLD} fg={theme.text}>
38
+ {props.title}
39
+ </text>
40
+ <text fg={theme.textMuted}>esc</text>
41
+ </box>
42
+ <box paddingBottom={1}>
43
+ <text fg={theme.textMuted}>{props.message}</text>
44
+ </box>
45
+ <box flexDirection="row" justifyContent="flex-end" paddingBottom={1}>
46
+ <For each={["cancel", "confirm"]}>
47
+ {(key) => (
48
+ <box
49
+ paddingLeft={1}
50
+ paddingRight={1}
51
+ backgroundColor={key === store.active ? theme.primary : undefined}
52
+ onMouseUp={(evt) => {
53
+ if (key === "confirm") props.onConfirm?.()
54
+ if (key === "cancel") props.onCancel?.()
55
+ dialog.clear()
56
+ }}
57
+ >
58
+ <text fg={key === store.active ? theme.selectedListItemText : theme.textMuted}>
59
+ {Locale.titlecase(key)}
60
+ </text>
61
+ </box>
62
+ )}
63
+ </For>
64
+ </box>
65
+ </box>
66
+ )
67
+ }
68
+
69
+ DialogConfirm.show = (dialog: DialogContext, title: string, message: string) => {
70
+ return new Promise<boolean>((resolve) => {
71
+ dialog.replace(
72
+ () => (
73
+ <DialogConfirm
74
+ title={title}
75
+ message={message}
76
+ onConfirm={() => resolve(true)}
77
+ onCancel={() => resolve(false)}
78
+ />
79
+ ),
80
+ () => resolve(false),
81
+ )
82
+ })
83
+ }
@@ -0,0 +1,38 @@
1
+ import { TextAttributes } from "@opentui/core"
2
+ import { useTheme } from "@tui/context/theme"
3
+ import { useDialog } from "./dialog"
4
+ import { useKeyboard } from "@opentui/solid"
5
+ import { useKeybind } from "@tui/context/keybind"
6
+
7
+ export function DialogHelp() {
8
+ const dialog = useDialog()
9
+ const { theme } = useTheme()
10
+ const keybind = useKeybind()
11
+
12
+ useKeyboard((evt) => {
13
+ if (evt.name === "return" || evt.name === "escape") {
14
+ dialog.clear()
15
+ }
16
+ })
17
+
18
+ return (
19
+ <box paddingLeft={2} paddingRight={2} gap={1}>
20
+ <box flexDirection="row" justifyContent="space-between">
21
+ <text attributes={TextAttributes.BOLD} fg={theme.text}>
22
+ Help
23
+ </text>
24
+ <text fg={theme.textMuted}>esc/enter</text>
25
+ </box>
26
+ <box paddingBottom={1}>
27
+ <text fg={theme.textMuted}>
28
+ Press {keybind.print("command_list")} to see all available actions and commands in any context.
29
+ </text>
30
+ </box>
31
+ <box flexDirection="row" justifyContent="flex-end" paddingBottom={1}>
32
+ <box paddingLeft={3} paddingRight={3} backgroundColor={theme.primary} onMouseUp={() => dialog.clear()}>
33
+ <text fg={theme.selectedListItemText}>ok</text>
34
+ </box>
35
+ </box>
36
+ </box>
37
+ )
38
+ }