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
package/src/index.ts ADDED
@@ -0,0 +1,217 @@
1
+ import yargs from "yargs"
2
+ import { hideBin } from "yargs/helpers"
3
+ import { RunCommand } from "./cli/cmd/run"
4
+ import { GenerateCommand } from "./cli/cmd/generate"
5
+ import { Log } from "./util/log"
6
+ import { AuthCommand } from "./cli/cmd/auth"
7
+ import { LoginCommand } from "./cli/cmd/login"
8
+ import { LogoutCommand } from "./cli/cmd/logout"
9
+ import { AgentCommand } from "./cli/cmd/agent"
10
+ import { UpgradeCommand } from "./cli/cmd/upgrade"
11
+ import { UninstallCommand } from "./cli/cmd/uninstall"
12
+ import { ModelsCommand } from "./cli/cmd/models"
13
+ import { UI } from "./cli/ui"
14
+ import { Installation } from "./installation"
15
+ import { NamedError } from "@bincode-ai/util/error"
16
+ import { FormatError } from "./cli/error"
17
+ import { ServeCommand } from "./cli/cmd/serve"
18
+ import { DebugCommand } from "./cli/cmd/debug"
19
+ import { StatsCommand } from "./cli/cmd/stats"
20
+ import { McpCommand } from "./cli/cmd/mcp"
21
+ import { GithubCommand } from "./cli/cmd/github"
22
+ import { ExportCommand } from "./cli/cmd/export"
23
+ import { ImportCommand } from "./cli/cmd/import"
24
+ import { AttachCommand } from "./cli/cmd/tui/attach"
25
+ import { TuiThreadCommand } from "./cli/cmd/tui/thread"
26
+ import { TuiSpawnCommand } from "./cli/cmd/tui/spawn"
27
+ import { AcpCommand } from "./cli/cmd/acp"
28
+ import { EOL } from "os"
29
+ import { WebCommand } from "./cli/cmd/web"
30
+ import { PrCommand } from "./cli/cmd/pr"
31
+ import { SessionCommand } from "./cli/cmd/session"
32
+ import { checkAuth } from "./cli/auth-check"
33
+ import { Filesystem } from "./util/filesystem"
34
+
35
+ // Load .env file from project root (up the directory tree)
36
+ // Bun automatically loads .env from current directory, but since we run with --cwd packages/bincode,
37
+ // we need to manually load from project root
38
+ async function loadEnvFile() {
39
+ const currentDir = process.cwd()
40
+ const envFiles = await Filesystem.findUp(".env", currentDir)
41
+ if (envFiles.length > 0) {
42
+ // Use the first .env file found (closest to current directory)
43
+ const envPath = envFiles[0]
44
+ try {
45
+ // Use Bun's built-in dotenv parser
46
+ const envContent = await Bun.file(envPath).text()
47
+ const lines = envContent.split("\n")
48
+ for (const line of lines) {
49
+ const trimmed = line.trim()
50
+ // Skip comments and empty lines
51
+ if (!trimmed || trimmed.startsWith("#")) continue
52
+ const match = trimmed.match(/^([^=]+)=(.*)$/)
53
+ if (match) {
54
+ const key = match[1].trim()
55
+ let value = match[2].trim()
56
+ // Remove quotes if present
57
+ if ((value.startsWith('"') && value.endsWith('"')) || (value.startsWith("'") && value.endsWith("'"))) {
58
+ value = value.slice(1, -1)
59
+ }
60
+ // Only set if not already set (env vars take precedence over .env file)
61
+ if (!process.env[key]) {
62
+ process.env[key] = value
63
+ }
64
+ }
65
+ }
66
+ } catch (error) {
67
+ // Silently fail if .env file can't be read
68
+ }
69
+ }
70
+ }
71
+
72
+ // Load .env file before anything else
73
+ await loadEnvFile()
74
+
75
+ process.on("unhandledRejection", (e) => {
76
+ Log.Default.error("rejection", {
77
+ e: e instanceof Error ? e.message : e,
78
+ })
79
+ })
80
+
81
+ process.on("uncaughtException", (e) => {
82
+ Log.Default.error("exception", {
83
+ e: e instanceof Error ? e.message : e,
84
+ })
85
+ })
86
+
87
+ const cli = yargs(hideBin(process.argv))
88
+ .parserConfiguration({ "populate--": true })
89
+ .scriptName("bincode")
90
+ .wrap(100)
91
+ .help("help", "show help")
92
+ .alias("help", "h")
93
+ .version("version", "show version number", Installation.VERSION)
94
+ .alias("version", "v")
95
+ .option("print-logs", {
96
+ describe: "print logs to stderr",
97
+ type: "boolean",
98
+ })
99
+ .option("log-level", {
100
+ describe: "log level",
101
+ type: "string",
102
+ choices: ["DEBUG", "INFO", "WARN", "ERROR"],
103
+ })
104
+ .middleware(async (opts) => {
105
+ await Log.init({
106
+ print: process.argv.includes("--print-logs"),
107
+ dev: Installation.isLocal(),
108
+ level: (() => {
109
+ if (opts.logLevel) return opts.logLevel as Log.Level
110
+ if (Installation.isLocal()) return "DEBUG"
111
+ return "INFO"
112
+ })(),
113
+ })
114
+
115
+ process.env.AGENT = "1"
116
+ process.env.BINCODE = "1"
117
+
118
+ Log.Default.info("bincode", {
119
+ version: Installation.VERSION,
120
+ args: process.argv.slice(2),
121
+ })
122
+
123
+ // Check authentication before running commands (except auth and login commands)
124
+ const command = process.argv[2]
125
+ const authStatus = await checkAuth(command)
126
+
127
+ if (authStatus.needsLogin) {
128
+ UI.error("Authentication required")
129
+ console.error("")
130
+ console.error("Please login first by running:")
131
+ console.error(" bincode login")
132
+ console.error("")
133
+ process.exit(1)
134
+ }
135
+ })
136
+ .usage("\n" + UI.logo())
137
+ .command(LoginCommand)
138
+ .command(LogoutCommand)
139
+ .command(AcpCommand)
140
+ .command(McpCommand)
141
+ .command(TuiThreadCommand)
142
+ .command(TuiSpawnCommand)
143
+ .command(AttachCommand)
144
+ .command(RunCommand)
145
+ .command(GenerateCommand)
146
+ .command(DebugCommand)
147
+ .command(AuthCommand)
148
+ .command(AgentCommand)
149
+ .command(UpgradeCommand)
150
+ .command(UninstallCommand)
151
+ .command(ServeCommand)
152
+ .command(WebCommand)
153
+ .command(ModelsCommand)
154
+ .command(StatsCommand)
155
+ .command(ExportCommand)
156
+ .command(ImportCommand)
157
+ .command(GithubCommand)
158
+ .command(PrCommand)
159
+ .command(SessionCommand)
160
+ .fail((msg) => {
161
+ if (
162
+ msg.startsWith("Unknown argument") ||
163
+ msg.startsWith("Not enough non-option arguments") ||
164
+ msg.startsWith("Invalid values:")
165
+ ) {
166
+ cli.showHelp("log")
167
+ }
168
+ process.exit(1)
169
+ })
170
+ .strict()
171
+
172
+ try {
173
+ await cli.parse()
174
+ } catch (e) {
175
+ let data: Record<string, any> = {}
176
+ if (e instanceof NamedError) {
177
+ const obj = e.toObject()
178
+ Object.assign(data, {
179
+ ...obj.data,
180
+ })
181
+ }
182
+
183
+ if (e instanceof Error) {
184
+ Object.assign(data, {
185
+ name: e.name,
186
+ message: e.message,
187
+ cause: e.cause?.toString(),
188
+ stack: e.stack,
189
+ })
190
+ }
191
+
192
+ if (e instanceof ResolveMessage) {
193
+ Object.assign(data, {
194
+ name: e.name,
195
+ message: e.message,
196
+ code: e.code,
197
+ specifier: e.specifier,
198
+ referrer: e.referrer,
199
+ position: e.position,
200
+ importKind: e.importKind,
201
+ })
202
+ }
203
+ Log.Default.error("fatal", data)
204
+ const formatted = FormatError(e)
205
+ if (formatted) UI.error(formatted)
206
+ if (formatted === undefined) {
207
+ UI.error("Unexpected error, check log file at " + Log.file() + " for more details" + EOL)
208
+ console.error(e)
209
+ }
210
+ process.exitCode = 1
211
+ } finally {
212
+ // Some subprocesses don't react properly to SIGTERM and similar signals.
213
+ // Most notably, some docker-container-based MCP servers don't handle such signals unless
214
+ // run using `docker run --init`.
215
+ // Explicitly exit to avoid any hanging subprocesses.
216
+ process.exit()
217
+ }
@@ -0,0 +1,196 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import path from "path"
3
+ import { $ } from "bun"
4
+ import z from "zod"
5
+ import { NamedError } from "@bincode-ai/util/error"
6
+ import { Log } from "../util/log"
7
+ import { iife } from "@/util/iife"
8
+ import { Flag } from "../flag/flag"
9
+
10
+ declare global {
11
+ const BINCODE_VERSION: string
12
+ const BINCODE_CHANNEL: string
13
+ }
14
+
15
+ export namespace Installation {
16
+ const log = Log.create({ service: "installation" })
17
+
18
+ export type Method = Awaited<ReturnType<typeof method>>
19
+
20
+ export const Event = {
21
+ Updated: BusEvent.define(
22
+ "installation.updated",
23
+ z.object({
24
+ version: z.string(),
25
+ }),
26
+ ),
27
+ UpdateAvailable: BusEvent.define(
28
+ "installation.update-available",
29
+ z.object({
30
+ version: z.string(),
31
+ }),
32
+ ),
33
+ }
34
+
35
+ export const Info = z
36
+ .object({
37
+ version: z.string(),
38
+ latest: z.string(),
39
+ })
40
+ .meta({
41
+ ref: "InstallationInfo",
42
+ })
43
+ export type Info = z.infer<typeof Info>
44
+
45
+ export async function info() {
46
+ return {
47
+ version: VERSION,
48
+ latest: await latest(),
49
+ }
50
+ }
51
+
52
+ export function isPreview() {
53
+ return CHANNEL !== "latest"
54
+ }
55
+
56
+ export function isLocal() {
57
+ return CHANNEL === "local"
58
+ }
59
+
60
+ export async function method() {
61
+ if (process.execPath.includes(path.join(".bincode", "bin"))) return "curl"
62
+ if (process.execPath.includes(path.join(".local", "bin"))) return "curl"
63
+ const exec = process.execPath.toLowerCase()
64
+
65
+ const checks = [
66
+ {
67
+ name: "npm" as const,
68
+ command: () => $`npm list -g --depth=0`.throws(false).quiet().text(),
69
+ },
70
+ {
71
+ name: "yarn" as const,
72
+ command: () => $`yarn global list`.throws(false).quiet().text(),
73
+ },
74
+ {
75
+ name: "pnpm" as const,
76
+ command: () => $`pnpm list -g --depth=0`.throws(false).quiet().text(),
77
+ },
78
+ {
79
+ name: "bun" as const,
80
+ command: () => $`bun pm ls -g`.throws(false).quiet().text(),
81
+ },
82
+ {
83
+ name: "brew" as const,
84
+ command: () => $`brew list --formula bincode`.throws(false).quiet().text(),
85
+ },
86
+ ]
87
+
88
+ checks.sort((a, b) => {
89
+ const aMatches = exec.includes(a.name)
90
+ const bMatches = exec.includes(b.name)
91
+ if (aMatches && !bMatches) return -1
92
+ if (!aMatches && bMatches) return 1
93
+ return 0
94
+ })
95
+
96
+ for (const check of checks) {
97
+ const output = await check.command()
98
+ if (output.includes(check.name === "brew" ? "bincode" : "bincode-ai")) {
99
+ return check.name
100
+ }
101
+ }
102
+
103
+ return "unknown"
104
+ }
105
+
106
+ export const UpgradeFailedError = NamedError.create(
107
+ "UpgradeFailedError",
108
+ z.object({
109
+ stderr: z.string(),
110
+ }),
111
+ )
112
+
113
+ async function getBrewFormula() {
114
+ const tapFormula = await $`brew list --formula sst/tap/bincode`.throws(false).quiet().text()
115
+ if (tapFormula.includes("bincode")) return "sst/tap/bincode"
116
+ const coreFormula = await $`brew list --formula bincode`.throws(false).quiet().text()
117
+ if (coreFormula.includes("bincode")) return "bincode"
118
+ return "bincode"
119
+ }
120
+
121
+ export async function upgrade(method: Method, target: string) {
122
+ let cmd
123
+ switch (method) {
124
+ case "curl":
125
+ cmd = $`curl -fsSL https://bincode.ai/install | bash`.env({
126
+ ...process.env,
127
+ VERSION: target,
128
+ })
129
+ break
130
+ case "npm":
131
+ cmd = $`npm install -g bincode-ai@${target}`
132
+ break
133
+ case "pnpm":
134
+ cmd = $`pnpm install -g bincode-ai@${target}`
135
+ break
136
+ case "bun":
137
+ cmd = $`bun install -g bincode-ai@${target}`
138
+ break
139
+ case "brew": {
140
+ const formula = await getBrewFormula()
141
+ cmd = $`brew install ${formula}`.env({
142
+ HOMEBREW_NO_AUTO_UPDATE: "1",
143
+ ...process.env,
144
+ })
145
+ break
146
+ }
147
+ default:
148
+ throw new Error(`Unknown method: ${method}`)
149
+ }
150
+ const result = await cmd.quiet().throws(false)
151
+ log.info("upgraded", {
152
+ method,
153
+ target,
154
+ stdout: result.stdout.toString(),
155
+ stderr: result.stderr.toString(),
156
+ })
157
+ if (result.exitCode !== 0)
158
+ throw new UpgradeFailedError({
159
+ stderr: result.stderr.toString("utf8"),
160
+ })
161
+ }
162
+
163
+ export const VERSION = typeof BINCODE_VERSION === "string" ? BINCODE_VERSION : "local"
164
+ export const CHANNEL = typeof BINCODE_CHANNEL === "string" ? BINCODE_CHANNEL : "local"
165
+ export const USER_AGENT = `bincode/${CHANNEL}/${VERSION}/${Flag.BINCODE_CLIENT}`
166
+
167
+ export async function latest(installMethod?: Method) {
168
+ const detectedMethod = installMethod || (await method())
169
+ if (detectedMethod === "brew") {
170
+ const formula = await getBrewFormula()
171
+ if (formula === "bincode") {
172
+ return fetch("https://formulae.brew.sh/api/formula/bincode.json")
173
+ .then((res) => {
174
+ if (!res.ok) throw new Error(res.statusText)
175
+ return res.json()
176
+ })
177
+ .then((data: any) => data.versions.stable)
178
+ }
179
+ }
180
+
181
+ const registry = await iife(async () => {
182
+ const r = (await $`npm config get registry`.quiet().nothrow().text()).trim()
183
+ const reg = r || "https://registry.npmjs.org"
184
+ return reg.endsWith("/") ? reg.slice(0, -1) : reg
185
+ })
186
+ const [major] = VERSION.split(".").map((x) => Number(x))
187
+ // const channel = CHANNEL === "latest" ? `latest-${major}` : CHANNEL
188
+ const channel = CHANNEL
189
+ return fetch(`${registry}/bincode-ai/${channel}`)
190
+ .then((res) => {
191
+ if (!res.ok) throw new Error(res.statusText)
192
+ return res.json()
193
+ })
194
+ .then((data: any) => data.version)
195
+ }
196
+ }
@@ -0,0 +1,229 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import { Bus } from "@/bus"
3
+ import path from "path"
4
+ import { pathToFileURL, fileURLToPath } from "url"
5
+ import { createMessageConnection, StreamMessageReader, StreamMessageWriter } from "vscode-jsonrpc/node"
6
+ import type { Diagnostic as VSCodeDiagnostic } from "vscode-languageserver-types"
7
+ import { Log } from "../util/log"
8
+ import { LANGUAGE_EXTENSIONS } from "./language"
9
+ import z from "zod"
10
+ import type { LSPServer } from "./server"
11
+ import { NamedError } from "@bincode-ai/util/error"
12
+ import { withTimeout } from "../util/timeout"
13
+ import { Instance } from "../project/instance"
14
+ import { Filesystem } from "../util/filesystem"
15
+
16
+ const DIAGNOSTICS_DEBOUNCE_MS = 150
17
+
18
+ export namespace LSPClient {
19
+ const log = Log.create({ service: "lsp.client" })
20
+
21
+ export type Info = NonNullable<Awaited<ReturnType<typeof create>>>
22
+
23
+ export type Diagnostic = VSCodeDiagnostic
24
+
25
+ export const InitializeError = NamedError.create(
26
+ "LSPInitializeError",
27
+ z.object({
28
+ serverID: z.string(),
29
+ }),
30
+ )
31
+
32
+ export const Event = {
33
+ Diagnostics: BusEvent.define(
34
+ "lsp.client.diagnostics",
35
+ z.object({
36
+ serverID: z.string(),
37
+ path: z.string(),
38
+ }),
39
+ ),
40
+ }
41
+
42
+ export async function create(input: { serverID: string; server: LSPServer.Handle; root: string }) {
43
+ const l = log.clone().tag("serverID", input.serverID)
44
+ l.info("starting client")
45
+
46
+ const connection = createMessageConnection(
47
+ new StreamMessageReader(input.server.process.stdout as any),
48
+ new StreamMessageWriter(input.server.process.stdin as any),
49
+ )
50
+
51
+ const diagnostics = new Map<string, Diagnostic[]>()
52
+ connection.onNotification("textDocument/publishDiagnostics", (params) => {
53
+ const filePath = Filesystem.normalizePath(fileURLToPath(params.uri))
54
+ l.info("textDocument/publishDiagnostics", {
55
+ path: filePath,
56
+ count: params.diagnostics.length,
57
+ })
58
+ const exists = diagnostics.has(filePath)
59
+ diagnostics.set(filePath, params.diagnostics)
60
+ if (!exists && input.serverID === "typescript") return
61
+ Bus.publish(Event.Diagnostics, { path: filePath, serverID: input.serverID })
62
+ })
63
+ connection.onRequest("window/workDoneProgress/create", (params) => {
64
+ l.info("window/workDoneProgress/create", params)
65
+ return null
66
+ })
67
+ connection.onRequest("workspace/configuration", async () => {
68
+ // Return server initialization options
69
+ return [input.server.initialization ?? {}]
70
+ })
71
+ connection.onRequest("client/registerCapability", async () => {})
72
+ connection.onRequest("client/unregisterCapability", async () => {})
73
+ connection.onRequest("workspace/workspaceFolders", async () => [
74
+ {
75
+ name: "workspace",
76
+ uri: pathToFileURL(input.root).href,
77
+ },
78
+ ])
79
+ connection.listen()
80
+
81
+ l.info("sending initialize")
82
+ await withTimeout(
83
+ connection.sendRequest("initialize", {
84
+ rootUri: pathToFileURL(input.root).href,
85
+ processId: input.server.process.pid,
86
+ workspaceFolders: [
87
+ {
88
+ name: "workspace",
89
+ uri: pathToFileURL(input.root).href,
90
+ },
91
+ ],
92
+ initializationOptions: {
93
+ ...input.server.initialization,
94
+ },
95
+ capabilities: {
96
+ window: {
97
+ workDoneProgress: true,
98
+ },
99
+ workspace: {
100
+ configuration: true,
101
+ },
102
+ textDocument: {
103
+ synchronization: {
104
+ didOpen: true,
105
+ didChange: true,
106
+ },
107
+ publishDiagnostics: {
108
+ versionSupport: true,
109
+ },
110
+ },
111
+ },
112
+ }),
113
+ 45_000,
114
+ ).catch((err) => {
115
+ l.error("initialize error", { error: err })
116
+ throw new InitializeError(
117
+ { serverID: input.serverID },
118
+ {
119
+ cause: err,
120
+ },
121
+ )
122
+ })
123
+
124
+ await connection.sendNotification("initialized", {})
125
+
126
+ if (input.server.initialization) {
127
+ await connection.sendNotification("workspace/didChangeConfiguration", {
128
+ settings: input.server.initialization,
129
+ })
130
+ }
131
+
132
+ const files: {
133
+ [path: string]: number
134
+ } = {}
135
+
136
+ const result = {
137
+ root: input.root,
138
+ get serverID() {
139
+ return input.serverID
140
+ },
141
+ get connection() {
142
+ return connection
143
+ },
144
+ notify: {
145
+ async open(input: { path: string }) {
146
+ input.path = path.isAbsolute(input.path) ? input.path : path.resolve(Instance.directory, input.path)
147
+ const file = Bun.file(input.path)
148
+ const text = await file.text()
149
+ const extension = path.extname(input.path)
150
+ const languageId = LANGUAGE_EXTENSIONS[extension] ?? "plaintext"
151
+
152
+ const version = files[input.path]
153
+ if (version !== undefined) {
154
+ const next = version + 1
155
+ files[input.path] = next
156
+ log.info("textDocument/didChange", {
157
+ path: input.path,
158
+ version: next,
159
+ })
160
+ await connection.sendNotification("textDocument/didChange", {
161
+ textDocument: {
162
+ uri: pathToFileURL(input.path).href,
163
+ version: next,
164
+ },
165
+ contentChanges: [{ text }],
166
+ })
167
+ return
168
+ }
169
+
170
+ log.info("textDocument/didOpen", input)
171
+ diagnostics.delete(input.path)
172
+ await connection.sendNotification("textDocument/didOpen", {
173
+ textDocument: {
174
+ uri: pathToFileURL(input.path).href,
175
+ languageId,
176
+ version: 0,
177
+ text,
178
+ },
179
+ })
180
+ files[input.path] = 0
181
+ return
182
+ },
183
+ },
184
+ get diagnostics() {
185
+ return diagnostics
186
+ },
187
+ async waitForDiagnostics(input: { path: string }) {
188
+ const normalizedPath = Filesystem.normalizePath(
189
+ path.isAbsolute(input.path) ? input.path : path.resolve(Instance.directory, input.path),
190
+ )
191
+ log.info("waiting for diagnostics", { path: normalizedPath })
192
+ let unsub: () => void
193
+ let debounceTimer: ReturnType<typeof setTimeout> | undefined
194
+ return await withTimeout(
195
+ new Promise<void>((resolve) => {
196
+ unsub = Bus.subscribe(Event.Diagnostics, (event) => {
197
+ if (event.properties.path === normalizedPath && event.properties.serverID === result.serverID) {
198
+ // Debounce to allow LSP to send follow-up diagnostics (e.g., semantic after syntax)
199
+ if (debounceTimer) clearTimeout(debounceTimer)
200
+ debounceTimer = setTimeout(() => {
201
+ log.info("got diagnostics", { path: normalizedPath })
202
+ unsub?.()
203
+ resolve()
204
+ }, DIAGNOSTICS_DEBOUNCE_MS)
205
+ }
206
+ })
207
+ }),
208
+ 3000,
209
+ )
210
+ .catch(() => {})
211
+ .finally(() => {
212
+ if (debounceTimer) clearTimeout(debounceTimer)
213
+ unsub?.()
214
+ })
215
+ },
216
+ async shutdown() {
217
+ l.info("shutting down")
218
+ connection.end()
219
+ connection.dispose()
220
+ input.server.process.kill()
221
+ l.info("shutdown")
222
+ },
223
+ }
224
+
225
+ l.info("initialized")
226
+
227
+ return result
228
+ }
229
+ }