pi-lens 2.2.9 → 3.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (304) hide show
  1. package/CHANGELOG.md +198 -0
  2. package/README.md +709 -519
  3. package/clients/__tests__/file-time.test.js +216 -0
  4. package/clients/__tests__/file-time.test.ts +276 -0
  5. package/clients/__tests__/format-service.test.js +245 -0
  6. package/clients/__tests__/format-service.test.ts +339 -0
  7. package/clients/__tests__/formatters.test.js +271 -0
  8. package/clients/__tests__/formatters.test.ts +401 -0
  9. package/clients/amain-types.js +164 -0
  10. package/clients/amain-types.ts +165 -0
  11. package/clients/architect-client.js +56 -12
  12. package/clients/architect-client.ts +81 -16
  13. package/clients/ast-grep-client.js +2 -2
  14. package/clients/ast-grep-client.ts +14 -39
  15. package/clients/ast-grep-parser.ts +1 -1
  16. package/clients/ast-grep-rule-manager.js +8 -0
  17. package/clients/ast-grep-rule-manager.ts +10 -1
  18. package/clients/ast-grep-types.js +9 -0
  19. package/clients/ast-grep-types.ts +106 -0
  20. package/clients/auto-loop.js +10 -0
  21. package/clients/auto-loop.ts +14 -1
  22. package/clients/biome-client.js +81 -19
  23. package/clients/biome-client.ts +103 -22
  24. package/clients/bus/bus.js +191 -0
  25. package/clients/bus/bus.ts +251 -0
  26. package/clients/bus/events.js +214 -0
  27. package/clients/bus/events.ts +279 -0
  28. package/clients/bus/index.js +8 -0
  29. package/clients/bus/index.ts +9 -0
  30. package/clients/bus/integration.js +158 -0
  31. package/clients/bus/integration.ts +214 -0
  32. package/clients/complexity-client.js +13 -7
  33. package/clients/complexity-client.ts +13 -7
  34. package/clients/config-validator.js +465 -0
  35. package/clients/config-validator.ts +558 -0
  36. package/clients/dependency-checker.js +4 -10
  37. package/clients/dependency-checker.ts +4 -10
  38. package/clients/dispatch/__tests__/autofix-integration.test.js +245 -0
  39. package/clients/dispatch/__tests__/autofix-integration.test.ts +300 -0
  40. package/clients/dispatch/__tests__/runner-registration.test.js +236 -0
  41. package/clients/dispatch/__tests__/runner-registration.test.ts +282 -0
  42. package/clients/dispatch/bus-dispatcher.js +177 -0
  43. package/clients/dispatch/bus-dispatcher.ts +251 -0
  44. package/clients/dispatch/dispatcher.edge.test.js +82 -0
  45. package/clients/dispatch/dispatcher.edge.test.ts +100 -0
  46. package/clients/dispatch/dispatcher.format.test.js +46 -0
  47. package/clients/dispatch/dispatcher.format.test.ts +58 -0
  48. package/clients/dispatch/dispatcher.inline.test.js +74 -0
  49. package/clients/dispatch/dispatcher.inline.test.ts +93 -0
  50. package/clients/dispatch/dispatcher.js +19 -53
  51. package/clients/dispatch/dispatcher.ts +20 -67
  52. package/clients/dispatch/plan.js +9 -4
  53. package/clients/dispatch/plan.ts +9 -4
  54. package/clients/dispatch/runners/architect.js +21 -7
  55. package/clients/dispatch/runners/architect.test.js +138 -0
  56. package/clients/dispatch/runners/architect.test.ts +162 -0
  57. package/clients/dispatch/runners/architect.ts +22 -7
  58. package/clients/dispatch/runners/ast-grep-napi.js +462 -0
  59. package/clients/dispatch/runners/ast-grep-napi.test.js +111 -0
  60. package/clients/dispatch/runners/ast-grep-napi.test.ts +133 -0
  61. package/clients/dispatch/runners/ast-grep-napi.ts +506 -0
  62. package/clients/dispatch/runners/ast-grep.js +62 -19
  63. package/clients/dispatch/runners/ast-grep.ts +70 -18
  64. package/clients/dispatch/runners/biome.js +29 -53
  65. package/clients/dispatch/runners/biome.ts +29 -63
  66. package/clients/dispatch/runners/config-validation.js +67 -0
  67. package/clients/dispatch/runners/config-validation.ts +82 -0
  68. package/clients/dispatch/runners/go-vet.js +4 -28
  69. package/clients/dispatch/runners/go-vet.ts +4 -32
  70. package/clients/dispatch/runners/index.js +30 -10
  71. package/clients/dispatch/runners/index.ts +30 -10
  72. package/clients/dispatch/runners/oxlint.js +141 -0
  73. package/clients/dispatch/runners/oxlint.test.js +230 -0
  74. package/clients/dispatch/runners/oxlint.test.ts +303 -0
  75. package/clients/dispatch/runners/oxlint.ts +175 -0
  76. package/clients/dispatch/runners/pyright.js +40 -70
  77. package/clients/dispatch/runners/pyright.test.js +16 -2
  78. package/clients/dispatch/runners/pyright.test.ts +14 -2
  79. package/clients/dispatch/runners/pyright.ts +48 -91
  80. package/clients/dispatch/runners/python-slop.js +97 -0
  81. package/clients/dispatch/runners/python-slop.test.js +203 -0
  82. package/clients/dispatch/runners/python-slop.test.ts +298 -0
  83. package/clients/dispatch/runners/python-slop.ts +124 -0
  84. package/clients/dispatch/runners/ruff.js +18 -71
  85. package/clients/dispatch/runners/ruff.ts +19 -79
  86. package/clients/dispatch/runners/rust-clippy.js +28 -32
  87. package/clients/dispatch/runners/rust-clippy.ts +29 -31
  88. package/clients/dispatch/runners/scan_codebase.test.js +89 -0
  89. package/clients/dispatch/runners/scan_codebase.test.ts +105 -0
  90. package/clients/dispatch/runners/shellcheck.js +147 -0
  91. package/clients/dispatch/runners/shellcheck.test.js +98 -0
  92. package/clients/dispatch/runners/shellcheck.test.ts +129 -0
  93. package/clients/dispatch/runners/shellcheck.ts +188 -0
  94. package/clients/dispatch/runners/similarity.js +230 -0
  95. package/clients/dispatch/runners/similarity.ts +339 -0
  96. package/clients/dispatch/runners/spellcheck.js +106 -0
  97. package/clients/dispatch/runners/spellcheck.test.js +158 -0
  98. package/clients/dispatch/runners/spellcheck.test.ts +214 -0
  99. package/clients/dispatch/runners/spellcheck.ts +136 -0
  100. package/clients/dispatch/runners/tree-sitter.js +107 -0
  101. package/clients/dispatch/runners/tree-sitter.ts +135 -0
  102. package/clients/dispatch/runners/ts-lsp.js +104 -33
  103. package/clients/dispatch/runners/ts-lsp.ts +120 -38
  104. package/clients/dispatch/runners/ts-slop.js +113 -0
  105. package/clients/dispatch/runners/ts-slop.test.js +180 -0
  106. package/clients/dispatch/runners/ts-slop.test.ts +230 -0
  107. package/clients/dispatch/runners/ts-slop.ts +142 -0
  108. package/clients/dispatch/runners/utils/diagnostic-parsers.js +134 -0
  109. package/clients/dispatch/runners/utils/diagnostic-parsers.ts +186 -0
  110. package/clients/dispatch/runners/utils/runner-helpers.js +115 -0
  111. package/clients/dispatch/runners/utils/runner-helpers.ts +167 -0
  112. package/clients/dispatch/runners/utils.js +2 -4
  113. package/clients/dispatch/runners/utils.ts +2 -4
  114. package/clients/dispatch/types.ts +1 -1
  115. package/clients/dispatch/utils/format-utils.js +49 -0
  116. package/clients/dispatch/utils/format-utils.ts +60 -0
  117. package/clients/dogfood.test.js +201 -0
  118. package/clients/dogfood.test.ts +269 -0
  119. package/clients/file-time.js +152 -0
  120. package/clients/file-time.ts +208 -0
  121. package/clients/file-utils.js +40 -0
  122. package/clients/file-utils.ts +44 -0
  123. package/clients/fix-scanners.js +10 -20
  124. package/clients/fix-scanners.ts +10 -22
  125. package/clients/format-service.js +172 -0
  126. package/clients/format-service.ts +254 -0
  127. package/clients/formatters.js +435 -0
  128. package/clients/formatters.ts +508 -0
  129. package/clients/go-client.js +5 -14
  130. package/clients/go-client.ts +5 -13
  131. package/clients/installer/index.js +356 -0
  132. package/clients/installer/index.ts +426 -0
  133. package/clients/jscpd-client.js +11 -9
  134. package/clients/jscpd-client.ts +12 -8
  135. package/clients/knip-client.js +3 -7
  136. package/clients/knip-client.ts +3 -6
  137. package/clients/lsp/__tests__/client.test.js +325 -0
  138. package/clients/lsp/__tests__/client.test.ts +434 -0
  139. package/clients/lsp/__tests__/config.test.js +166 -0
  140. package/clients/lsp/__tests__/config.test.ts +209 -0
  141. package/clients/lsp/__tests__/error-recovery.test.js +213 -0
  142. package/clients/lsp/__tests__/error-recovery.test.ts +279 -0
  143. package/clients/lsp/__tests__/integration.test.js +127 -0
  144. package/clients/lsp/__tests__/integration.test.ts +160 -0
  145. package/clients/lsp/__tests__/launch.test.js +260 -0
  146. package/clients/lsp/__tests__/launch.test.ts +329 -0
  147. package/clients/lsp/__tests__/server.test.js +259 -0
  148. package/clients/lsp/__tests__/server.test.ts +332 -0
  149. package/clients/lsp/__tests__/service.test.js +417 -0
  150. package/clients/lsp/__tests__/service.test.ts +499 -0
  151. package/clients/lsp/client.js +235 -0
  152. package/clients/lsp/client.ts +328 -0
  153. package/clients/lsp/config.js +115 -0
  154. package/clients/lsp/config.ts +149 -0
  155. package/clients/lsp/index.js +222 -0
  156. package/clients/lsp/index.ts +280 -0
  157. package/clients/lsp/installer/index.js +391 -0
  158. package/clients/lsp/interactive-install.js +210 -0
  159. package/clients/lsp/interactive-install.ts +251 -0
  160. package/clients/lsp/language.js +170 -0
  161. package/clients/lsp/language.ts +216 -0
  162. package/clients/lsp/launch.js +174 -0
  163. package/clients/lsp/launch.ts +240 -0
  164. package/clients/lsp/lsp/launch.js +116 -0
  165. package/clients/lsp/lsp/server.js +532 -0
  166. package/clients/lsp/lsp-index.js +10 -0
  167. package/clients/lsp/lsp-index.ts +11 -0
  168. package/clients/lsp/path-utils.js +48 -0
  169. package/clients/lsp/path-utils.ts +52 -0
  170. package/clients/lsp/server.js +615 -0
  171. package/clients/lsp/server.ts +800 -0
  172. package/clients/lsp/test-py-spawn/requirements.txt +1 -0
  173. package/clients/lsp/test-py-spawn/test.py +3 -0
  174. package/clients/lsp/test-py-svc/requirements.txt +1 -0
  175. package/clients/lsp/test-py-svc/test.py +3 -0
  176. package/clients/lsp/test-python-project/requirements.txt +1 -0
  177. package/clients/lsp/test-python-project/test.py +5 -0
  178. package/clients/metrics-history.js +2 -2
  179. package/clients/metrics-history.ts +2 -2
  180. package/clients/production-readiness.js +522 -0
  181. package/clients/production-readiness.ts +556 -0
  182. package/clients/project-index.js +255 -0
  183. package/clients/project-index.ts +383 -0
  184. package/clients/project-metadata.js +531 -0
  185. package/clients/project-metadata.ts +624 -0
  186. package/clients/ruff-client.js +56 -16
  187. package/clients/ruff-client.ts +72 -15
  188. package/clients/runner-tracker.js +152 -0
  189. package/clients/runner-tracker.ts +213 -0
  190. package/clients/rust-client.js +4 -11
  191. package/clients/rust-client.ts +5 -11
  192. package/clients/safe-spawn.js +96 -0
  193. package/clients/safe-spawn.ts +128 -0
  194. package/clients/scan-architectural-debt.js +3 -6
  195. package/clients/scan-architectural-debt.ts +3 -6
  196. package/clients/scan-utils.js +5 -20
  197. package/clients/scan-utils.ts +5 -29
  198. package/clients/secrets-scanner.js +3 -17
  199. package/clients/secrets-scanner.ts +4 -20
  200. package/clients/services/__tests__/effect-integration.test.js +86 -0
  201. package/clients/services/__tests__/effect-integration.test.ts +111 -0
  202. package/clients/services/effect-integration.js +194 -0
  203. package/clients/services/effect-integration.ts +268 -0
  204. package/clients/services/index.js +7 -0
  205. package/clients/services/index.ts +8 -0
  206. package/clients/services/runner-service.js +105 -0
  207. package/clients/services/runner-service.ts +179 -0
  208. package/clients/sg-runner.js +87 -13
  209. package/clients/sg-runner.ts +97 -13
  210. package/clients/state-matrix.js +160 -0
  211. package/clients/state-matrix.ts +202 -0
  212. package/clients/subprocess-client.js +10 -9
  213. package/clients/subprocess-client.ts +10 -8
  214. package/clients/test-runner-client.js +3 -7
  215. package/clients/test-runner-client.ts +3 -6
  216. package/clients/tool-availability.js +4 -10
  217. package/clients/tool-availability.ts +4 -9
  218. package/clients/tree-sitter-client.js +564 -0
  219. package/clients/tree-sitter-client.ts +797 -0
  220. package/clients/tree-sitter-query-loader.js +355 -0
  221. package/clients/tree-sitter-query-loader.ts +425 -0
  222. package/clients/type-coverage-client.js +3 -7
  223. package/clients/type-coverage-client.ts +3 -6
  224. package/clients/typescript-client.codefix.test.js +157 -0
  225. package/clients/typescript-client.codefix.test.ts +186 -0
  226. package/clients/typescript-client.js +43 -0
  227. package/clients/typescript-client.ts +98 -0
  228. package/commands/booboo.js +799 -219
  229. package/commands/booboo.ts +1004 -225
  230. package/commands/clients/ast-grep-client.js +250 -0
  231. package/commands/clients/ast-grep-parser.js +86 -0
  232. package/commands/clients/ast-grep-rule-manager.js +91 -0
  233. package/commands/clients/ast-grep-types.js +9 -0
  234. package/commands/clients/biome-client.js +380 -0
  235. package/commands/clients/complexity-client.js +667 -0
  236. package/commands/clients/file-kinds.js +177 -0
  237. package/commands/clients/file-utils.js +40 -0
  238. package/commands/clients/jscpd-client.js +169 -0
  239. package/commands/clients/knip-client.js +211 -0
  240. package/commands/clients/ruff-client.js +297 -0
  241. package/commands/clients/safe-spawn.js +88 -0
  242. package/commands/clients/scan-utils.js +83 -0
  243. package/commands/clients/sg-runner.js +190 -0
  244. package/commands/clients/types.js +11 -0
  245. package/commands/clients/typescript-client.js +505 -0
  246. package/commands/fix-from-booboo.js +398 -0
  247. package/commands/fix-from-booboo.ts +485 -0
  248. package/commands/fix-simplified.js +618 -0
  249. package/commands/fix-simplified.ts +768 -0
  250. package/commands/rate.js +10 -14
  251. package/commands/rate.ts +9 -16
  252. package/default-architect.yaml +59 -15
  253. package/index.ts +342 -429
  254. package/package.json +16 -3
  255. package/rules/ast-grep-rules/rules/empty-catch.yml +38 -13
  256. package/rules/ast-grep-rules/rules/no-array-constructor.yml +1 -0
  257. package/rules/ast-grep-rules/rules/no-debugger.yml +2 -0
  258. package/rules/python-slop-rules/.sgconfig.yml +4 -0
  259. package/rules/python-slop-rules/rules/slop-rules.yml +647 -0
  260. package/rules/tree-sitter-queries/python/bare-except.yml +54 -0
  261. package/rules/tree-sitter-queries/python/eval-exec.yml +50 -0
  262. package/rules/tree-sitter-queries/python/is-vs-equals.yml +60 -0
  263. package/rules/tree-sitter-queries/python/mutable-default-arg.yml +57 -0
  264. package/rules/tree-sitter-queries/python/unreachable-except.yml +60 -0
  265. package/rules/tree-sitter-queries/python/wildcard-import.yml +46 -0
  266. package/rules/tree-sitter-queries/tsx/dangerously-set-inner-html.yml +63 -0
  267. package/rules/tree-sitter-queries/typescript/await-in-loop.yml +56 -0
  268. package/rules/tree-sitter-queries/typescript/console-statement.yml +47 -0
  269. package/rules/tree-sitter-queries/typescript/debugger.yml +47 -0
  270. package/rules/tree-sitter-queries/typescript/deep-nesting.yml +117 -0
  271. package/rules/tree-sitter-queries/typescript/deep-promise-chain.yml +73 -0
  272. package/rules/tree-sitter-queries/typescript/empty-catch.yml +64 -0
  273. package/rules/tree-sitter-queries/typescript/eval.yml +48 -0
  274. package/rules/tree-sitter-queries/typescript/hardcoded-secrets.yml +78 -0
  275. package/rules/tree-sitter-queries/typescript/long-parameter-list.yml +62 -0
  276. package/rules/tree-sitter-queries/typescript/mixed-async-styles.yml +49 -0
  277. package/rules/tree-sitter-queries/typescript/nested-ternary.yml +45 -0
  278. package/rules/ts-slop-rules/.sgconfig.yml +4 -0
  279. package/rules/ts-slop-rules/rules/in-correct-optional-input-type.yml +10 -0
  280. package/rules/ts-slop-rules/rules/jwt-no-verify.yml +13 -0
  281. package/rules/ts-slop-rules/rules/no-architecture-violation.yml +10 -0
  282. package/rules/ts-slop-rules/rules/no-case-declarations.yml +10 -0
  283. package/rules/ts-slop-rules/rules/no-dangerously-set-inner-html.yml +10 -0
  284. package/rules/ts-slop-rules/rules/no-debugger.yml +10 -0
  285. package/rules/ts-slop-rules/rules/no-dupe-args.yml +10 -0
  286. package/rules/ts-slop-rules/rules/no-dupe-class-members.yml +10 -0
  287. package/rules/ts-slop-rules/rules/no-dupe-keys.yml +10 -0
  288. package/rules/ts-slop-rules/rules/no-eval.yml +13 -0
  289. package/rules/ts-slop-rules/rules/no-hardcoded-secrets.yml +12 -0
  290. package/rules/ts-slop-rules/rules/no-implied-eval.yml +12 -0
  291. package/rules/ts-slop-rules/rules/no-inner-html.yml +13 -0
  292. package/rules/ts-slop-rules/rules/no-javascript-url.yml +10 -0
  293. package/rules/ts-slop-rules/rules/no-mutable-default.yml +10 -0
  294. package/rules/ts-slop-rules/rules/no-nested-links.yml +12 -0
  295. package/rules/ts-slop-rules/rules/no-new-symbol.yml +10 -0
  296. package/rules/ts-slop-rules/rules/no-new-wrappers.yml +13 -0
  297. package/rules/ts-slop-rules/rules/no-open-redirect.yml +16 -0
  298. package/rules/ts-slop-rules/rules/slop-rules.yml +455 -0
  299. package/rules/ts-slop-rules/rules/weak-rsa-key.yml +12 -0
  300. package/skills/ast-grep/SKILL.md +182 -0
  301. package/clients/dispatch/runners/secrets.js +0 -109
  302. package/commands/fix.js +0 -244
  303. package/commands/fix.ts +0 -373
  304. package/rules/ast-grep-rules/rules/no-lonely-if.yml +0 -13
