@nghyane/arcane 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (738) hide show
  1. package/CHANGELOG.md +3 -0
  2. package/README.md +12 -0
  3. package/examples/README.md +21 -0
  4. package/examples/custom-tools/README.md +109 -0
  5. package/examples/custom-tools/hello/index.ts +20 -0
  6. package/examples/custom-tools/todo/index.ts +206 -0
  7. package/examples/extensions/README.md +143 -0
  8. package/examples/extensions/api-demo.ts +89 -0
  9. package/examples/extensions/chalk-logger.ts +25 -0
  10. package/examples/extensions/hello.ts +32 -0
  11. package/examples/extensions/pirate.ts +43 -0
  12. package/examples/extensions/plan-mode.ts +550 -0
  13. package/examples/extensions/reload-runtime.ts +37 -0
  14. package/examples/extensions/todo.ts +296 -0
  15. package/examples/extensions/tools.ts +144 -0
  16. package/examples/extensions/with-deps/index.ts +35 -0
  17. package/examples/extensions/with-deps/package-lock.json +31 -0
  18. package/examples/extensions/with-deps/package.json +16 -0
  19. package/examples/hooks/README.md +56 -0
  20. package/examples/hooks/auto-commit-on-exit.ts +48 -0
  21. package/examples/hooks/confirm-destructive.ts +58 -0
  22. package/examples/hooks/custom-compaction.ts +116 -0
  23. package/examples/hooks/dirty-repo-guard.ts +51 -0
  24. package/examples/hooks/file-trigger.ts +40 -0
  25. package/examples/hooks/git-checkpoint.ts +52 -0
  26. package/examples/hooks/handoff.ts +150 -0
  27. package/examples/hooks/permission-gate.ts +33 -0
  28. package/examples/hooks/protected-paths.ts +29 -0
  29. package/examples/hooks/qna.ts +119 -0
  30. package/examples/hooks/status-line.ts +39 -0
  31. package/examples/sdk/01-minimal.ts +21 -0
  32. package/examples/sdk/02-custom-model.ts +49 -0
  33. package/examples/sdk/03-custom-prompt.ts +43 -0
  34. package/examples/sdk/04-skills.ts +43 -0
  35. package/examples/sdk/06-extensions.ts +80 -0
  36. package/examples/sdk/06-hooks.ts +61 -0
  37. package/examples/sdk/07-context-files.ts +35 -0
  38. package/examples/sdk/08-prompt-templates.ts +36 -0
  39. package/examples/sdk/08-slash-commands.ts +41 -0
  40. package/examples/sdk/09-api-keys-and-oauth.ts +54 -0
  41. package/examples/sdk/11-sessions.ts +47 -0
  42. package/examples/sdk/README.md +150 -0
  43. package/package.json +464 -0
  44. package/scripts/format-prompts.ts +184 -0
  45. package/scripts/generate-docs-index.ts +40 -0
  46. package/scripts/generate-template.ts +32 -0
  47. package/src/bun-imports.d.ts +22 -0
  48. package/src/capability/context-file.ts +39 -0
  49. package/src/capability/extension-module.ts +33 -0
  50. package/src/capability/extension.ts +47 -0
  51. package/src/capability/fs.ts +89 -0
  52. package/src/capability/hook.ts +39 -0
  53. package/src/capability/index.ts +432 -0
  54. package/src/capability/instruction.ts +36 -0
  55. package/src/capability/mcp.ts +60 -0
  56. package/src/capability/prompt.ts +34 -0
  57. package/src/capability/rule.ts +223 -0
  58. package/src/capability/settings.ts +34 -0
  59. package/src/capability/skill.ts +48 -0
  60. package/src/capability/slash-command.ts +39 -0
  61. package/src/capability/ssh.ts +41 -0
  62. package/src/capability/system-prompt.ts +34 -0
  63. package/src/capability/tool.ts +37 -0
  64. package/src/capability/types.ts +156 -0
  65. package/src/cli/args.ts +259 -0
  66. package/src/cli/config-cli.ts +357 -0
  67. package/src/cli/file-processor.ts +124 -0
  68. package/src/cli/grep-cli.ts +152 -0
  69. package/src/cli/jupyter-cli.ts +106 -0
  70. package/src/cli/list-models.ts +103 -0
  71. package/src/cli/plugin-cli.ts +661 -0
  72. package/src/cli/session-picker.ts +42 -0
  73. package/src/cli/setup-cli.ts +376 -0
  74. package/src/cli/shell-cli.ts +174 -0
  75. package/src/cli/ssh-cli.ts +179 -0
  76. package/src/cli/stats-cli.ts +197 -0
  77. package/src/cli/update-cli.ts +286 -0
  78. package/src/cli/web-search-cli.ts +143 -0
  79. package/src/cli.ts +65 -0
  80. package/src/commands/commit.ts +36 -0
  81. package/src/commands/config.ts +51 -0
  82. package/src/commands/grep.ts +41 -0
  83. package/src/commands/jupyter.ts +32 -0
  84. package/src/commands/launch.ts +139 -0
  85. package/src/commands/plugin.ts +70 -0
  86. package/src/commands/setup.ts +42 -0
  87. package/src/commands/shell.ts +29 -0
  88. package/src/commands/ssh.ts +60 -0
  89. package/src/commands/stats.ts +29 -0
  90. package/src/commands/update.ts +21 -0
  91. package/src/commands/web-search.ts +42 -0
  92. package/src/commit/agentic/agent.ts +311 -0
  93. package/src/commit/agentic/fallback.ts +96 -0
  94. package/src/commit/agentic/index.ts +359 -0
  95. package/src/commit/agentic/prompts/analyze-file.md +22 -0
  96. package/src/commit/agentic/prompts/session-user.md +25 -0
  97. package/src/commit/agentic/prompts/split-confirm.md +1 -0
  98. package/src/commit/agentic/prompts/system.md +38 -0
  99. package/src/commit/agentic/state.ts +69 -0
  100. package/src/commit/agentic/tools/analyze-file.ts +118 -0
  101. package/src/commit/agentic/tools/git-file-diff.ts +194 -0
  102. package/src/commit/agentic/tools/git-hunk.ts +50 -0
  103. package/src/commit/agentic/tools/git-overview.ts +84 -0
  104. package/src/commit/agentic/tools/index.ts +56 -0
  105. package/src/commit/agentic/tools/propose-changelog.ts +128 -0
  106. package/src/commit/agentic/tools/propose-commit.ts +154 -0
  107. package/src/commit/agentic/tools/recent-commits.ts +81 -0
  108. package/src/commit/agentic/tools/split-commit.ts +280 -0
  109. package/src/commit/agentic/topo-sort.ts +44 -0
  110. package/src/commit/agentic/trivial.ts +51 -0
  111. package/src/commit/agentic/validation.ts +200 -0
  112. package/src/commit/analysis/conventional.ts +165 -0
  113. package/src/commit/analysis/index.ts +4 -0
  114. package/src/commit/analysis/scope.ts +242 -0
  115. package/src/commit/analysis/summary.ts +112 -0
  116. package/src/commit/analysis/validation.ts +66 -0
  117. package/src/commit/changelog/detect.ts +37 -0
  118. package/src/commit/changelog/generate.ts +110 -0
  119. package/src/commit/changelog/index.ts +234 -0
  120. package/src/commit/changelog/parse.ts +44 -0
  121. package/src/commit/cli.ts +93 -0
  122. package/src/commit/git/diff.ts +148 -0
  123. package/src/commit/git/errors.ts +9 -0
  124. package/src/commit/git/index.ts +211 -0
  125. package/src/commit/git/operations.ts +54 -0
  126. package/src/commit/index.ts +5 -0
  127. package/src/commit/map-reduce/index.ts +64 -0
  128. package/src/commit/map-reduce/map-phase.ts +178 -0
  129. package/src/commit/map-reduce/reduce-phase.ts +145 -0
  130. package/src/commit/map-reduce/utils.ts +9 -0
  131. package/src/commit/message.ts +11 -0
  132. package/src/commit/model-selection.ts +69 -0
  133. package/src/commit/pipeline.ts +243 -0
  134. package/src/commit/prompts/analysis-system.md +148 -0
  135. package/src/commit/prompts/analysis-user.md +38 -0
  136. package/src/commit/prompts/changelog-system.md +50 -0
  137. package/src/commit/prompts/changelog-user.md +18 -0
  138. package/src/commit/prompts/file-observer-system.md +24 -0
  139. package/src/commit/prompts/file-observer-user.md +8 -0
  140. package/src/commit/prompts/reduce-system.md +50 -0
  141. package/src/commit/prompts/reduce-user.md +17 -0
  142. package/src/commit/prompts/summary-retry.md +3 -0
  143. package/src/commit/prompts/summary-system.md +38 -0
  144. package/src/commit/prompts/summary-user.md +13 -0
  145. package/src/commit/prompts/types-description.md +2 -0
  146. package/src/commit/types.ts +109 -0
  147. package/src/commit/utils/exclusions.ts +42 -0
  148. package/src/config/file-lock.ts +121 -0
  149. package/src/config/keybindings.ts +280 -0
  150. package/src/config/model-registry.ts +1140 -0
  151. package/src/config/model-resolver.ts +812 -0
  152. package/src/config/prompt-templates.ts +526 -0
  153. package/src/config/resolve-config-value.ts +92 -0
  154. package/src/config/settings-schema.ts +1236 -0
  155. package/src/config/settings.ts +706 -0
  156. package/src/config.ts +414 -0
  157. package/src/cursor.ts +239 -0
  158. package/src/debug/index.ts +431 -0
  159. package/src/debug/log-formatting.ts +60 -0
  160. package/src/debug/log-viewer.ts +903 -0
  161. package/src/debug/profiler.ts +158 -0
  162. package/src/debug/report-bundle.ts +366 -0
  163. package/src/debug/system-info.ts +112 -0
  164. package/src/discovery/agents-md.ts +68 -0
  165. package/src/discovery/agents.ts +199 -0
  166. package/src/discovery/builtin.ts +815 -0
  167. package/src/discovery/claude-plugins.ts +205 -0
  168. package/src/discovery/claude.ts +506 -0
  169. package/src/discovery/cline.ts +83 -0
  170. package/src/discovery/codex.ts +532 -0
  171. package/src/discovery/cursor.ts +218 -0
  172. package/src/discovery/gemini.ts +395 -0
  173. package/src/discovery/github.ts +117 -0
  174. package/src/discovery/helpers.ts +698 -0
  175. package/src/discovery/index.ts +89 -0
  176. package/src/discovery/mcp-json.ts +156 -0
  177. package/src/discovery/opencode.ts +394 -0
  178. package/src/discovery/ssh.ts +160 -0
  179. package/src/discovery/vscode.ts +103 -0
  180. package/src/discovery/windsurf.ts +145 -0
  181. package/src/exa/company.ts +57 -0
  182. package/src/exa/index.ts +62 -0
  183. package/src/exa/linkedin.ts +57 -0
  184. package/src/exa/mcp-client.ts +289 -0
  185. package/src/exa/render.ts +244 -0
  186. package/src/exa/researcher.ts +89 -0
  187. package/src/exa/search.ts +330 -0
  188. package/src/exa/types.ts +166 -0
  189. package/src/exa/websets.ts +247 -0
  190. package/src/exec/bash-executor.ts +184 -0
  191. package/src/exec/exec.ts +53 -0
  192. package/src/export/custom-share.ts +65 -0
  193. package/src/export/html/index.ts +162 -0
  194. package/src/export/html/template.css +889 -0
  195. package/src/export/html/template.generated.ts +2 -0
  196. package/src/export/html/template.html +45 -0
  197. package/src/export/html/template.js +1329 -0
  198. package/src/export/html/template.macro.ts +24 -0
  199. package/src/export/html/vendor/highlight.min.js +1213 -0
  200. package/src/export/html/vendor/marked.min.js +6 -0
  201. package/src/export/ttsr.ts +434 -0
  202. package/src/extensibility/custom-commands/bundled/review/index.ts +433 -0
  203. package/src/extensibility/custom-commands/index.ts +15 -0
  204. package/src/extensibility/custom-commands/loader.ts +231 -0
  205. package/src/extensibility/custom-commands/types.ts +111 -0
  206. package/src/extensibility/custom-tools/index.ts +22 -0
  207. package/src/extensibility/custom-tools/loader.ts +235 -0
  208. package/src/extensibility/custom-tools/types.ts +226 -0
  209. package/src/extensibility/custom-tools/wrapper.ts +45 -0
  210. package/src/extensibility/extensions/index.ts +136 -0
  211. package/src/extensibility/extensions/loader.ts +520 -0
  212. package/src/extensibility/extensions/runner.ts +774 -0
  213. package/src/extensibility/extensions/types.ts +1293 -0
  214. package/src/extensibility/extensions/wrapper.ts +188 -0
  215. package/src/extensibility/hooks/index.ts +16 -0
  216. package/src/extensibility/hooks/loader.ts +273 -0
  217. package/src/extensibility/hooks/runner.ts +441 -0
  218. package/src/extensibility/hooks/tool-wrapper.ts +106 -0
  219. package/src/extensibility/hooks/types.ts +817 -0
  220. package/src/extensibility/plugins/doctor.ts +65 -0
  221. package/src/extensibility/plugins/git-url.ts +281 -0
  222. package/src/extensibility/plugins/index.ts +33 -0
  223. package/src/extensibility/plugins/installer.ts +192 -0
  224. package/src/extensibility/plugins/loader.ts +338 -0
  225. package/src/extensibility/plugins/manager.ts +716 -0
  226. package/src/extensibility/plugins/parser.ts +105 -0
  227. package/src/extensibility/plugins/types.ts +190 -0
  228. package/src/extensibility/skills.ts +385 -0
  229. package/src/extensibility/slash-commands.ts +287 -0
  230. package/src/extensibility/tool-proxy.ts +25 -0
  231. package/src/index.ts +275 -0
  232. package/src/internal-urls/agent-protocol.ts +136 -0
  233. package/src/internal-urls/artifact-protocol.ts +97 -0
  234. package/src/internal-urls/docs-index.generated.ts +54 -0
  235. package/src/internal-urls/docs-protocol.ts +84 -0
  236. package/src/internal-urls/index.ts +31 -0
  237. package/src/internal-urls/json-query.ts +126 -0
  238. package/src/internal-urls/memory-protocol.ts +133 -0
  239. package/src/internal-urls/router.ts +70 -0
  240. package/src/internal-urls/rule-protocol.ts +55 -0
  241. package/src/internal-urls/skill-protocol.ts +111 -0
  242. package/src/internal-urls/types.ts +52 -0
  243. package/src/ipy/executor.ts +556 -0
  244. package/src/ipy/gateway-coordinator.ts +426 -0
  245. package/src/ipy/kernel.ts +892 -0
  246. package/src/ipy/modules.ts +109 -0
  247. package/src/ipy/prelude.py +831 -0
  248. package/src/ipy/prelude.ts +3 -0
  249. package/src/ipy/runtime.ts +222 -0
  250. package/src/lsp/client.ts +867 -0
  251. package/src/lsp/clients/biome-client.ts +202 -0
  252. package/src/lsp/clients/index.ts +50 -0
  253. package/src/lsp/clients/lsp-linter-client.ts +93 -0
  254. package/src/lsp/clients/swiftlint-client.ts +120 -0
  255. package/src/lsp/config.ts +397 -0
  256. package/src/lsp/defaults.json +464 -0
  257. package/src/lsp/edits.ts +109 -0
  258. package/src/lsp/index.ts +1268 -0
  259. package/src/lsp/lspmux.ts +250 -0
  260. package/src/lsp/render.ts +689 -0
  261. package/src/lsp/types.ts +414 -0
  262. package/src/lsp/utils.ts +549 -0
  263. package/src/main.ts +773 -0
  264. package/src/mcp/client.ts +239 -0
  265. package/src/mcp/config-writer.ts +215 -0
  266. package/src/mcp/config.ts +363 -0
  267. package/src/mcp/index.ts +55 -0
  268. package/src/mcp/json-rpc.ts +84 -0
  269. package/src/mcp/loader.ts +124 -0
  270. package/src/mcp/manager.ts +490 -0
  271. package/src/mcp/oauth-discovery.ts +274 -0
  272. package/src/mcp/oauth-flow.ts +229 -0
  273. package/src/mcp/render.ts +123 -0
  274. package/src/mcp/tool-bridge.ts +372 -0
  275. package/src/mcp/tool-cache.ts +121 -0
  276. package/src/mcp/transports/http.ts +332 -0
  277. package/src/mcp/transports/index.ts +6 -0
  278. package/src/mcp/transports/stdio.ts +281 -0
  279. package/src/mcp/types.ts +248 -0
  280. package/src/memories/index.ts +1099 -0
  281. package/src/memories/storage.ts +563 -0
  282. package/src/modes/components/agent-dashboard.ts +1130 -0
  283. package/src/modes/components/assistant-message.ts +144 -0
  284. package/src/modes/components/bash-execution.ts +218 -0
  285. package/src/modes/components/bordered-loader.ts +41 -0
  286. package/src/modes/components/branch-summary-message.ts +45 -0
  287. package/src/modes/components/codemode-group.ts +369 -0
  288. package/src/modes/components/compaction-summary-message.ts +51 -0
  289. package/src/modes/components/countdown-timer.ts +46 -0
  290. package/src/modes/components/custom-editor.ts +181 -0
  291. package/src/modes/components/custom-message.ts +91 -0
  292. package/src/modes/components/diff.ts +186 -0
  293. package/src/modes/components/dynamic-border.ts +25 -0
  294. package/src/modes/components/extensions/extension-dashboard.ts +325 -0
  295. package/src/modes/components/extensions/extension-list.ts +484 -0
  296. package/src/modes/components/extensions/index.ts +9 -0
  297. package/src/modes/components/extensions/inspector-panel.ts +321 -0
  298. package/src/modes/components/extensions/state-manager.ts +586 -0
  299. package/src/modes/components/extensions/types.ts +191 -0
  300. package/src/modes/components/footer.ts +315 -0
  301. package/src/modes/components/history-search.ts +157 -0
  302. package/src/modes/components/hook-editor.ts +101 -0
  303. package/src/modes/components/hook-input.ts +72 -0
  304. package/src/modes/components/hook-message.ts +100 -0
  305. package/src/modes/components/hook-selector.ts +155 -0
  306. package/src/modes/components/index.ts +41 -0
  307. package/src/modes/components/keybinding-hints.ts +65 -0
  308. package/src/modes/components/login-dialog.ts +164 -0
  309. package/src/modes/components/mcp-add-wizard.ts +1295 -0
  310. package/src/modes/components/model-selector.ts +625 -0
  311. package/src/modes/components/oauth-selector.ts +210 -0
  312. package/src/modes/components/plugin-settings.ts +477 -0
  313. package/src/modes/components/python-execution.ts +196 -0
  314. package/src/modes/components/queue-mode-selector.ts +56 -0
  315. package/src/modes/components/read-tool-group.ts +119 -0
  316. package/src/modes/components/session-selector.ts +242 -0
  317. package/src/modes/components/settings-defs.ts +340 -0
  318. package/src/modes/components/settings-selector.ts +529 -0
  319. package/src/modes/components/show-images-selector.ts +45 -0
  320. package/src/modes/components/skill-message.ts +90 -0
  321. package/src/modes/components/status-line/index.ts +4 -0
  322. package/src/modes/components/status-line/presets.ts +94 -0
  323. package/src/modes/components/status-line/segments.ts +352 -0
  324. package/src/modes/components/status-line/separators.ts +55 -0
  325. package/src/modes/components/status-line/types.ts +75 -0
  326. package/src/modes/components/status-line-segment-editor.ts +354 -0
  327. package/src/modes/components/status-line.ts +421 -0
  328. package/src/modes/components/theme-selector.ts +63 -0
  329. package/src/modes/components/thinking-selector.ts +64 -0
  330. package/src/modes/components/todo-display.ts +115 -0
  331. package/src/modes/components/todo-reminder.ts +40 -0
  332. package/src/modes/components/tool-execution.ts +703 -0
  333. package/src/modes/components/tree-selector.ts +904 -0
  334. package/src/modes/components/ttsr-notification.ts +80 -0
  335. package/src/modes/components/user-message-selector.ts +146 -0
  336. package/src/modes/components/user-message.ts +22 -0
  337. package/src/modes/components/visual-truncate.ts +63 -0
  338. package/src/modes/components/welcome.ts +247 -0
  339. package/src/modes/controllers/command-controller.ts +1120 -0
  340. package/src/modes/controllers/event-controller.ts +479 -0
  341. package/src/modes/controllers/extension-ui-controller.ts +778 -0
  342. package/src/modes/controllers/input-controller.ts +671 -0
  343. package/src/modes/controllers/mcp-command-controller.ts +1315 -0
  344. package/src/modes/controllers/selector-controller.ts +712 -0
  345. package/src/modes/controllers/ssh-command-controller.ts +452 -0
  346. package/src/modes/index.ts +15 -0
  347. package/src/modes/interactive-mode.ts +1027 -0
  348. package/src/modes/print-mode.ts +191 -0
  349. package/src/modes/rpc/rpc-client.ts +583 -0
  350. package/src/modes/rpc/rpc-mode.ts +700 -0
  351. package/src/modes/rpc/rpc-types.ts +236 -0
  352. package/src/modes/theme/dark.json +95 -0
  353. package/src/modes/theme/defaults/alabaster.json +93 -0
  354. package/src/modes/theme/defaults/amethyst.json +96 -0
  355. package/src/modes/theme/defaults/anthracite.json +93 -0
  356. package/src/modes/theme/defaults/basalt.json +91 -0
  357. package/src/modes/theme/defaults/birch.json +95 -0
  358. package/src/modes/theme/defaults/dark-abyss.json +91 -0
  359. package/src/modes/theme/defaults/dark-arctic.json +104 -0
  360. package/src/modes/theme/defaults/dark-aurora.json +95 -0
  361. package/src/modes/theme/defaults/dark-catppuccin.json +107 -0
  362. package/src/modes/theme/defaults/dark-cavern.json +91 -0
  363. package/src/modes/theme/defaults/dark-copper.json +95 -0
  364. package/src/modes/theme/defaults/dark-cosmos.json +90 -0
  365. package/src/modes/theme/defaults/dark-cyberpunk.json +102 -0
  366. package/src/modes/theme/defaults/dark-dracula.json +98 -0
  367. package/src/modes/theme/defaults/dark-eclipse.json +91 -0
  368. package/src/modes/theme/defaults/dark-ember.json +95 -0
  369. package/src/modes/theme/defaults/dark-equinox.json +90 -0
  370. package/src/modes/theme/defaults/dark-forest.json +96 -0
  371. package/src/modes/theme/defaults/dark-github.json +105 -0
  372. package/src/modes/theme/defaults/dark-gruvbox.json +112 -0
  373. package/src/modes/theme/defaults/dark-lavender.json +95 -0
  374. package/src/modes/theme/defaults/dark-lunar.json +89 -0
  375. package/src/modes/theme/defaults/dark-midnight.json +95 -0
  376. package/src/modes/theme/defaults/dark-monochrome.json +94 -0
  377. package/src/modes/theme/defaults/dark-monokai.json +98 -0
  378. package/src/modes/theme/defaults/dark-nebula.json +90 -0
  379. package/src/modes/theme/defaults/dark-nord.json +97 -0
  380. package/src/modes/theme/defaults/dark-ocean.json +101 -0
  381. package/src/modes/theme/defaults/dark-one.json +100 -0
  382. package/src/modes/theme/defaults/dark-rainforest.json +91 -0
  383. package/src/modes/theme/defaults/dark-reef.json +91 -0
  384. package/src/modes/theme/defaults/dark-retro.json +92 -0
  385. package/src/modes/theme/defaults/dark-rose-pine.json +96 -0
  386. package/src/modes/theme/defaults/dark-sakura.json +95 -0
  387. package/src/modes/theme/defaults/dark-slate.json +95 -0
  388. package/src/modes/theme/defaults/dark-solarized.json +97 -0
  389. package/src/modes/theme/defaults/dark-solstice.json +90 -0
  390. package/src/modes/theme/defaults/dark-starfall.json +91 -0
  391. package/src/modes/theme/defaults/dark-sunset.json +99 -0
  392. package/src/modes/theme/defaults/dark-swamp.json +90 -0
  393. package/src/modes/theme/defaults/dark-synthwave.json +103 -0
  394. package/src/modes/theme/defaults/dark-taiga.json +91 -0
  395. package/src/modes/theme/defaults/dark-terminal.json +95 -0
  396. package/src/modes/theme/defaults/dark-tokyo-night.json +101 -0
  397. package/src/modes/theme/defaults/dark-tundra.json +91 -0
  398. package/src/modes/theme/defaults/dark-twilight.json +91 -0
  399. package/src/modes/theme/defaults/dark-volcanic.json +91 -0
  400. package/src/modes/theme/defaults/graphite.json +92 -0
  401. package/src/modes/theme/defaults/index.ts +195 -0
  402. package/src/modes/theme/defaults/light-arctic.json +107 -0
  403. package/src/modes/theme/defaults/light-aurora-day.json +91 -0
  404. package/src/modes/theme/defaults/light-canyon.json +91 -0
  405. package/src/modes/theme/defaults/light-catppuccin.json +106 -0
  406. package/src/modes/theme/defaults/light-cirrus.json +90 -0
  407. package/src/modes/theme/defaults/light-coral.json +95 -0
  408. package/src/modes/theme/defaults/light-cyberpunk.json +96 -0
  409. package/src/modes/theme/defaults/light-dawn.json +90 -0
  410. package/src/modes/theme/defaults/light-dunes.json +91 -0
  411. package/src/modes/theme/defaults/light-eucalyptus.json +95 -0
  412. package/src/modes/theme/defaults/light-forest.json +100 -0
  413. package/src/modes/theme/defaults/light-frost.json +95 -0
  414. package/src/modes/theme/defaults/light-github.json +115 -0
  415. package/src/modes/theme/defaults/light-glacier.json +91 -0
  416. package/src/modes/theme/defaults/light-gruvbox.json +108 -0
  417. package/src/modes/theme/defaults/light-haze.json +90 -0
  418. package/src/modes/theme/defaults/light-honeycomb.json +95 -0
  419. package/src/modes/theme/defaults/light-lagoon.json +91 -0
  420. package/src/modes/theme/defaults/light-lavender.json +95 -0
  421. package/src/modes/theme/defaults/light-meadow.json +91 -0
  422. package/src/modes/theme/defaults/light-mint.json +95 -0
  423. package/src/modes/theme/defaults/light-monochrome.json +101 -0
  424. package/src/modes/theme/defaults/light-ocean.json +99 -0
  425. package/src/modes/theme/defaults/light-one.json +99 -0
  426. package/src/modes/theme/defaults/light-opal.json +91 -0
  427. package/src/modes/theme/defaults/light-orchard.json +91 -0
  428. package/src/modes/theme/defaults/light-paper.json +95 -0
  429. package/src/modes/theme/defaults/light-prism.json +90 -0
  430. package/src/modes/theme/defaults/light-retro.json +98 -0
  431. package/src/modes/theme/defaults/light-sand.json +95 -0
  432. package/src/modes/theme/defaults/light-savanna.json +91 -0
  433. package/src/modes/theme/defaults/light-solarized.json +102 -0
  434. package/src/modes/theme/defaults/light-soleil.json +90 -0
  435. package/src/modes/theme/defaults/light-sunset.json +99 -0
  436. package/src/modes/theme/defaults/light-synthwave.json +98 -0
  437. package/src/modes/theme/defaults/light-tokyo-night.json +111 -0
  438. package/src/modes/theme/defaults/light-wetland.json +91 -0
  439. package/src/modes/theme/defaults/light-zenith.json +89 -0
  440. package/src/modes/theme/defaults/limestone.json +94 -0
  441. package/src/modes/theme/defaults/mahogany.json +97 -0
  442. package/src/modes/theme/defaults/marble.json +93 -0
  443. package/src/modes/theme/defaults/obsidian.json +91 -0
  444. package/src/modes/theme/defaults/onyx.json +91 -0
  445. package/src/modes/theme/defaults/pearl.json +93 -0
  446. package/src/modes/theme/defaults/porcelain.json +91 -0
  447. package/src/modes/theme/defaults/quartz.json +96 -0
  448. package/src/modes/theme/defaults/sandstone.json +95 -0
  449. package/src/modes/theme/defaults/titanium.json +90 -0
  450. package/src/modes/theme/light.json +93 -0
  451. package/src/modes/theme/mermaid-cache.ts +111 -0
  452. package/src/modes/theme/theme-schema.json +429 -0
  453. package/src/modes/theme/theme.ts +2333 -0
  454. package/src/modes/types.ts +216 -0
  455. package/src/modes/utils/ui-helpers.ts +529 -0
  456. package/src/patch/applicator.ts +1482 -0
  457. package/src/patch/diff.ts +425 -0
  458. package/src/patch/fuzzy.ts +784 -0
  459. package/src/patch/hashline.ts +972 -0
  460. package/src/patch/index.ts +964 -0
  461. package/src/patch/normalize.ts +397 -0
  462. package/src/patch/normative.ts +72 -0
  463. package/src/patch/parser.ts +532 -0
  464. package/src/patch/shared.ts +400 -0
  465. package/src/patch/types.ts +292 -0
  466. package/src/priority.json +35 -0
  467. package/src/prompts/agents/explore.md +48 -0
  468. package/src/prompts/agents/frontmatter.md +9 -0
  469. package/src/prompts/agents/init.md +36 -0
  470. package/src/prompts/agents/librarian.md +53 -0
  471. package/src/prompts/agents/oracle.md +51 -0
  472. package/src/prompts/agents/reviewer.md +70 -0
  473. package/src/prompts/agents/task.md +14 -0
  474. package/src/prompts/compaction/branch-summary-context.md +5 -0
  475. package/src/prompts/compaction/branch-summary-preamble.md +2 -0
  476. package/src/prompts/compaction/branch-summary.md +30 -0
  477. package/src/prompts/compaction/compaction-short-summary.md +9 -0
  478. package/src/prompts/compaction/compaction-summary-context.md +5 -0
  479. package/src/prompts/compaction/compaction-summary.md +38 -0
  480. package/src/prompts/compaction/compaction-turn-prefix.md +17 -0
  481. package/src/prompts/compaction/compaction-update-summary.md +45 -0
  482. package/src/prompts/memories/consolidation.md +30 -0
  483. package/src/prompts/memories/read_path.md +11 -0
  484. package/src/prompts/memories/stage_one_input.md +6 -0
  485. package/src/prompts/memories/stage_one_system.md +21 -0
  486. package/src/prompts/review-request.md +64 -0
  487. package/src/prompts/system/agent-creation-architect.md +65 -0
  488. package/src/prompts/system/agent-creation-user.md +6 -0
  489. package/src/prompts/system/custom-system-prompt.md +68 -0
  490. package/src/prompts/system/file-operations.md +10 -0
  491. package/src/prompts/system/subagent-submit-reminder.md +11 -0
  492. package/src/prompts/system/subagent-system-prompt.md +31 -0
  493. package/src/prompts/system/subagent-user-prompt.md +8 -0
  494. package/src/prompts/system/summarization-system.md +3 -0
  495. package/src/prompts/system/system-prompt.md +300 -0
  496. package/src/prompts/system/title-system.md +2 -0
  497. package/src/prompts/system/ttsr-interrupt.md +7 -0
  498. package/src/prompts/system/web-search.md +28 -0
  499. package/src/prompts/tools/ask.md +44 -0
  500. package/src/prompts/tools/bash.md +24 -0
  501. package/src/prompts/tools/browser.md +33 -0
  502. package/src/prompts/tools/calculator.md +12 -0
  503. package/src/prompts/tools/explore.md +29 -0
  504. package/src/prompts/tools/fetch.md +16 -0
  505. package/src/prompts/tools/find.md +18 -0
  506. package/src/prompts/tools/gemini-image.md +23 -0
  507. package/src/prompts/tools/grep.md +28 -0
  508. package/src/prompts/tools/hashline.md +232 -0
  509. package/src/prompts/tools/librarian.md +24 -0
  510. package/src/prompts/tools/lsp.md +28 -0
  511. package/src/prompts/tools/oracle.md +26 -0
  512. package/src/prompts/tools/patch.md +74 -0
  513. package/src/prompts/tools/python.md +66 -0
  514. package/src/prompts/tools/read.md +36 -0
  515. package/src/prompts/tools/replace.md +38 -0
  516. package/src/prompts/tools/reviewer.md +41 -0
  517. package/src/prompts/tools/ssh.md +51 -0
  518. package/src/prompts/tools/task-summary.md +28 -0
  519. package/src/prompts/tools/task.md +275 -0
  520. package/src/prompts/tools/todo-write.md +65 -0
  521. package/src/prompts/tools/undo-edit.md +7 -0
  522. package/src/prompts/tools/web-search.md +19 -0
  523. package/src/prompts/tools/write.md +18 -0
  524. package/src/sdk.ts +1287 -0
  525. package/src/secrets/index.ts +116 -0
  526. package/src/secrets/obfuscator.ts +269 -0
  527. package/src/secrets/regex.ts +21 -0
  528. package/src/session/agent-session.ts +4669 -0
  529. package/src/session/agent-storage.ts +621 -0
  530. package/src/session/artifacts.ts +132 -0
  531. package/src/session/auth-storage.ts +1433 -0
  532. package/src/session/blob-store.ts +103 -0
  533. package/src/session/compaction/branch-summarization.ts +315 -0
  534. package/src/session/compaction/compaction.ts +864 -0
  535. package/src/session/compaction/index.ts +7 -0
  536. package/src/session/compaction/pruning.ts +91 -0
  537. package/src/session/compaction/utils.ts +171 -0
  538. package/src/session/history-storage.ts +170 -0
  539. package/src/session/messages.ts +317 -0
  540. package/src/session/session-manager.ts +2276 -0
  541. package/src/session/session-storage.ts +342 -0
  542. package/src/session/streaming-output.ts +565 -0
  543. package/src/slash-commands/builtin-registry.ts +439 -0
  544. package/src/ssh/config-writer.ts +183 -0
  545. package/src/ssh/connection-manager.ts +444 -0
  546. package/src/ssh/ssh-executor.ts +127 -0
  547. package/src/ssh/sshfs-mount.ts +135 -0
  548. package/src/stt/downloader.ts +71 -0
  549. package/src/stt/index.ts +3 -0
  550. package/src/stt/recorder.ts +351 -0
  551. package/src/stt/setup.ts +52 -0
  552. package/src/stt/stt-controller.ts +160 -0
  553. package/src/stt/transcribe.py +70 -0
  554. package/src/stt/transcriber.ts +91 -0
  555. package/src/system-prompt.ts +685 -0
  556. package/src/task/agents.ts +155 -0
  557. package/src/task/batch.ts +102 -0
  558. package/src/task/commands.ts +134 -0
  559. package/src/task/discovery.ts +126 -0
  560. package/src/task/executor.ts +908 -0
  561. package/src/task/index.ts +223 -0
  562. package/src/task/output-manager.ts +107 -0
  563. package/src/task/parallel.ts +84 -0
  564. package/src/task/render.ts +326 -0
  565. package/src/task/subprocess-tool-registry.ts +88 -0
  566. package/src/task/template.ts +32 -0
  567. package/src/task/types.ts +144 -0
  568. package/src/tools/ask.ts +523 -0
  569. package/src/tools/bash-interactive.ts +419 -0
  570. package/src/tools/bash-interceptor.ts +105 -0
  571. package/src/tools/bash-normalize.ts +107 -0
  572. package/src/tools/bash-skill-urls.ts +177 -0
  573. package/src/tools/bash.ts +347 -0
  574. package/src/tools/browser.ts +1374 -0
  575. package/src/tools/calculator.ts +537 -0
  576. package/src/tools/context.ts +39 -0
  577. package/src/tools/explore.ts +23 -0
  578. package/src/tools/fetch.ts +1091 -0
  579. package/src/tools/find.ts +540 -0
  580. package/src/tools/fs-cache-invalidation.ts +28 -0
  581. package/src/tools/gemini-image.ts +907 -0
  582. package/src/tools/grep.ts +489 -0
  583. package/src/tools/index.ts +337 -0
  584. package/src/tools/json-tree.ts +231 -0
  585. package/src/tools/jtd-to-json-schema.ts +247 -0
  586. package/src/tools/jtd-to-typescript.ts +198 -0
  587. package/src/tools/librarian.ts +33 -0
  588. package/src/tools/list-limit.ts +40 -0
  589. package/src/tools/notebook.ts +287 -0
  590. package/src/tools/oracle.ts +40 -0
  591. package/src/tools/output-meta.ts +459 -0
  592. package/src/tools/output-utils.ts +63 -0
  593. package/src/tools/path-utils.ts +116 -0
  594. package/src/tools/puppeteer/00_stealth_tampering.txt +63 -0
  595. package/src/tools/puppeteer/01_stealth_activity.txt +20 -0
  596. package/src/tools/puppeteer/02_stealth_hairline.txt +11 -0
  597. package/src/tools/puppeteer/03_stealth_botd.txt +384 -0
  598. package/src/tools/puppeteer/04_stealth_iframe.txt +81 -0
  599. package/src/tools/puppeteer/05_stealth_webgl.txt +75 -0
  600. package/src/tools/puppeteer/06_stealth_screen.txt +72 -0
  601. package/src/tools/puppeteer/07_stealth_fonts.txt +97 -0
  602. package/src/tools/puppeteer/08_stealth_audio.txt +51 -0
  603. package/src/tools/puppeteer/09_stealth_locale.txt +46 -0
  604. package/src/tools/puppeteer/10_stealth_plugins.txt +206 -0
  605. package/src/tools/puppeteer/11_stealth_hardware.txt +8 -0
  606. package/src/tools/puppeteer/12_stealth_codecs.txt +40 -0
  607. package/src/tools/puppeteer/13_stealth_worker.txt +74 -0
  608. package/src/tools/python.ts +1118 -0
  609. package/src/tools/read.ts +1193 -0
  610. package/src/tools/render-utils.ts +680 -0
  611. package/src/tools/renderers.ts +60 -0
  612. package/src/tools/reviewer-tool.ts +41 -0
  613. package/src/tools/ssh.ts +326 -0
  614. package/src/tools/subagent-tool.ts +169 -0
  615. package/src/tools/submit-result.ts +152 -0
  616. package/src/tools/todo-write.ts +255 -0
  617. package/src/tools/tool-errors.ts +92 -0
  618. package/src/tools/tool-result.ts +86 -0
  619. package/src/tools/undo-edit.ts +145 -0
  620. package/src/tools/undo-history.ts +22 -0
  621. package/src/tools/write.ts +274 -0
  622. package/src/tui/code-cell.ts +108 -0
  623. package/src/tui/file-list.ts +47 -0
  624. package/src/tui/index.ts +11 -0
  625. package/src/tui/output-block.ts +144 -0
  626. package/src/tui/status-line.ts +39 -0
  627. package/src/tui/tree-list.ts +53 -0
  628. package/src/tui/types.ts +16 -0
  629. package/src/tui/utils.ts +116 -0
  630. package/src/utils/changelog.ts +98 -0
  631. package/src/utils/event-bus.ts +33 -0
  632. package/src/utils/external-editor.ts +59 -0
  633. package/src/utils/file-display-mode.ts +36 -0
  634. package/src/utils/file-mentions.ts +384 -0
  635. package/src/utils/frontmatter.ts +101 -0
  636. package/src/utils/fuzzy.ts +108 -0
  637. package/src/utils/ignore-files.ts +119 -0
  638. package/src/utils/image-convert.ts +27 -0
  639. package/src/utils/image-resize.ts +236 -0
  640. package/src/utils/mime.ts +30 -0
  641. package/src/utils/open.ts +20 -0
  642. package/src/utils/shell-snapshot.ts +199 -0
  643. package/src/utils/timings.ts +26 -0
  644. package/src/utils/title-generator.ts +167 -0
  645. package/src/utils/tools-manager.ts +362 -0
  646. package/src/web/scrapers/artifacthub.ts +215 -0
  647. package/src/web/scrapers/arxiv.ts +88 -0
  648. package/src/web/scrapers/aur.ts +175 -0
  649. package/src/web/scrapers/biorxiv.ts +141 -0
  650. package/src/web/scrapers/bluesky.ts +284 -0
  651. package/src/web/scrapers/brew.ts +177 -0
  652. package/src/web/scrapers/cheatsh.ts +78 -0
  653. package/src/web/scrapers/chocolatey.ts +158 -0
  654. package/src/web/scrapers/choosealicense.ts +110 -0
  655. package/src/web/scrapers/cisa-kev.ts +100 -0
  656. package/src/web/scrapers/clojars.ts +180 -0
  657. package/src/web/scrapers/coingecko.ts +184 -0
  658. package/src/web/scrapers/crates-io.ts +128 -0
  659. package/src/web/scrapers/crossref.ts +149 -0
  660. package/src/web/scrapers/devto.ts +177 -0
  661. package/src/web/scrapers/discogs.ts +307 -0
  662. package/src/web/scrapers/discourse.ts +221 -0
  663. package/src/web/scrapers/dockerhub.ts +160 -0
  664. package/src/web/scrapers/fdroid.ts +158 -0
  665. package/src/web/scrapers/firefox-addons.ts +214 -0
  666. package/src/web/scrapers/flathub.ts +239 -0
  667. package/src/web/scrapers/github-gist.ts +68 -0
  668. package/src/web/scrapers/github.ts +490 -0
  669. package/src/web/scrapers/gitlab.ts +456 -0
  670. package/src/web/scrapers/go-pkg.ts +275 -0
  671. package/src/web/scrapers/hackage.ts +94 -0
  672. package/src/web/scrapers/hackernews.ts +208 -0
  673. package/src/web/scrapers/hex.ts +121 -0
  674. package/src/web/scrapers/huggingface.ts +385 -0
  675. package/src/web/scrapers/iacr.ts +86 -0
  676. package/src/web/scrapers/index.ts +249 -0
  677. package/src/web/scrapers/jetbrains-marketplace.ts +169 -0
  678. package/src/web/scrapers/lemmy.ts +220 -0
  679. package/src/web/scrapers/lobsters.ts +186 -0
  680. package/src/web/scrapers/mastodon.ts +310 -0
  681. package/src/web/scrapers/maven.ts +152 -0
  682. package/src/web/scrapers/mdn.ts +172 -0
  683. package/src/web/scrapers/metacpan.ts +253 -0
  684. package/src/web/scrapers/musicbrainz.ts +272 -0
  685. package/src/web/scrapers/npm.ts +114 -0
  686. package/src/web/scrapers/nuget.ts +205 -0
  687. package/src/web/scrapers/nvd.ts +243 -0
  688. package/src/web/scrapers/ollama.ts +265 -0
  689. package/src/web/scrapers/open-vsx.ts +119 -0
  690. package/src/web/scrapers/opencorporates.ts +275 -0
  691. package/src/web/scrapers/openlibrary.ts +319 -0
  692. package/src/web/scrapers/orcid.ts +298 -0
  693. package/src/web/scrapers/osv.ts +192 -0
  694. package/src/web/scrapers/packagist.ts +174 -0
  695. package/src/web/scrapers/pub-dev.ts +185 -0
  696. package/src/web/scrapers/pubmed.ts +177 -0
  697. package/src/web/scrapers/pypi.ts +129 -0
  698. package/src/web/scrapers/rawg.ts +124 -0
  699. package/src/web/scrapers/readthedocs.ts +125 -0
  700. package/src/web/scrapers/reddit.ts +104 -0
  701. package/src/web/scrapers/repology.ts +262 -0
  702. package/src/web/scrapers/rfc.ts +209 -0
  703. package/src/web/scrapers/rubygems.ts +117 -0
  704. package/src/web/scrapers/searchcode.ts +217 -0
  705. package/src/web/scrapers/sec-edgar.ts +274 -0
  706. package/src/web/scrapers/semantic-scholar.ts +190 -0
  707. package/src/web/scrapers/snapcraft.ts +200 -0
  708. package/src/web/scrapers/sourcegraph.ts +373 -0
  709. package/src/web/scrapers/spdx.ts +121 -0
  710. package/src/web/scrapers/spotify.ts +217 -0
  711. package/src/web/scrapers/stackoverflow.ts +124 -0
  712. package/src/web/scrapers/terraform.ts +304 -0
  713. package/src/web/scrapers/tldr.ts +51 -0
  714. package/src/web/scrapers/twitter.ts +97 -0
  715. package/src/web/scrapers/types.ts +200 -0
  716. package/src/web/scrapers/utils.ts +142 -0
  717. package/src/web/scrapers/vimeo.ts +152 -0
  718. package/src/web/scrapers/vscode-marketplace.ts +195 -0
  719. package/src/web/scrapers/w3c.ts +163 -0
  720. package/src/web/scrapers/wikidata.ts +357 -0
  721. package/src/web/scrapers/wikipedia.ts +95 -0
  722. package/src/web/scrapers/youtube.ts +312 -0
  723. package/src/web/search/auth.ts +178 -0
  724. package/src/web/search/index.ts +598 -0
  725. package/src/web/search/provider.ts +77 -0
  726. package/src/web/search/providers/anthropic.ts +284 -0
  727. package/src/web/search/providers/base.ts +22 -0
  728. package/src/web/search/providers/brave.ts +165 -0
  729. package/src/web/search/providers/codex.ts +377 -0
  730. package/src/web/search/providers/exa.ts +158 -0
  731. package/src/web/search/providers/gemini.ts +437 -0
  732. package/src/web/search/providers/jina.ts +99 -0
  733. package/src/web/search/providers/kimi.ts +196 -0
  734. package/src/web/search/providers/perplexity.ts +546 -0
  735. package/src/web/search/providers/synthetic.ts +136 -0
  736. package/src/web/search/providers/zai.ts +352 -0
  737. package/src/web/search/render.ts +299 -0
  738. package/src/web/search/types.ts +437 -0
