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,485 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import { Bus } from "@/bus"
3
+ import { Log } from "../util/log"
4
+ import { LSPClient } from "./client"
5
+ import path from "path"
6
+ import { pathToFileURL } from "url"
7
+ import { LSPServer } from "./server"
8
+ import z from "zod"
9
+ import { Config } from "../config/config"
10
+ import { spawn } from "child_process"
11
+ import { Instance } from "../project/instance"
12
+ import { Flag } from "@/flag/flag"
13
+
14
+ export namespace LSP {
15
+ const log = Log.create({ service: "lsp" })
16
+
17
+ export const Event = {
18
+ Updated: BusEvent.define("lsp.updated", z.object({})),
19
+ }
20
+
21
+ export const Range = z
22
+ .object({
23
+ start: z.object({
24
+ line: z.number(),
25
+ character: z.number(),
26
+ }),
27
+ end: z.object({
28
+ line: z.number(),
29
+ character: z.number(),
30
+ }),
31
+ })
32
+ .meta({
33
+ ref: "Range",
34
+ })
35
+ export type Range = z.infer<typeof Range>
36
+
37
+ export const Symbol = z
38
+ .object({
39
+ name: z.string(),
40
+ kind: z.number(),
41
+ location: z.object({
42
+ uri: z.string(),
43
+ range: Range,
44
+ }),
45
+ })
46
+ .meta({
47
+ ref: "Symbol",
48
+ })
49
+ export type Symbol = z.infer<typeof Symbol>
50
+
51
+ export const DocumentSymbol = z
52
+ .object({
53
+ name: z.string(),
54
+ detail: z.string().optional(),
55
+ kind: z.number(),
56
+ range: Range,
57
+ selectionRange: Range,
58
+ })
59
+ .meta({
60
+ ref: "DocumentSymbol",
61
+ })
62
+ export type DocumentSymbol = z.infer<typeof DocumentSymbol>
63
+
64
+ const filterExperimentalServers = (servers: Record<string, LSPServer.Info>) => {
65
+ if (Flag.BINCODE_EXPERIMENTAL_LSP_TY) {
66
+ // If experimental flag is enabled, disable pyright
67
+ if (servers["pyright"]) {
68
+ log.info("LSP server pyright is disabled because BINCODE_EXPERIMENTAL_LSP_TY is enabled")
69
+ delete servers["pyright"]
70
+ }
71
+ } else {
72
+ // If experimental flag is disabled, disable ty
73
+ if (servers["ty"]) {
74
+ delete servers["ty"]
75
+ }
76
+ }
77
+ }
78
+
79
+ const state = Instance.state(
80
+ async () => {
81
+ const clients: LSPClient.Info[] = []
82
+ const servers: Record<string, LSPServer.Info> = {}
83
+ const cfg = await Config.get()
84
+
85
+ if (cfg.lsp === false) {
86
+ log.info("all LSPs are disabled")
87
+ return {
88
+ broken: new Set<string>(),
89
+ servers,
90
+ clients,
91
+ spawning: new Map<string, Promise<LSPClient.Info | undefined>>(),
92
+ }
93
+ }
94
+
95
+ for (const server of Object.values(LSPServer)) {
96
+ servers[server.id] = server
97
+ }
98
+
99
+ filterExperimentalServers(servers)
100
+
101
+ for (const [name, item] of Object.entries(cfg.lsp ?? {})) {
102
+ const existing = servers[name]
103
+ if (item.disabled) {
104
+ log.info(`LSP server ${name} is disabled`)
105
+ delete servers[name]
106
+ continue
107
+ }
108
+ servers[name] = {
109
+ ...existing,
110
+ id: name,
111
+ root: existing?.root ?? (async () => Instance.directory),
112
+ extensions: item.extensions ?? existing?.extensions ?? [],
113
+ spawn: async (root) => {
114
+ return {
115
+ process: spawn(item.command[0], item.command.slice(1), {
116
+ cwd: root,
117
+ env: {
118
+ ...process.env,
119
+ ...item.env,
120
+ },
121
+ }),
122
+ initialization: item.initialization,
123
+ }
124
+ },
125
+ }
126
+ }
127
+
128
+ log.info("enabled LSP servers", {
129
+ serverIds: Object.values(servers)
130
+ .map((server) => server.id)
131
+ .join(", "),
132
+ })
133
+
134
+ return {
135
+ broken: new Set<string>(),
136
+ servers,
137
+ clients,
138
+ spawning: new Map<string, Promise<LSPClient.Info | undefined>>(),
139
+ }
140
+ },
141
+ async (state) => {
142
+ await Promise.all(state.clients.map((client) => client.shutdown()))
143
+ },
144
+ )
145
+
146
+ export async function init() {
147
+ return state()
148
+ }
149
+
150
+ export const Status = z
151
+ .object({
152
+ id: z.string(),
153
+ name: z.string(),
154
+ root: z.string(),
155
+ status: z.union([z.literal("connected"), z.literal("error")]),
156
+ })
157
+ .meta({
158
+ ref: "LSPStatus",
159
+ })
160
+ export type Status = z.infer<typeof Status>
161
+
162
+ export async function status() {
163
+ return state().then((x) => {
164
+ const result: Status[] = []
165
+ for (const client of x.clients) {
166
+ result.push({
167
+ id: client.serverID,
168
+ name: x.servers[client.serverID].id,
169
+ root: path.relative(Instance.directory, client.root),
170
+ status: "connected",
171
+ })
172
+ }
173
+ return result
174
+ })
175
+ }
176
+
177
+ async function getClients(file: string) {
178
+ const s = await state()
179
+ const extension = path.parse(file).ext || file
180
+ const result: LSPClient.Info[] = []
181
+
182
+ async function schedule(server: LSPServer.Info, root: string, key: string) {
183
+ const handle = await server
184
+ .spawn(root)
185
+ .then((value) => {
186
+ if (!value) s.broken.add(key)
187
+ return value
188
+ })
189
+ .catch((err) => {
190
+ s.broken.add(key)
191
+ log.error(`Failed to spawn LSP server ${server.id}`, { error: err })
192
+ return undefined
193
+ })
194
+
195
+ if (!handle) return undefined
196
+ log.info("spawned lsp server", { serverID: server.id })
197
+
198
+ const client = await LSPClient.create({
199
+ serverID: server.id,
200
+ server: handle,
201
+ root,
202
+ }).catch((err) => {
203
+ s.broken.add(key)
204
+ handle.process.kill()
205
+ log.error(`Failed to initialize LSP client ${server.id}`, { error: err })
206
+ return undefined
207
+ })
208
+
209
+ if (!client) {
210
+ handle.process.kill()
211
+ return undefined
212
+ }
213
+
214
+ const existing = s.clients.find((x) => x.root === root && x.serverID === server.id)
215
+ if (existing) {
216
+ handle.process.kill()
217
+ return existing
218
+ }
219
+
220
+ s.clients.push(client)
221
+ return client
222
+ }
223
+
224
+ for (const server of Object.values(s.servers)) {
225
+ if (server.extensions.length && !server.extensions.includes(extension)) continue
226
+
227
+ const root = await server.root(file)
228
+ if (!root) continue
229
+ if (s.broken.has(root + server.id)) continue
230
+
231
+ const match = s.clients.find((x) => x.root === root && x.serverID === server.id)
232
+ if (match) {
233
+ result.push(match)
234
+ continue
235
+ }
236
+
237
+ const inflight = s.spawning.get(root + server.id)
238
+ if (inflight) {
239
+ const client = await inflight
240
+ if (!client) continue
241
+ result.push(client)
242
+ continue
243
+ }
244
+
245
+ const task = schedule(server, root, root + server.id)
246
+ s.spawning.set(root + server.id, task)
247
+
248
+ task.finally(() => {
249
+ if (s.spawning.get(root + server.id) === task) {
250
+ s.spawning.delete(root + server.id)
251
+ }
252
+ })
253
+
254
+ const client = await task
255
+ if (!client) continue
256
+
257
+ result.push(client)
258
+ Bus.publish(Event.Updated, {})
259
+ }
260
+
261
+ return result
262
+ }
263
+
264
+ export async function hasClients(file: string) {
265
+ const s = await state()
266
+ const extension = path.parse(file).ext || file
267
+ for (const server of Object.values(s.servers)) {
268
+ if (server.extensions.length && !server.extensions.includes(extension)) continue
269
+ const root = await server.root(file)
270
+ if (!root) continue
271
+ if (s.broken.has(root + server.id)) continue
272
+ return true
273
+ }
274
+ return false
275
+ }
276
+
277
+ export async function touchFile(input: string, waitForDiagnostics?: boolean) {
278
+ log.info("touching file", { file: input })
279
+ const clients = await getClients(input)
280
+ await Promise.all(
281
+ clients.map(async (client) => {
282
+ const wait = waitForDiagnostics ? client.waitForDiagnostics({ path: input }) : Promise.resolve()
283
+ await client.notify.open({ path: input })
284
+ return wait
285
+ }),
286
+ ).catch((err) => {
287
+ log.error("failed to touch file", { err, file: input })
288
+ })
289
+ }
290
+
291
+ export async function diagnostics() {
292
+ const results: Record<string, LSPClient.Diagnostic[]> = {}
293
+ for (const result of await runAll(async (client) => client.diagnostics)) {
294
+ for (const [path, diagnostics] of result.entries()) {
295
+ const arr = results[path] || []
296
+ arr.push(...diagnostics)
297
+ results[path] = arr
298
+ }
299
+ }
300
+ return results
301
+ }
302
+
303
+ export async function hover(input: { file: string; line: number; character: number }) {
304
+ return run(input.file, (client) => {
305
+ return client.connection
306
+ .sendRequest("textDocument/hover", {
307
+ textDocument: {
308
+ uri: pathToFileURL(input.file).href,
309
+ },
310
+ position: {
311
+ line: input.line,
312
+ character: input.character,
313
+ },
314
+ })
315
+ .catch(() => null)
316
+ })
317
+ }
318
+
319
+ enum SymbolKind {
320
+ File = 1,
321
+ Module = 2,
322
+ Namespace = 3,
323
+ Package = 4,
324
+ Class = 5,
325
+ Method = 6,
326
+ Property = 7,
327
+ Field = 8,
328
+ Constructor = 9,
329
+ Enum = 10,
330
+ Interface = 11,
331
+ Function = 12,
332
+ Variable = 13,
333
+ Constant = 14,
334
+ String = 15,
335
+ Number = 16,
336
+ Boolean = 17,
337
+ Array = 18,
338
+ Object = 19,
339
+ Key = 20,
340
+ Null = 21,
341
+ EnumMember = 22,
342
+ Struct = 23,
343
+ Event = 24,
344
+ Operator = 25,
345
+ TypeParameter = 26,
346
+ }
347
+
348
+ const kinds = [
349
+ SymbolKind.Class,
350
+ SymbolKind.Function,
351
+ SymbolKind.Method,
352
+ SymbolKind.Interface,
353
+ SymbolKind.Variable,
354
+ SymbolKind.Constant,
355
+ SymbolKind.Struct,
356
+ SymbolKind.Enum,
357
+ ]
358
+
359
+ export async function workspaceSymbol(query: string) {
360
+ return runAll((client) =>
361
+ client.connection
362
+ .sendRequest("workspace/symbol", {
363
+ query,
364
+ })
365
+ .then((result: any) => result.filter((x: LSP.Symbol) => kinds.includes(x.kind)))
366
+ .then((result: any) => result.slice(0, 10))
367
+ .catch(() => []),
368
+ ).then((result) => result.flat() as LSP.Symbol[])
369
+ }
370
+
371
+ export async function documentSymbol(uri: string) {
372
+ const file = new URL(uri).pathname
373
+ return run(file, (client) =>
374
+ client.connection
375
+ .sendRequest("textDocument/documentSymbol", {
376
+ textDocument: {
377
+ uri,
378
+ },
379
+ })
380
+ .catch(() => []),
381
+ )
382
+ .then((result) => result.flat() as (LSP.DocumentSymbol | LSP.Symbol)[])
383
+ .then((result) => result.filter(Boolean))
384
+ }
385
+
386
+ export async function definition(input: { file: string; line: number; character: number }) {
387
+ return run(input.file, (client) =>
388
+ client.connection
389
+ .sendRequest("textDocument/definition", {
390
+ textDocument: { uri: pathToFileURL(input.file).href },
391
+ position: { line: input.line, character: input.character },
392
+ })
393
+ .catch(() => null),
394
+ ).then((result) => result.flat().filter(Boolean))
395
+ }
396
+
397
+ export async function references(input: { file: string; line: number; character: number }) {
398
+ return run(input.file, (client) =>
399
+ client.connection
400
+ .sendRequest("textDocument/references", {
401
+ textDocument: { uri: pathToFileURL(input.file).href },
402
+ position: { line: input.line, character: input.character },
403
+ context: { includeDeclaration: true },
404
+ })
405
+ .catch(() => []),
406
+ ).then((result) => result.flat().filter(Boolean))
407
+ }
408
+
409
+ export async function implementation(input: { file: string; line: number; character: number }) {
410
+ return run(input.file, (client) =>
411
+ client.connection
412
+ .sendRequest("textDocument/implementation", {
413
+ textDocument: { uri: pathToFileURL(input.file).href },
414
+ position: { line: input.line, character: input.character },
415
+ })
416
+ .catch(() => null),
417
+ ).then((result) => result.flat().filter(Boolean))
418
+ }
419
+
420
+ export async function prepareCallHierarchy(input: { file: string; line: number; character: number }) {
421
+ return run(input.file, (client) =>
422
+ client.connection
423
+ .sendRequest("textDocument/prepareCallHierarchy", {
424
+ textDocument: { uri: pathToFileURL(input.file).href },
425
+ position: { line: input.line, character: input.character },
426
+ })
427
+ .catch(() => []),
428
+ ).then((result) => result.flat().filter(Boolean))
429
+ }
430
+
431
+ export async function incomingCalls(input: { file: string; line: number; character: number }) {
432
+ return run(input.file, async (client) => {
433
+ const items = (await client.connection
434
+ .sendRequest("textDocument/prepareCallHierarchy", {
435
+ textDocument: { uri: pathToFileURL(input.file).href },
436
+ position: { line: input.line, character: input.character },
437
+ })
438
+ .catch(() => [])) as any[]
439
+ if (!items?.length) return []
440
+ return client.connection.sendRequest("callHierarchy/incomingCalls", { item: items[0] }).catch(() => [])
441
+ }).then((result) => result.flat().filter(Boolean))
442
+ }
443
+
444
+ export async function outgoingCalls(input: { file: string; line: number; character: number }) {
445
+ return run(input.file, async (client) => {
446
+ const items = (await client.connection
447
+ .sendRequest("textDocument/prepareCallHierarchy", {
448
+ textDocument: { uri: pathToFileURL(input.file).href },
449
+ position: { line: input.line, character: input.character },
450
+ })
451
+ .catch(() => [])) as any[]
452
+ if (!items?.length) return []
453
+ return client.connection.sendRequest("callHierarchy/outgoingCalls", { item: items[0] }).catch(() => [])
454
+ }).then((result) => result.flat().filter(Boolean))
455
+ }
456
+
457
+ async function runAll<T>(input: (client: LSPClient.Info) => Promise<T>): Promise<T[]> {
458
+ const clients = await state().then((x) => x.clients)
459
+ const tasks = clients.map((x) => input(x))
460
+ return Promise.all(tasks)
461
+ }
462
+
463
+ async function run<T>(file: string, input: (client: LSPClient.Info) => Promise<T>): Promise<T[]> {
464
+ const clients = await getClients(file)
465
+ const tasks = clients.map((x) => input(x))
466
+ return Promise.all(tasks)
467
+ }
468
+
469
+ export namespace Diagnostic {
470
+ export function pretty(diagnostic: LSPClient.Diagnostic) {
471
+ const severityMap = {
472
+ 1: "ERROR",
473
+ 2: "WARN",
474
+ 3: "INFO",
475
+ 4: "HINT",
476
+ }
477
+
478
+ const severity = severityMap[diagnostic.severity || 1]
479
+ const line = diagnostic.range.start.line + 1
480
+ const col = diagnostic.range.start.character + 1
481
+
482
+ return `${severity} [${line}:${col}] ${diagnostic.message}`
483
+ }
484
+ }
485
+ }
@@ -0,0 +1,116 @@
1
+ export const LANGUAGE_EXTENSIONS: Record<string, string> = {
2
+ ".abap": "abap",
3
+ ".bat": "bat",
4
+ ".bib": "bibtex",
5
+ ".bibtex": "bibtex",
6
+ ".clj": "clojure",
7
+ ".cljs": "clojure",
8
+ ".cljc": "clojure",
9
+ ".edn": "clojure",
10
+ ".coffee": "coffeescript",
11
+ ".c": "c",
12
+ ".cpp": "cpp",
13
+ ".cxx": "cpp",
14
+ ".cc": "cpp",
15
+ ".c++": "cpp",
16
+ ".cs": "csharp",
17
+ ".css": "css",
18
+ ".d": "d",
19
+ ".pas": "pascal",
20
+ ".pascal": "pascal",
21
+ ".diff": "diff",
22
+ ".patch": "diff",
23
+ ".dart": "dart",
24
+ ".dockerfile": "dockerfile",
25
+ ".ex": "elixir",
26
+ ".exs": "elixir",
27
+ ".erl": "erlang",
28
+ ".ets": "typescript",
29
+ ".hrl": "erlang",
30
+ ".fs": "fsharp",
31
+ ".fsi": "fsharp",
32
+ ".fsx": "fsharp",
33
+ ".fsscript": "fsharp",
34
+ ".gitcommit": "git-commit",
35
+ ".gitrebase": "git-rebase",
36
+ ".go": "go",
37
+ ".groovy": "groovy",
38
+ ".gleam": "gleam",
39
+ ".hbs": "handlebars",
40
+ ".handlebars": "handlebars",
41
+ ".hs": "haskell",
42
+ ".html": "html",
43
+ ".htm": "html",
44
+ ".ini": "ini",
45
+ ".java": "java",
46
+ ".js": "javascript",
47
+ ".jsx": "javascriptreact",
48
+ ".json": "json",
49
+ ".tex": "latex",
50
+ ".latex": "latex",
51
+ ".less": "less",
52
+ ".lua": "lua",
53
+ ".makefile": "makefile",
54
+ makefile: "makefile",
55
+ ".md": "markdown",
56
+ ".markdown": "markdown",
57
+ ".m": "objective-c",
58
+ ".mm": "objective-cpp",
59
+ ".pl": "perl",
60
+ ".pm": "perl",
61
+ ".pm6": "perl6",
62
+ ".php": "php",
63
+ ".ps1": "powershell",
64
+ ".psm1": "powershell",
65
+ ".pug": "jade",
66
+ ".jade": "jade",
67
+ ".py": "python",
68
+ ".r": "r",
69
+ ".cshtml": "razor",
70
+ ".razor": "razor",
71
+ ".rb": "ruby",
72
+ ".rake": "ruby",
73
+ ".gemspec": "ruby",
74
+ ".ru": "ruby",
75
+ ".erb": "erb",
76
+ ".html.erb": "erb",
77
+ ".js.erb": "erb",
78
+ ".css.erb": "erb",
79
+ ".json.erb": "erb",
80
+ ".rs": "rust",
81
+ ".scss": "scss",
82
+ ".sass": "sass",
83
+ ".scala": "scala",
84
+ ".shader": "shaderlab",
85
+ ".sh": "shellscript",
86
+ ".bash": "shellscript",
87
+ ".zsh": "shellscript",
88
+ ".ksh": "shellscript",
89
+ ".sql": "sql",
90
+ ".svelte": "svelte",
91
+ ".swift": "swift",
92
+ ".ts": "typescript",
93
+ ".tsx": "typescriptreact",
94
+ ".mts": "typescript",
95
+ ".cts": "typescript",
96
+ ".mtsx": "typescriptreact",
97
+ ".ctsx": "typescriptreact",
98
+ ".xml": "xml",
99
+ ".xsl": "xsl",
100
+ ".yaml": "yaml",
101
+ ".yml": "yaml",
102
+ ".mjs": "javascript",
103
+ ".cjs": "javascript",
104
+ ".vue": "vue",
105
+ ".zig": "zig",
106
+ ".zon": "zig",
107
+ ".astro": "astro",
108
+ ".ml": "ocaml",
109
+ ".mli": "ocaml",
110
+ ".tf": "terraform",
111
+ ".tfvars": "terraform-vars",
112
+ ".hcl": "hcl",
113
+ ".nix": "nix",
114
+ ".typ": "typst",
115
+ ".typc": "typst",
116
+ } as const