@@ -0,0 +1,615 @@
1
+ /**
2
+ * LSP Server Definitions for pi-lens
3
+ *
4
+ * Defines 40+ language servers with:
5
+ * - Root detection (monorepo support)
6
+ * - Auto-installation strategies
7
+ * - Platform-specific handling
8
+ */
9
+ import path from "node:path";
10
+ import { ensureTool, getToolEnvironment } from "../installer/index.js";
11
+ import { promptForInstall, supportsInteractiveInstall, } from "./interactive-install.js";
12
+ import { launchLSP, launchViaPackageManager, } from "./launch.js";
13
+ // --- Root Detection Helpers ---
14
+ import { dirname } from "node:path";
15
+ import { fileURLToPath } from "node:url";
16
+ const __filename = fileURLToPath(import.meta.url);
17
+ const __dirname = dirname(__filename);
18
+ // --- Interactive Install Helper ---
19
+ /**
20
+ * Spawn LSP with interactive install support for common languages
21
+ *
22
+ * For Go, Rust, YAML, JSON, Bash: prompts user to install if tool not found
23
+ * Other languages: throws error with install instructions
24
+ */
25
+ async function spawnWithInteractiveInstall(language, _command, _args, options, spawnFn) {
26
+ try {
27
+ return spawnFn();
28
+ }
29
+ catch (error) {
30
+ // Check if this is a "command not found" error
31
+ const errorMsg = String(error);
32
+ if (!errorMsg.includes("not found") && !errorMsg.includes("ENOENT")) {
33
+ throw error; // Re-throw if it's a different error
34
+ }
35
+ // Check if language supports interactive install
36
+ if (supportsInteractiveInstall(language)) {
37
+ const shouldInstall = await promptForInstall(language, options.cwd);
38
+ if (shouldInstall) {
39
+ // Try again after install
40
+ return spawnFn();
41
+ }
42
+ // User declined, return undefined to skip this LSP
43
+ return undefined;
44
+ }
45
+ // For other languages, throw with install instructions
46
+ throw error;
47
+ }
48
+ }
49
+ /**
50
+ * Walk up the tree looking for project root markers
51
+ */
52
+ export function createRootDetector(includePatterns, excludePatterns) {
53
+ return async (file) => {
54
+ let currentDir = path.dirname(file);
55
+ const root = path.parse(currentDir).root;
56
+ while (currentDir !== root) {
57
+ // Check exclude patterns first
58
+ if (excludePatterns) {
59
+ for (const pattern of excludePatterns) {
60
+ const checkPath = path.join(currentDir, pattern);
61
+ try {
62
+ const stat = await import("node:fs/promises").then((fs) => fs.stat(checkPath));
63
+ if (stat)
64
+ return undefined; // Excluded
65
+ }
66
+ catch {
67
+ /* not found */
68
+ }
69
+ }
70
+ }
71
+ // Check include patterns
72
+ for (const pattern of includePatterns) {
73
+ const checkPath = path.join(currentDir, pattern);
74
+ try {
75
+ const stat = await import("node:fs/promises").then((fs) => fs.stat(checkPath));
76
+ if (stat)
77
+ return currentDir;
78
+ }
79
+ catch {
80
+ /* not found */
81
+ }
82
+ }
83
+ currentDir = path.dirname(currentDir);
84
+ }
85
+ return undefined;
86
+ };
87
+ }
88
+ // --- Server Definitions ---
89
+ export const TypeScriptServer = {
90
+ id: "typescript",
91
+ name: "TypeScript Language Server",
92
+ extensions: [".ts", ".tsx", ".js", ".jsx", ".mjs", ".cjs", ".mts", ".cts"],
93
+ root: createRootDetector([
94
+ "package-lock.json",
95
+ "bun.lockb",
96
+ "bun.lock",
97
+ "pnpm-lock.yaml",
98
+ "yarn.lock",
99
+ "package.json",
100
+ ], [".pi-lens"]),
101
+ async spawn(root) {
102
+ const path = await import("node:path");
103
+ const fs = await import("node:fs/promises");
104
+ // Find typescript-language-server - prefer local project version
105
+ let lspPath;
106
+ const localLsp = path.join(root, "node_modules", ".bin", "typescript-language-server");
107
+ const localLspCmd = path.join(root, "node_modules", ".bin", "typescript-language-server.cmd");
108
+ // Check for local version first (Windows .cmd first, then Unix)
109
+ for (const checkPath of [localLspCmd, localLsp]) {
110
+ try {
111
+ await fs.access(checkPath);
112
+ lspPath = checkPath;
113
+ break;
114
+ }
115
+ catch {
116
+ /* not found */
117
+ }
118
+ }
119
+ // Fall back to auto-installed version
120
+ if (!lspPath) {
121
+ lspPath = await ensureTool("typescript-language-server");
122
+ if (!lspPath) {
123
+ console.error("[lsp] typescript-language-server not found");
124
+ return undefined;
125
+ }
126
+ }
127
+ // Find tsserver.js path (needed for TypeScript LSP initialization)
128
+ // Check relative to the LSP path first, then project root
129
+ let tsserverPath;
130
+ const tsserverCandidates = [
131
+ // Relative to LSP binary (for locally installed)
132
+ path.join(path.dirname(lspPath), "..", "typescript", "lib", "tsserver.js"),
133
+ // Project root
134
+ path.join(root, "node_modules", "typescript", "lib", "tsserver.js"),
135
+ // Current working directory
136
+ path.join(process.cwd(), "node_modules", "typescript", "lib", "tsserver.js"),
137
+ ];
138
+ for (const checkPath of tsserverCandidates) {
139
+ try {
140
+ await fs.access(checkPath);
141
+ tsserverPath = checkPath;
142
+ break;
143
+ }
144
+ catch {
145
+ /* not found */
146
+ }
147
+ }
148
+ // Use absolute path and proper environment
149
+ const env = await getToolEnvironment();
150
+ const proc = launchLSP(lspPath, ["--stdio"], {
151
+ cwd: root,
152
+ env: {
153
+ ...env,
154
+ TSSERVER_PATH: tsserverPath,
155
+ },
156
+ });
157
+ return {
158
+ process: proc,
159
+ initialization: tsserverPath
160
+ ? { tsserver: { path: tsserverPath } }
161
+ : undefined,
162
+ };
163
+ },
164
+ };
165
+ export const PythonServer = {
166
+ id: "python",
167
+ name: "Pyright Language Server",
168
+ extensions: [".py", ".pyi"],
169
+ root: createRootDetector([
170
+ "pyproject.toml",
171
+ "setup.py",
172
+ "setup.cfg",
173
+ "requirements.txt",
174
+ "Pipfile",
175
+ "poetry.lock",
176
+ ], [".pi-lens"]),
177
+ async spawn(root) {
178
+ const env = await getToolEnvironment();
179
+ const proc = launchViaPackageManager("pyright-langserver", ["--stdio"], {
180
+ cwd: root,
181
+ env,
182
+ });
183
+ // Detect virtual environment
184
+ const initialization = {};
185
+ const venvPaths = [
186
+ path.join(root, ".venv"),
187
+ path.join(root, "venv"),
188
+ process.env.VIRTUAL_ENV,
189
+ ].filter(Boolean);
190
+ for (const venv of venvPaths) {
191
+ if (!venv)
192
+ continue;
193
+ try {
194
+ const pythonPath = process.platform === "win32"
195
+ ? path.join(venv, "Scripts", "python.exe")
196
+ : path.join(venv, "bin", "python");
197
+ await import("node:fs/promises").then((fs) => fs.access(pythonPath));
198
+ // Pyright expects pythonPath at top level, not nested
199
+ initialization.pythonPath = pythonPath;
200
+ break;
201
+ }
202
+ catch {
203
+ /* not found */
204
+ }
205
+ }
206
+ return { process: proc, initialization };
207
+ },
208
+ };
209
+ export const GoServer = {
210
+ id: "go",
211
+ name: "gopls",
212
+ extensions: [".go"],
213
+ root: createRootDetector(["go.mod", "go.sum"], [".pi-lens"]),
214
+ async spawn(root) {
215
+ const proc = await spawnWithInteractiveInstall("go", "gopls", [], { cwd: root }, () => launchLSP("gopls", [], { cwd: root }));
216
+ return proc ? { process: proc } : undefined;
217
+ },
218
+ };
219
+ export const RustServer = {
220
+ id: "rust",
221
+ name: "rust-analyzer",
222
+ extensions: [".rs"],
223
+ root: createRootDetector(["Cargo.toml", "Cargo.lock"], [".pi-lens"]),
224
+ async spawn(root) {
225
+ const proc = await spawnWithInteractiveInstall("rust", "rust-analyzer", [], { cwd: root }, () => launchLSP("rust-analyzer", [], { cwd: root }));
226
+ return proc ? { process: proc } : undefined;
227
+ },
228
+ };
229
+ export const RubyServer = {
230
+ id: "ruby",
231
+ name: "Ruby LSP",
232
+ extensions: [".rb", ".rake", ".gemspec", ".ru"],
233
+ root: createRootDetector(["Gemfile", ".ruby-version"], [".pi-lens"]),
234
+ async spawn(root) {
235
+ // Try ruby-lsp first, fall back to solargraph
236
+ try {
237
+ const proc = launchLSP("ruby-lsp", [], { cwd: root });
238
+ return { process: proc };
239
+ }
240
+ catch {
241
+ const proc = launchViaPackageManager("solargraph", ["stdio"], {
242
+ cwd: root,
243
+ });
244
+ return { process: proc };
245
+ }
246
+ },
247
+ };
248
+ export const PHPServer = {
249
+ id: "php",
250
+ name: "Intelephense",
251
+ extensions: [".php"],
252
+ root: createRootDetector(["composer.json", "composer.lock"], [".pi-lens"]),
253
+ async spawn(root) {
254
+ const proc = launchViaPackageManager("intelephense", ["--stdio"], {
255
+ cwd: root,
256
+ });
257
+ return {
258
+ process: proc,
259
+ initialization: { storagePath: path.join(__dirname, ".intelephense") },
260
+ };
261
+ },
262
+ };
263
+ export const CSharpServer = {
264
+ id: "csharp",
265
+ name: "csharp-ls",
266
+ extensions: [".cs"],
267
+ root: createRootDetector([".sln", ".csproj", ".slnx"], [".pi-lens"]),
268
+ async spawn(root) {
269
+ const proc = launchLSP("csharp-ls", [], { cwd: root });
270
+ return { process: proc };
271
+ },
272
+ };
273
+ export const FSharpServer = {
274
+ id: "fsharp",
275
+ name: "FSAutocomplete",
276
+ extensions: [".fs", ".fsi", ".fsx"],
277
+ root: createRootDetector([".sln", ".fsproj"], [".pi-lens"]),
278
+ async spawn(root) {
279
+ const proc = launchLSP("fsautocomplete", [], { cwd: root });
280
+ return { process: proc };
281
+ },
282
+ };
283
+ export const JavaServer = {
284
+ id: "java",
285
+ name: "JDT Language Server",
286
+ extensions: [".java"],
287
+ root: createRootDetector(["pom.xml", "build.gradle", ".classpath"], [".pi-lens"]),
288
+ async spawn(root) {
289
+ // JDTLS requires special handling - paths to launcher jar
290
+ const jdtlsPath = process.env.JDTLS_PATH || "jdtls";
291
+ const proc = launchLSP(jdtlsPath, [], { cwd: root });
292
+ return { process: proc };
293
+ },
294
+ };
295
+ export const KotlinServer = {
296
+ id: "kotlin",
297
+ name: "Kotlin Language Server",
298
+ extensions: [".kt", ".kts"],
299
+ root: createRootDetector(["build.gradle.kts", "build.gradle", "pom.xml"], [".pi-lens"]),
300
+ async spawn(root) {
301
+ const proc = launchLSP("kotlin-language-server", [], { cwd: root });
302
+ return { process: proc };
303
+ },
304
+ };
305
+ export const SwiftServer = {
306
+ id: "swift",
307
+ name: "SourceKit-LSP",
308
+ extensions: [".swift"],
309
+ root: createRootDetector(["Package.swift"], [".pi-lens"]),
310
+ async spawn(root) {
311
+ const proc = launchLSP("sourcekit-lsp", [], { cwd: root });
312
+ return { process: proc };
313
+ },
314
+ };
315
+ export const DartServer = {
316
+ id: "dart",
317
+ name: "Dart Analysis Server",
318
+ extensions: [".dart"],
319
+ root: createRootDetector(["pubspec.yaml"], [".pi-lens"]),
320
+ async spawn(root) {
321
+ const proc = launchLSP("dart", ["language-server", "--protocol=lsp"], {
322
+ cwd: root,
323
+ });
324
+ return { process: proc };
325
+ },
326
+ };
327
+ export const LuaServer = {
328
+ id: "lua",
329
+ name: "Lua Language Server",
330
+ extensions: [".lua"],
331
+ root: createRootDetector([".luarc.json", ".luacheckrc"], [".pi-lens"]),
332
+ async spawn(root) {
333
+ const proc = launchLSP("lua-language-server", [], { cwd: root });
334
+ return { process: proc };
335
+ },
336
+ };
337
+ export const CppServer = {
338
+ id: "cpp",
339
+ name: "clangd",
340
+ extensions: [".c", ".cpp", ".cc", ".cxx", ".h", ".hpp"],
341
+ root: createRootDetector([
342
+ "compile_commands.json",
343
+ ".clangd",
344
+ "CMakeLists.txt",
345
+ "Makefile",
346
+ ]),
347
+ async spawn(root) {
348
+ const proc = launchLSP("clangd", ["--background-index"], { cwd: root });
349
+ return { process: proc };
350
+ },
351
+ };
352
+ export const ZigServer = {
353
+ id: "zig",
354
+ name: "ZLS",
355
+ extensions: [".zig", ".zon"],
356
+ root: createRootDetector(["build.zig"], [".pi-lens"]),
357
+ async spawn(root) {
358
+ const proc = launchLSP("zls", [], { cwd: root });
359
+ return { process: proc };
360
+ },
361
+ };
362
+ export const HaskellServer = {
363
+ id: "haskell",
364
+ name: "Haskell Language Server",
365
+ extensions: [".hs", ".lhs"],
366
+ root: createRootDetector(["stack.yaml", "cabal.project", "*.cabal"], [".pi-lens"]),
367
+ async spawn(root) {
368
+ const proc = launchLSP("haskell-language-server-wrapper", ["--lsp"], {
369
+ cwd: root,
370
+ });
371
+ return { process: proc };
372
+ },
373
+ };
374
+ export const ElixirServer = {
375
+ id: "elixir",
376
+ name: "ElixirLS",
377
+ extensions: [".ex", ".exs"],
378
+ root: createRootDetector(["mix.exs"], [".pi-lens"]),
379
+ async spawn(root) {
380
+ const proc = launchLSP("elixir-ls", [], { cwd: root });
381
+ return { process: proc };
382
+ },
383
+ };
384
+ export const GleamServer = {
385
+ id: "gleam",
386
+ name: "Gleam LSP",
387
+ extensions: [".gleam"],
388
+ root: createRootDetector(["gleam.toml"], [".pi-lens"]),
389
+ async spawn(root) {
390
+ const proc = launchLSP("gleam", ["lsp"], { cwd: root });
391
+ return { process: proc };
392
+ },
393
+ };
394
+ export const OCamlServer = {
395
+ id: "ocaml",
396
+ name: "ocamllsp",
397
+ extensions: [".ml", ".mli"],
398
+ root: createRootDetector(["dune-project", "opam"], [".pi-lens"]),
399
+ async spawn(root) {
400
+ const proc = launchLSP("ocamllsp", [], { cwd: root });
401
+ return { process: proc };
402
+ },
403
+ };
404
+ export const ClojureServer = {
405
+ id: "clojure",
406
+ name: "Clojure LSP",
407
+ extensions: [".clj", ".cljs", ".cljc", ".edn"],
408
+ root: createRootDetector(["deps.edn", "project.clj"], [".pi-lens"]),
409
+ async spawn(root) {
410
+ const proc = launchLSP("clojure-lsp", [], { cwd: root });
411
+ return { process: proc };
412
+ },
413
+ };
414
+ export const TerraformServer = {
415
+ id: "terraform",
416
+ name: "Terraform LSP",
417
+ extensions: [".tf", ".tfvars"],
418
+ root: createRootDetector([".terraform.lock.hcl"], [".pi-lens"]),
419
+ async spawn(root) {
420
+ const proc = launchLSP("terraform-ls", ["serve"], { cwd: root });
421
+ return { process: proc };
422
+ },
423
+ };
424
+ export const NixServer = {
425
+ id: "nix",
426
+ name: "nixd",
427
+ extensions: [".nix"],
428
+ root: createRootDetector(["flake.nix"], [".pi-lens"]),
429
+ async spawn(root) {
430
+ const proc = launchLSP("nixd", [], { cwd: root });
431
+ return { process: proc };
432
+ },
433
+ };
434
+ export const BashServer = {
435
+ id: "bash",
436
+ name: "Bash Language Server",
437
+ extensions: [".sh", ".bash", ".zsh"],
438
+ root: async () => process.cwd(),
439
+ async spawn() {
440
+ const cwd = process.cwd();
441
+ const proc = await spawnWithInteractiveInstall("bash", "bash-language-server", ["start"], { cwd }, () => launchLSP("bash-language-server", ["start"], {}));
442
+ return proc ? { process: proc } : undefined;
443
+ },
444
+ };
445
+ export const DockerServer = {
446
+ id: "docker",
447
+ name: "Dockerfile Language Server",
448
+ extensions: [".dockerfile", "Dockerfile"],
449
+ root: async () => process.cwd(),
450
+ async spawn() {
451
+ // Use npx since it's not auto-installed
452
+ const proc = launchViaPackageManager("dockerfile-language-server-nodejs", ["--stdio"], {});
453
+ return { process: proc };
454
+ },
455
+ };
456
+ export const YamlServer = {
457
+ id: "yaml",
458
+ name: "YAML Language Server",
459
+ extensions: [".yaml", ".yml"],
460
+ root: async () => process.cwd(),
461
+ async spawn() {
462
+ const cwd = process.cwd();
463
+ const proc = await spawnWithInteractiveInstall("yaml", "yaml-language-server", ["--stdio"], { cwd }, () => launchLSP("yaml-language-server", ["--stdio"], {}));
464
+ return proc ? { process: proc } : undefined;
465
+ },
466
+ };
467
+ export const JsonServer = {
468
+ id: "json",
469
+ name: "VSCode JSON Language Server",
470
+ extensions: [".json", ".jsonc"],
471
+ root: async () => process.cwd(),
472
+ async spawn() {
473
+ const cwd = process.cwd();
474
+ const proc = await spawnWithInteractiveInstall("json", "vscode-json-languageserver", ["--stdio"], { cwd }, () => launchLSP("vscode-json-languageserver", ["--stdio"], {}));
475
+ return proc ? { process: proc } : undefined;
476
+ },
477
+ };
478
+ export const PrismaServer = {
479
+ id: "prisma",
480
+ name: "Prisma Language Server",
481
+ extensions: [".prisma"],
482
+ root: createRootDetector(["prisma/schema.prisma"], [".pi-lens"]),
483
+ async spawn(root) {
484
+ // Use npx since it's not auto-installed
485
+ const proc = launchViaPackageManager("@prisma/language-server", ["--stdio"], { cwd: root });
486
+ return { process: proc };
487
+ },
488
+ };
489
+ // --- Web Framework & Styling Servers ---
490
+ export const VueServer = {
491
+ id: "vue",
492
+ name: "Vue Language Server",
493
+ extensions: [".vue"],
494
+ root: createRootDetector([
495
+ "package-lock.json",
496
+ "bun.lockb",
497
+ "bun.lock",
498
+ "pnpm-lock.yaml",
499
+ "yarn.lock",
500
+ ], [".pi-lens"]),
501
+ async spawn(root) {
502
+ // Use npx since it's not auto-installed
503
+ const proc = launchViaPackageManager("@vue/language-server", ["--stdio"], {
504
+ cwd: root,
505
+ });
506
+ return { process: proc };
507
+ },
508
+ };
509
+ export const SvelteServer = {
510
+ id: "svelte",
511
+ name: "Svelte Language Server",
512
+ extensions: [".svelte"],
513
+ root: createRootDetector([
514
+ "package-lock.json",
515
+ "bun.lockb",
516
+ "bun.lock",
517
+ "pnpm-lock.yaml",
518
+ "yarn.lock",
519
+ ], [".pi-lens"]),
520
+ async spawn(root) {
521
+ // Use npx since it's not auto-installed
522
+ const proc = launchViaPackageManager("svelte-language-server", ["--stdio"], { cwd: root });
523
+ return { process: proc };
524
+ },
525
+ };
526
+ export const ESLintServer = {
527
+ id: "eslint",
528
+ name: "ESLint Language Server",
529
+ extensions: [".js", ".jsx", ".vue", ".svelte"], // Note: .ts/.tsx handled by TypeScript LSP + Biome
530
+ root: createRootDetector([
531
+ ".eslintrc",
532
+ ".eslintrc.json",
533
+ ".eslintrc.js",
534
+ "eslint.config.js",
535
+ "eslint.config.mjs",
536
+ "package.json",
537
+ ]),
538
+ async spawn(root) {
539
+ // Try via package manager (npx) since it's not auto-installed
540
+ try {
541
+ const proc = launchViaPackageManager("vscode-eslint-language-server", ["--stdio"], { cwd: root });
542
+ return { process: proc };
543
+ }
544
+ catch {
545
+ // Fall back to global install message
546
+ console.error("[lsp] ESLint Language Server not found. Install: npm install -g vscode-langservers-extracted");
547
+ return undefined;
548
+ }
549
+ },
550
+ };
551
+ export const CssServer = {
552
+ id: "css",
553
+ name: "CSS Language Server",
554
+ extensions: [".css", ".scss", ".sass", ".less"],
555
+ root: async () => process.cwd(),
556
+ async spawn() {
557
+ // Use npx since it's not auto-installed
558
+ const proc = launchViaPackageManager("vscode-css-languageserver", ["--stdio"], {});
559
+ return { process: proc };
560
+ },
561
+ };
562
+ // --- Registry ---
563
+ export const LSP_SERVERS = [
564
+ TypeScriptServer,
565
+ PythonServer,
566
+ GoServer,
567
+ RustServer,
568
+ RubyServer,
569
+ PHPServer,
570
+ CSharpServer,
571
+ FSharpServer,
572
+ JavaServer,
573
+ KotlinServer,
574
+ SwiftServer,
575
+ DartServer,
576
+ LuaServer,
577
+ CppServer,
578
+ ZigServer,
579
+ HaskellServer,
580
+ ElixirServer,
581
+ GleamServer,
582
+ OCamlServer,
583
+ ClojureServer,
584
+ TerraformServer,
585
+ NixServer,
586
+ BashServer,
587
+ DockerServer,
588
+ YamlServer,
589
+ JsonServer,
590
+ PrismaServer,
591
+ // Web frameworks & styling
592
+ VueServer,
593
+ SvelteServer,
594
+ ESLintServer,
595
+ CssServer,
596
+ ];
597
+ /**
598
+ * Get server for a file extension
599
+ */
600
+ export function getServerForExtension(ext) {
601
+ return LSP_SERVERS.find((server) => server.extensions.includes(ext));
602
+ }
603
+ /**
604
+ * Get server by ID
605
+ */
606
+ export function getServerById(id) {
607
+ return LSP_SERVERS.find((server) => server.id === id);
608
+ }
609
+ /**
610
+ * Get all servers for a file (may have multiple matches)
611
+ */
612
+ export function getServersForFile(filePath) {
613
+ const ext = path.extname(filePath).toLowerCase();
614
+ return LSP_SERVERS.filter((server) => server.extensions.includes(ext));
615
+ }