@@ -0,0 +1,867 @@
1
+ import { isEnoent, logger, ptree, untilAborted } from "@nghyane/arcane-utils";
2
+ import { ToolAbortError, throwIfAborted } from "../tools/tool-errors";
3
+ import { applyWorkspaceEdit } from "./edits";
4
+ import { getLspmuxCommand, isLspmuxSupported } from "./lspmux";
5
+ import type {
6
+ Diagnostic,
7
+ LspClient,
8
+ LspJsonRpcNotification,
9
+ LspJsonRpcRequest,
10
+ LspJsonRpcResponse,
11
+ ServerConfig,
12
+ WorkspaceEdit,
13
+ } from "./types";
14
+ import { detectLanguageId, fileToUri } from "./utils";
15
+
16
+ // =============================================================================
17
+ // Client State
18
+ // =============================================================================
19
+
20
+ const clients = new Map<string, LspClient>();
21
+ const clientLocks = new Map<string, Promise<LspClient>>();
22
+ const fileOperationLocks = new Map<string, Promise<void>>();
23
+
24
+ // Idle timeout configuration (disabled by default)
25
+ let idleTimeoutMs: number | null = null;
26
+ let idleCheckInterval: Timer | null = null;
27
+ const IDLE_CHECK_INTERVAL_MS = 60 * 1000;
28
+
29
+ /**
30
+ * Configure the idle timeout for LSP clients.
31
+ * @param ms - Timeout in milliseconds, or null/undefined to disable
32
+ */
33
+ export function setIdleTimeout(ms: number | null | undefined): void {
34
+ idleTimeoutMs = ms ?? null;
35
+
36
+ if (idleTimeoutMs && idleTimeoutMs > 0) {
37
+ startIdleChecker();
38
+ } else {
39
+ stopIdleChecker();
40
+ }
41
+ }
42
+
43
+ function startIdleChecker(): void {
44
+ if (idleCheckInterval) return;
45
+ idleCheckInterval = setInterval(() => {
46
+ if (!idleTimeoutMs) return;
47
+ const now = Date.now();
48
+ for (const [key, client] of Array.from(clients.entries())) {
49
+ if (now - client.lastActivity > idleTimeoutMs) {
50
+ shutdownClient(key);
51
+ }
52
+ }
53
+ }, IDLE_CHECK_INTERVAL_MS);
54
+ }
55
+
56
+ function stopIdleChecker(): void {
57
+ if (idleCheckInterval) {
58
+ clearInterval(idleCheckInterval);
59
+ idleCheckInterval = null;
60
+ }
61
+ }
62
+
63
+ // =============================================================================
64
+ // Client Capabilities
65
+ // =============================================================================
66
+
67
+ const CLIENT_CAPABILITIES = {
68
+ textDocument: {
69
+ synchronization: {
70
+ didSave: true,
71
+ dynamicRegistration: false,
72
+ willSave: false,
73
+ willSaveWaitUntil: false,
74
+ },
75
+ hover: {
76
+ contentFormat: ["markdown", "plaintext"],
77
+ dynamicRegistration: false,
78
+ },
79
+ definition: {
80
+ dynamicRegistration: false,
81
+ linkSupport: true,
82
+ },
83
+ references: {
84
+ dynamicRegistration: false,
85
+ },
86
+ documentSymbol: {
87
+ dynamicRegistration: false,
88
+ hierarchicalDocumentSymbolSupport: true,
89
+ symbolKind: {
90
+ valueSet: [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26],
91
+ },
92
+ },
93
+ rename: {
94
+ dynamicRegistration: false,
95
+ prepareSupport: true,
96
+ },
97
+ codeAction: {
98
+ dynamicRegistration: false,
99
+ codeActionLiteralSupport: {
100
+ codeActionKind: {
101
+ valueSet: [
102
+ "quickfix",
103
+ "refactor",
104
+ "refactor.extract",
105
+ "refactor.inline",
106
+ "refactor.rewrite",
107
+ "source",
108
+ "source.organizeImports",
109
+ "source.fixAll",
110
+ ],
111
+ },
112
+ },
113
+ resolveSupport: {
114
+ properties: ["edit"],
115
+ },
116
+ },
117
+ formatting: {
118
+ dynamicRegistration: false,
119
+ },
120
+ rangeFormatting: {
121
+ dynamicRegistration: false,
122
+ },
123
+ publishDiagnostics: {
124
+ relatedInformation: true,
125
+ versionSupport: false,
126
+ tagSupport: { valueSet: [1, 2] },
127
+ codeDescriptionSupport: true,
128
+ dataSupport: true,
129
+ },
130
+ },
131
+ workspace: {
132
+ applyEdit: true,
133
+ workspaceEdit: {
134
+ documentChanges: true,
135
+ resourceOperations: ["create", "rename", "delete"],
136
+ failureHandling: "textOnlyTransactional",
137
+ },
138
+ configuration: true,
139
+ symbol: {
140
+ dynamicRegistration: false,
141
+ symbolKind: {
142
+ valueSet: [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26],
143
+ },
144
+ },
145
+ },
146
+ experimental: {
147
+ snippetTextEdit: true,
148
+ },
149
+ };
150
+
151
+ // =============================================================================
152
+ // LSP Message Protocol
153
+ // =============================================================================
154
+
155
+ /**
156
+ * Parse a single LSP message from a buffer.
157
+ * Returns the parsed message and remaining buffer, or null if incomplete.
158
+ */
159
+ function parseMessage(
160
+ buffer: Buffer,
161
+ ): { message: LspJsonRpcResponse | LspJsonRpcNotification; remaining: Buffer } | null {
162
+ // Only decode enough to find the header
163
+ const headerEndIndex = findHeaderEnd(buffer);
164
+ if (headerEndIndex === -1) return null;
165
+
166
+ const headerText = new TextDecoder().decode(buffer.slice(0, headerEndIndex));
167
+ const contentLengthMatch = headerText.match(/Content-Length: (\d+)/i);
168
+ if (!contentLengthMatch) return null;
169
+
170
+ const contentLength = Number.parseInt(contentLengthMatch[1], 10);
171
+ const messageStart = headerEndIndex + 4; // Skip \r\n\r\n
172
+ const messageEnd = messageStart + contentLength;
173
+
174
+ if (buffer.length < messageEnd) return null;
175
+
176
+ const messageBytes = buffer.subarray(messageStart, messageEnd);
177
+ const messageText = new TextDecoder().decode(messageBytes);
178
+ const remaining = buffer.subarray(messageEnd);
179
+
180
+ return {
181
+ message: JSON.parse(messageText),
182
+ remaining,
183
+ };
184
+ }
185
+
186
+ /**
187
+ * Find the end of the header section (before \r\n\r\n)
188
+ */
189
+ function findHeaderEnd(buffer: Uint8Array): number {
190
+ for (let i = 0; i < buffer.length - 3; i++) {
191
+ if (buffer[i] === 13 && buffer[i + 1] === 10 && buffer[i + 2] === 13 && buffer[i + 3] === 10) {
192
+ return i;
193
+ }
194
+ }
195
+ return -1;
196
+ }
197
+
198
+ async function writeMessage(
199
+ sink: Bun.FileSink,
200
+ message: LspJsonRpcRequest | LspJsonRpcNotification | LspJsonRpcResponse,
201
+ ): Promise<void> {
202
+ const content = JSON.stringify(message);
203
+ sink.write(`Content-Length: ${Buffer.byteLength(content, "utf-8")}\r\n\r\n`);
204
+ sink.write(content);
205
+ await sink.flush();
206
+ }
207
+
208
+ // =============================================================================
209
+ // Message Reader
210
+ // =============================================================================
211
+
212
+ /**
213
+ * Start background message reader for a client.
214
+ * Routes responses to pending requests and handles notifications.
215
+ */
216
+ async function startMessageReader(client: LspClient): Promise<void> {
217
+ if (client.isReading) return;
218
+ client.isReading = true;
219
+
220
+ const reader = (client.proc.stdout as ReadableStream<Uint8Array>).getReader();
221
+
222
+ try {
223
+ while (true) {
224
+ const { done, value } = await reader.read();
225
+ if (done) break;
226
+
227
+ // Atomically update buffer before processing
228
+ const currentBuffer: Buffer = Buffer.concat([client.messageBuffer, value]);
229
+ client.messageBuffer = currentBuffer;
230
+
231
+ // Process all complete messages in buffer
232
+ // Use local variable to avoid race with concurrent buffer updates
233
+ let workingBuffer = currentBuffer;
234
+ let parsed = parseMessage(workingBuffer);
235
+ while (parsed) {
236
+ const { message, remaining } = parsed;
237
+ workingBuffer = remaining;
238
+
239
+ // Route message
240
+ if ("id" in message && message.id !== undefined) {
241
+ // Response to a request
242
+ const pending = client.pendingRequests.get(message.id);
243
+ if (pending) {
244
+ client.pendingRequests.delete(message.id);
245
+ if ("error" in message && message.error) {
246
+ pending.reject(new Error(`LSP error: ${message.error.message}`));
247
+ } else {
248
+ pending.resolve(message.result);
249
+ }
250
+ } else if ("method" in message) {
251
+ await handleServerRequest(client, message as LspJsonRpcRequest);
252
+ }
253
+ } else if ("method" in message) {
254
+ // Server notification
255
+ if (message.method === "textDocument/publishDiagnostics" && message.params) {
256
+ const params = message.params as { uri: string; diagnostics: Diagnostic[] };
257
+ client.diagnostics.set(params.uri, params.diagnostics);
258
+ client.diagnosticsVersion += 1;
259
+ }
260
+ }
261
+
262
+ parsed = parseMessage(workingBuffer);
263
+ }
264
+
265
+ // Atomically commit processed buffer
266
+ client.messageBuffer = workingBuffer;
267
+ }
268
+ } catch (err) {
269
+ // Connection closed or error - reject all pending requests
270
+ for (const pending of Array.from(client.pendingRequests.values())) {
271
+ pending.reject(new Error(`LSP connection closed: ${err}`));
272
+ }
273
+ client.pendingRequests.clear();
274
+ } finally {
275
+ reader.releaseLock();
276
+ client.isReading = false;
277
+ }
278
+ }
279
+
280
+ /**
281
+ * Handle workspace/configuration requests from the server.
282
+ */
283
+ async function handleConfigurationRequest(client: LspClient, message: LspJsonRpcRequest): Promise<void> {
284
+ if (typeof message.id !== "number") return;
285
+ const params = message.params as { items?: Array<{ section?: string }> };
286
+ const items = params?.items ?? [];
287
+ const result = items.map(item => {
288
+ const section = item.section ?? "";
289
+ return client.config.settings?.[section] ?? {};
290
+ });
291
+ await sendResponse(client, message.id, result, "workspace/configuration");
292
+ }
293
+
294
+ /**
295
+ * Handle workspace/applyEdit requests from the server.
296
+ */
297
+ async function handleApplyEditRequest(client: LspClient, message: LspJsonRpcRequest): Promise<void> {
298
+ if (typeof message.id !== "number") return;
299
+ const params = message.params as { edit?: WorkspaceEdit };
300
+ if (!params?.edit) {
301
+ await sendResponse(
302
+ client,
303
+ message.id,
304
+ { applied: false, failureReason: "No edit provided" },
305
+ "workspace/applyEdit",
306
+ );
307
+ return;
308
+ }
309
+
310
+ try {
311
+ await applyWorkspaceEdit(params.edit, client.cwd);
312
+ await sendResponse(client, message.id, { applied: true }, "workspace/applyEdit");
313
+ } catch (err) {
314
+ await sendResponse(client, message.id, { applied: false, failureReason: String(err) }, "workspace/applyEdit");
315
+ }
316
+ }
317
+
318
+ /**
319
+ * Respond to a server-initiated request.
320
+ */
321
+ async function handleServerRequest(client: LspClient, message: LspJsonRpcRequest): Promise<void> {
322
+ if (message.method === "workspace/configuration") {
323
+ await handleConfigurationRequest(client, message);
324
+ return;
325
+ }
326
+ if (message.method === "workspace/applyEdit") {
327
+ await handleApplyEditRequest(client, message);
328
+ return;
329
+ }
330
+ if (typeof message.id !== "number") return;
331
+ await sendResponse(client, message.id, null, message.method, {
332
+ code: -32601,
333
+ message: `Method not found: ${message.method}`,
334
+ });
335
+ }
336
+
337
+ /**
338
+ * Send an LSP response to the server.
339
+ */
340
+ async function sendResponse(
341
+ client: LspClient,
342
+ id: number,
343
+ result: unknown,
344
+ method: string,
345
+ error?: { code: number; message: string; data?: unknown },
346
+ ): Promise<void> {
347
+ const response: LspJsonRpcResponse = {
348
+ jsonrpc: "2.0",
349
+ id,
350
+ ...(error ? { error } : { result }),
351
+ };
352
+
353
+ try {
354
+ await writeMessage(client.proc.stdin, response);
355
+ } catch (err) {
356
+ logger.error("LSP failed to respond.", { method, error: String(err) });
357
+ }
358
+ }
359
+
360
+ // =============================================================================
361
+ // Client Management
362
+ // =============================================================================
363
+
364
+ /** Timeout for warmup initialize requests (5 seconds) */
365
+ export const WARMUP_TIMEOUT_MS = 5000;
366
+
367
+ /**
368
+ * Get or create an LSP client for the given server configuration and working directory.
369
+ * @param config - Server configuration
370
+ * @param cwd - Working directory
371
+ * @param initTimeoutMs - Optional timeout for the initialize request (defaults to 30s)
372
+ */
373
+ export async function getOrCreateClient(config: ServerConfig, cwd: string, initTimeoutMs?: number): Promise<LspClient> {
374
+ const key = `${config.command}:${cwd}`;
375
+
376
+ // Check if client already exists
377
+ const existingClient = clients.get(key);
378
+ if (existingClient) {
379
+ existingClient.lastActivity = Date.now();
380
+ return existingClient;
381
+ }
382
+
383
+ // Check if another coroutine is already creating this client
384
+ const existingLock = clientLocks.get(key);
385
+ if (existingLock) {
386
+ return existingLock;
387
+ }
388
+
389
+ // Create new client with lock
390
+ const clientPromise = (async () => {
391
+ const baseCommand = config.resolvedCommand ?? config.command;
392
+ const baseArgs = config.args ?? [];
393
+
394
+ // Wrap with lspmux if available and supported
395
+ const { command, args, env } = isLspmuxSupported(baseCommand)
396
+ ? await getLspmuxCommand(baseCommand, baseArgs)
397
+ : { command: baseCommand, args: baseArgs };
398
+
399
+ const proc = ptree.spawn([command, ...args], {
400
+ cwd,
401
+ stdin: "pipe",
402
+ env: env ? { ...Bun.env, ...env } : undefined,
403
+ });
404
+
405
+ const client: LspClient = {
406
+ name: key,
407
+ cwd,
408
+ proc,
409
+ config,
410
+ requestId: 0,
411
+ diagnostics: new Map(),
412
+ diagnosticsVersion: 0,
413
+ openFiles: new Map(),
414
+ pendingRequests: new Map(),
415
+ messageBuffer: new Uint8Array(0),
416
+ isReading: false,
417
+ lastActivity: Date.now(),
418
+ };
419
+ clients.set(key, client);
420
+
421
+ // Register crash recovery - remove client on process exit
422
+ proc.exited.then(() => {
423
+ clients.delete(key);
424
+ clientLocks.delete(key);
425
+
426
+ // Reject any pending requests — the server is gone, they will never complete.
427
+ if (client.pendingRequests.size > 0) {
428
+ const stderr = proc.peekStderr().trim();
429
+ const code = proc.exitCode;
430
+ const err = new Error(
431
+ stderr ? `LSP server exited (code ${code}): ${stderr}` : `LSP server exited unexpectedly (code ${code})`,
432
+ );
433
+ for (const pending of client.pendingRequests.values()) {
434
+ pending.reject(err);
435
+ }
436
+ client.pendingRequests.clear();
437
+ }
438
+ });
439
+
440
+ // Start background message reader
441
+ startMessageReader(client);
442
+
443
+ try {
444
+ // Send initialize request
445
+ const initResult = (await sendRequest(
446
+ client,
447
+ "initialize",
448
+ {
449
+ processId: process.pid,
450
+ rootUri: fileToUri(cwd),
451
+ rootPath: cwd,
452
+ capabilities: CLIENT_CAPABILITIES,
453
+ initializationOptions: config.initOptions ?? {},
454
+ workspaceFolders: [{ uri: fileToUri(cwd), name: cwd.split("/").pop() ?? "workspace" }],
455
+ },
456
+ undefined, // signal
457
+ initTimeoutMs,
458
+ )) as { capabilities?: unknown };
459
+
460
+ if (!initResult) {
461
+ throw new Error("Failed to initialize LSP: no response");
462
+ }
463
+
464
+ client.serverCapabilities = initResult.capabilities as LspClient["serverCapabilities"];
465
+
466
+ // Send initialized notification
467
+ await sendNotification(client, "initialized", {});
468
+
469
+ return client;
470
+ } catch (err) {
471
+ // Clean up on initialization failure
472
+ clients.delete(key);
473
+ clientLocks.delete(key);
474
+ proc.kill();
475
+ throw err;
476
+ } finally {
477
+ clientLocks.delete(key);
478
+ }
479
+ })();
480
+
481
+ clientLocks.set(key, clientPromise);
482
+ return clientPromise;
483
+ }
484
+
485
+ /**
486
+ * Ensure a file is opened in the LSP client.
487
+ * Sends didOpen notification if the file is not already tracked.
488
+ */
489
+ export async function ensureFileOpen(client: LspClient, filePath: string, signal?: AbortSignal): Promise<void> {
490
+ throwIfAborted(signal);
491
+ const uri = fileToUri(filePath);
492
+ const lockKey = `${client.name}:${uri}`;
493
+
494
+ // Check if file is already open
495
+ if (client.openFiles.has(uri)) {
496
+ return;
497
+ }
498
+
499
+ // Check if another operation is already opening this file
500
+ const existingLock = fileOperationLocks.get(lockKey);
501
+ if (existingLock) {
502
+ await untilAborted(signal, () => existingLock);
503
+ return;
504
+ }
505
+
506
+ // Lock and open file
507
+ const openPromise = (async () => {
508
+ throwIfAborted(signal);
509
+ // Double-check after acquiring lock
510
+ if (client.openFiles.has(uri)) {
511
+ return;
512
+ }
513
+
514
+ let content: string;
515
+ try {
516
+ content = await Bun.file(filePath).text();
517
+ throwIfAborted(signal);
518
+ } catch (err) {
519
+ if (isEnoent(err)) return;
520
+ throw err;
521
+ }
522
+ const languageId = detectLanguageId(filePath);
523
+ throwIfAborted(signal);
524
+
525
+ await sendNotification(client, "textDocument/didOpen", {
526
+ textDocument: {
527
+ uri,
528
+ languageId,
529
+ version: 1,
530
+ text: content,
531
+ },
532
+ });
533
+
534
+ client.openFiles.set(uri, { version: 1, languageId });
535
+ client.lastActivity = Date.now();
536
+ })();
537
+
538
+ fileOperationLocks.set(lockKey, openPromise);
539
+ try {
540
+ await openPromise;
541
+ } finally {
542
+ fileOperationLocks.delete(lockKey);
543
+ }
544
+ }
545
+
546
+ /**
547
+ * Sync in-memory content to the LSP client without reading from disk.
548
+ * Use this to provide instant feedback during edits before the file is saved.
549
+ */
550
+ export async function syncContent(
551
+ client: LspClient,
552
+ filePath: string,
553
+ content: string,
554
+ signal?: AbortSignal,
555
+ ): Promise<void> {
556
+ const uri = fileToUri(filePath);
557
+ const lockKey = `${client.name}:${uri}`;
558
+ throwIfAborted(signal);
559
+
560
+ const existingLock = fileOperationLocks.get(lockKey);
561
+ if (existingLock) {
562
+ await untilAborted(signal, () => existingLock);
563
+ }
564
+
565
+ const syncPromise = (async () => {
566
+ // Clear stale diagnostics before syncing new content
567
+ client.diagnostics.delete(uri);
568
+
569
+ const info = client.openFiles.get(uri);
570
+
571
+ if (!info) {
572
+ // Open file with provided content instead of reading from disk
573
+ const languageId = detectLanguageId(filePath);
574
+ throwIfAborted(signal);
575
+ await sendNotification(client, "textDocument/didOpen", {
576
+ textDocument: {
577
+ uri,
578
+ languageId,
579
+ version: 1,
580
+ text: content,
581
+ },
582
+ });
583
+ client.openFiles.set(uri, { version: 1, languageId });
584
+ client.lastActivity = Date.now();
585
+ return;
586
+ }
587
+
588
+ const version = ++info.version;
589
+ throwIfAborted(signal);
590
+ await sendNotification(client, "textDocument/didChange", {
591
+ textDocument: { uri, version },
592
+ contentChanges: [{ text: content }],
593
+ });
594
+ client.lastActivity = Date.now();
595
+ })();
596
+
597
+ fileOperationLocks.set(lockKey, syncPromise);
598
+ try {
599
+ await syncPromise;
600
+ } finally {
601
+ fileOperationLocks.delete(lockKey);
602
+ }
603
+ }
604
+
605
+ /**
606
+ * Notify LSP that a file was saved.
607
+ * Assumes content was already synced via syncContent - just sends didSave.
608
+ */
609
+ export async function notifySaved(client: LspClient, filePath: string, signal?: AbortSignal): Promise<void> {
610
+ const uri = fileToUri(filePath);
611
+ const info = client.openFiles.get(uri);
612
+ if (!info) return; // File not open, nothing to notify
613
+
614
+ throwIfAborted(signal);
615
+ await sendNotification(client, "textDocument/didSave", {
616
+ textDocument: { uri },
617
+ });
618
+ client.lastActivity = Date.now();
619
+ }
620
+
621
+ /**
622
+ * Refresh a file in the LSP client.
623
+ * Increments version, sends didChange and didSave notifications.
624
+ */
625
+ export async function refreshFile(client: LspClient, filePath: string, signal?: AbortSignal): Promise<void> {
626
+ throwIfAborted(signal);
627
+ const uri = fileToUri(filePath);
628
+ const lockKey = `${client.name}:${uri}`;
629
+
630
+ // Check if another operation is in progress
631
+ const existingLock = fileOperationLocks.get(lockKey);
632
+ if (existingLock) {
633
+ await untilAborted(signal, () => existingLock);
634
+ }
635
+
636
+ // Lock and refresh file
637
+ const refreshPromise = (async () => {
638
+ throwIfAborted(signal);
639
+ const info = client.openFiles.get(uri);
640
+
641
+ if (!info) {
642
+ await ensureFileOpen(client, filePath, signal);
643
+ return;
644
+ }
645
+
646
+ let content: string;
647
+ try {
648
+ content = await Bun.file(filePath).text();
649
+ throwIfAborted(signal);
650
+ } catch (err) {
651
+ if (isEnoent(err)) return;
652
+ throw err;
653
+ }
654
+ const version = ++info.version;
655
+ throwIfAborted(signal);
656
+
657
+ await sendNotification(client, "textDocument/didChange", {
658
+ textDocument: { uri, version },
659
+ contentChanges: [{ text: content }],
660
+ });
661
+ throwIfAborted(signal);
662
+
663
+ await sendNotification(client, "textDocument/didSave", {
664
+ textDocument: { uri },
665
+ text: content,
666
+ });
667
+
668
+ client.lastActivity = Date.now();
669
+ })();
670
+
671
+ fileOperationLocks.set(lockKey, refreshPromise);
672
+ try {
673
+ await refreshPromise;
674
+ } finally {
675
+ fileOperationLocks.delete(lockKey);
676
+ }
677
+ }
678
+
679
+ /**
680
+ * Shutdown a specific client by key.
681
+ */
682
+ export function shutdownClient(key: string): void {
683
+ const client = clients.get(key);
684
+ if (!client) return;
685
+
686
+ // Reject all pending requests
687
+ for (const pending of Array.from(client.pendingRequests.values())) {
688
+ pending.reject(new Error("LSP client shutdown"));
689
+ }
690
+ client.pendingRequests.clear();
691
+
692
+ // Send shutdown request (best effort, don't wait)
693
+ sendRequest(client, "shutdown", null).catch(() => {});
694
+
695
+ // Kill process
696
+ client.proc.kill();
697
+ clients.delete(key);
698
+ }
699
+
700
+ // =============================================================================
701
+ // LSP Protocol Methods
702
+ // =============================================================================
703
+
704
+ /** Default timeout for LSP requests (30 seconds) */
705
+ const DEFAULT_REQUEST_TIMEOUT_MS = 30000;
706
+
707
+ /**
708
+ * Send an LSP request and wait for response.
709
+ */
710
+ export async function sendRequest(
711
+ client: LspClient,
712
+ method: string,
713
+ params: unknown,
714
+ signal?: AbortSignal,
715
+ timeoutMs: number = DEFAULT_REQUEST_TIMEOUT_MS,
716
+ ): Promise<unknown> {
717
+ // Atomically increment and capture request ID
718
+ const id = ++client.requestId;
719
+ if (signal?.aborted) {
720
+ const reason = signal.reason instanceof Error ? signal.reason : new ToolAbortError();
721
+ return Promise.reject(reason);
722
+ }
723
+
724
+ const request: LspJsonRpcRequest = {
725
+ jsonrpc: "2.0",
726
+ id,
727
+ method,
728
+ params,
729
+ };
730
+
731
+ client.lastActivity = Date.now();
732
+
733
+ const { promise, resolve, reject } = Promise.withResolvers<unknown>();
734
+ let timeout: NodeJS.Timeout | undefined;
735
+ const cleanup = () => {
736
+ if (signal) {
737
+ signal.removeEventListener("abort", abortHandler);
738
+ }
739
+ };
740
+ const abortHandler = () => {
741
+ if (client.pendingRequests.has(id)) {
742
+ client.pendingRequests.delete(id);
743
+ }
744
+ void sendNotification(client, "$/cancelRequest", { id }).catch(() => {});
745
+ if (timeout) clearTimeout(timeout);
746
+ cleanup();
747
+ const reason = signal?.reason instanceof Error ? signal.reason : new ToolAbortError();
748
+ reject(reason);
749
+ };
750
+
751
+ // Set timeout
752
+ timeout = setTimeout(() => {
753
+ if (client.pendingRequests.has(id)) {
754
+ client.pendingRequests.delete(id);
755
+ const err = new Error(`LSP request ${method} timed out after ${timeoutMs}ms`);
756
+ cleanup();
757
+ reject(err);
758
+ }
759
+ }, timeoutMs);
760
+ if (signal) {
761
+ signal.addEventListener("abort", abortHandler, { once: true });
762
+ if (signal.aborted) {
763
+ abortHandler();
764
+ return promise;
765
+ }
766
+ }
767
+
768
+ // Register pending request with timeout wrapper
769
+ client.pendingRequests.set(id, {
770
+ resolve: result => {
771
+ if (timeout) clearTimeout(timeout);
772
+ cleanup();
773
+ resolve(result);
774
+ },
775
+ reject: err => {
776
+ if (timeout) clearTimeout(timeout);
777
+ cleanup();
778
+ reject(err);
779
+ },
780
+ method,
781
+ });
782
+
783
+ // Write request
784
+ writeMessage(client.proc.stdin, request).catch(err => {
785
+ if (timeout) clearTimeout(timeout);
786
+ client.pendingRequests.delete(id);
787
+ cleanup();
788
+ reject(err);
789
+ });
790
+ return promise;
791
+ }
792
+
793
+ /**
794
+ * Send an LSP notification (no response expected).
795
+ */
796
+ export async function sendNotification(client: LspClient, method: string, params: unknown): Promise<void> {
797
+ const notification: LspJsonRpcNotification = {
798
+ jsonrpc: "2.0",
799
+ method,
800
+ params,
801
+ };
802
+
803
+ client.lastActivity = Date.now();
804
+ await writeMessage(client.proc.stdin, notification);
805
+ }
806
+
807
+ /**
808
+ * Shutdown all LSP clients.
809
+ */
810
+ export function shutdownAll(): void {
811
+ const clientsToShutdown = Array.from(clients.values());
812
+ clients.clear();
813
+
814
+ const err = new Error("LSP client shutdown");
815
+ for (const client of clientsToShutdown) {
816
+ /// Reject all pending requests
817
+ const reqs = Array.from(client.pendingRequests.values());
818
+ client.pendingRequests.clear();
819
+ for (const pending of reqs) {
820
+ pending.reject(err);
821
+ }
822
+
823
+ void (async () => {
824
+ // Send shutdown request (best effort, don't wait)
825
+ const timeout = Bun.sleep(5_000);
826
+ const result = sendRequest(client, "shutdown", null).catch(() => {});
827
+ await Promise.race([result, timeout]);
828
+ client.proc.kill();
829
+ })().catch(() => {});
830
+ }
831
+ }
832
+
833
+ /** Status of an LSP server */
834
+ export interface LspServerStatus {
835
+ name: string;
836
+ status: "connecting" | "ready" | "error";
837
+ fileTypes: string[];
838
+ error?: string;
839
+ }
840
+
841
+ /**
842
+ * Get status of all active LSP clients.
843
+ */
844
+ export function getActiveClients(): LspServerStatus[] {
845
+ return Array.from(clients.values()).map(client => ({
846
+ name: client.config.command,
847
+ status: "ready" as const,
848
+ fileTypes: client.config.fileTypes,
849
+ }));
850
+ }
851
+
852
+ // =============================================================================
853
+ // Process Cleanup
854
+ // =============================================================================
855
+
856
+ // Register cleanup on module unload
857
+ if (typeof process !== "undefined") {
858
+ process.on("beforeExit", shutdownAll);
859
+ process.on("SIGINT", () => {
860
+ shutdownAll();
861
+ process.exit(0);
862
+ });
863
+ process.on("SIGTERM", () => {
864
+ shutdownAll();
865
+ process.exit(0);
866
+ });
867
+ }