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,532 @@
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 "path";
10
+ import { launchLSP, launchViaPackageManager } from "./launch.js";
11
+ import { ensureTool, getToolEnvironment } from "../installer/index.js";
12
+ // --- Root Detection Helpers ---
13
+ import { fileURLToPath } from "url";
14
+ import { dirname } from "path";
15
+ const __filename = fileURLToPath(import.meta.url);
16
+ const __dirname = dirname(__filename);
17
+ /**
18
+ * Walk up the tree looking for project root markers
19
+ */
20
+ export function createRootDetector(includePatterns, excludePatterns) {
21
+ return async (file) => {
22
+ let currentDir = path.dirname(file);
23
+ const root = path.parse(currentDir).root;
24
+ while (currentDir !== root) {
25
+ // Check exclude patterns first
26
+ if (excludePatterns) {
27
+ for (const pattern of excludePatterns) {
28
+ const checkPath = path.join(currentDir, pattern);
29
+ try {
30
+ const stat = await import("fs/promises").then(fs => fs.stat(checkPath));
31
+ if (stat)
32
+ return undefined; // Excluded
33
+ }
34
+ catch { /* not found */ }
35
+ }
36
+ }
37
+ // Check include patterns
38
+ for (const pattern of includePatterns) {
39
+ const checkPath = path.join(currentDir, pattern);
40
+ try {
41
+ const stat = await import("fs/promises").then(fs => fs.stat(checkPath));
42
+ if (stat)
43
+ return currentDir;
44
+ }
45
+ catch { /* not found */ }
46
+ }
47
+ currentDir = path.dirname(currentDir);
48
+ }
49
+ return undefined;
50
+ };
51
+ }
52
+ // --- Server Definitions ---
53
+ export const TypeScriptServer = {
54
+ id: "typescript",
55
+ name: "TypeScript Language Server",
56
+ extensions: [".ts", ".tsx", ".js", ".jsx", ".mjs", ".cjs", ".mts", ".cts"],
57
+ root: createRootDetector([
58
+ "package-lock.json",
59
+ "bun.lockb",
60
+ "bun.lock",
61
+ "pnpm-lock.yaml",
62
+ "yarn.lock",
63
+ "package.json",
64
+ ]),
65
+ async spawn(root) {
66
+ const path = await import("path");
67
+ const fs = await import("fs/promises");
68
+ // Find typescript-language-server - prefer local project version
69
+ let lspPath;
70
+ const localLsp = path.join(root, "node_modules", ".bin", "typescript-language-server");
71
+ const localLspCmd = path.join(root, "node_modules", ".bin", "typescript-language-server.cmd");
72
+ // Check for local version first (Windows .cmd first, then Unix)
73
+ for (const checkPath of [localLspCmd, localLsp]) {
74
+ try {
75
+ await fs.access(checkPath);
76
+ lspPath = checkPath;
77
+ break;
78
+ }
79
+ catch { /* not found */ }
80
+ }
81
+ // Fall back to auto-installed version
82
+ if (!lspPath) {
83
+ lspPath = await ensureTool("typescript-language-server");
84
+ if (!lspPath) {
85
+ console.error("[lsp] typescript-language-server not found");
86
+ return undefined;
87
+ }
88
+ }
89
+ // Find tsserver.js path (needed for TypeScript LSP initialization)
90
+ // Check relative to the LSP path first, then project root
91
+ let tsserverPath;
92
+ const tsserverCandidates = [
93
+ // Relative to LSP binary (for locally installed)
94
+ path.join(path.dirname(lspPath), "..", "typescript", "lib", "tsserver.js"),
95
+ // Project root
96
+ path.join(root, "node_modules", "typescript", "lib", "tsserver.js"),
97
+ // Current working directory
98
+ path.join(process.cwd(), "node_modules", "typescript", "lib", "tsserver.js"),
99
+ ];
100
+ for (const checkPath of tsserverCandidates) {
101
+ try {
102
+ await fs.access(checkPath);
103
+ tsserverPath = checkPath;
104
+ break;
105
+ }
106
+ catch { /* not found */ }
107
+ }
108
+ // Use absolute path and proper environment
109
+ const env = await getToolEnvironment();
110
+ const proc = launchLSP(lspPath, ["--stdio"], {
111
+ cwd: root,
112
+ env: {
113
+ ...env,
114
+ TSSERVER_PATH: tsserverPath,
115
+ }
116
+ });
117
+ return {
118
+ process: proc,
119
+ initialization: tsserverPath ? { tsserver: { path: tsserverPath } } : undefined
120
+ };
121
+ },
122
+ };
123
+ export const PythonServer = {
124
+ id: "python",
125
+ name: "Pyright Language Server",
126
+ extensions: [".py", ".pyi"],
127
+ root: createRootDetector([
128
+ "pyproject.toml",
129
+ "setup.py",
130
+ "setup.cfg",
131
+ "requirements.txt",
132
+ "Pipfile",
133
+ "poetry.lock",
134
+ ]),
135
+ async spawn(root) {
136
+ // Ensure pyright is installed
137
+ const toolPath = await ensureTool("pyright");
138
+ if (!toolPath) {
139
+ console.error("[lsp] pyright not found and could not be installed");
140
+ return undefined;
141
+ }
142
+ const env = await getToolEnvironment();
143
+ const proc = launchLSP("pyright-langserver", ["--stdio"], { cwd: root, env });
144
+ // Detect virtual environment
145
+ const initialization = {};
146
+ const venvPaths = [
147
+ path.join(root, ".venv"),
148
+ path.join(root, "venv"),
149
+ process.env.VIRTUAL_ENV,
150
+ ].filter(Boolean);
151
+ for (const venv of venvPaths) {
152
+ if (!venv)
153
+ continue;
154
+ try {
155
+ const pythonPath = process.platform === "win32"
156
+ ? path.join(venv, "Scripts", "python.exe")
157
+ : path.join(venv, "bin", "python");
158
+ await import("fs/promises").then(fs => fs.access(pythonPath));
159
+ initialization.python = { pythonPath };
160
+ break;
161
+ }
162
+ catch { /* not found */ }
163
+ }
164
+ return { process: proc, initialization };
165
+ },
166
+ };
167
+ export const GoServer = {
168
+ id: "go",
169
+ name: "gopls",
170
+ extensions: [".go"],
171
+ root: createRootDetector(["go.mod", "go.sum"]),
172
+ async spawn(root) {
173
+ const proc = launchLSP("gopls", [], { cwd: root });
174
+ return { process: proc };
175
+ },
176
+ };
177
+ export const RustServer = {
178
+ id: "rust",
179
+ name: "rust-analyzer",
180
+ extensions: [".rs"],
181
+ root: createRootDetector(["Cargo.toml", "Cargo.lock"]),
182
+ async spawn(root) {
183
+ const proc = launchLSP("rust-analyzer", [], { cwd: root });
184
+ return { process: proc };
185
+ },
186
+ };
187
+ export const RubyServer = {
188
+ id: "ruby",
189
+ name: "Ruby LSP",
190
+ extensions: [".rb", ".rake", ".gemspec", ".ru"],
191
+ root: createRootDetector(["Gemfile", ".ruby-version"]),
192
+ async spawn(root) {
193
+ // Try ruby-lsp first, fall back to solargraph
194
+ try {
195
+ const proc = launchLSP("ruby-lsp", [], { cwd: root });
196
+ return { process: proc };
197
+ }
198
+ catch {
199
+ const proc = launchViaPackageManager("solargraph", ["stdio"], { cwd: root });
200
+ return { process: proc };
201
+ }
202
+ },
203
+ };
204
+ export const PHPServer = {
205
+ id: "php",
206
+ name: "Intelephense",
207
+ extensions: [".php"],
208
+ root: createRootDetector(["composer.json", "composer.lock"]),
209
+ async spawn(root) {
210
+ const proc = launchViaPackageManager("intelephense", ["--stdio"], { cwd: root });
211
+ return { process: proc, initialization: { storagePath: path.join(__dirname, ".intelephense") } };
212
+ },
213
+ };
214
+ export const CSharpServer = {
215
+ id: "csharp",
216
+ name: "csharp-ls",
217
+ extensions: [".cs"],
218
+ root: createRootDetector([".sln", ".csproj", ".slnx"]),
219
+ async spawn(root) {
220
+ const proc = launchLSP("csharp-ls", [], { cwd: root });
221
+ return { process: proc };
222
+ },
223
+ };
224
+ export const FSharpServer = {
225
+ id: "fsharp",
226
+ name: "FSAutocomplete",
227
+ extensions: [".fs", ".fsi", ".fsx"],
228
+ root: createRootDetector([".sln", ".fsproj"]),
229
+ async spawn(root) {
230
+ const proc = launchLSP("fsautocomplete", [], { cwd: root });
231
+ return { process: proc };
232
+ },
233
+ };
234
+ export const JavaServer = {
235
+ id: "java",
236
+ name: "JDT Language Server",
237
+ extensions: [".java"],
238
+ root: createRootDetector(["pom.xml", "build.gradle", ".classpath"]),
239
+ async spawn(root) {
240
+ // JDTLS requires special handling - paths to launcher jar
241
+ const jdtlsPath = process.env.JDTLS_PATH || "jdtls";
242
+ const proc = launchLSP(jdtlsPath, [], { cwd: root });
243
+ return { process: proc };
244
+ },
245
+ };
246
+ export const KotlinServer = {
247
+ id: "kotlin",
248
+ name: "Kotlin Language Server",
249
+ extensions: [".kt", ".kts"],
250
+ root: createRootDetector(["build.gradle.kts", "build.gradle", "pom.xml"]),
251
+ async spawn(root) {
252
+ const proc = launchLSP("kotlin-language-server", [], { cwd: root });
253
+ return { process: proc };
254
+ },
255
+ };
256
+ export const SwiftServer = {
257
+ id: "swift",
258
+ name: "SourceKit-LSP",
259
+ extensions: [".swift"],
260
+ root: createRootDetector(["Package.swift"]),
261
+ async spawn(root) {
262
+ const proc = launchLSP("sourcekit-lsp", [], { cwd: root });
263
+ return { process: proc };
264
+ },
265
+ };
266
+ export const DartServer = {
267
+ id: "dart",
268
+ name: "Dart Analysis Server",
269
+ extensions: [".dart"],
270
+ root: createRootDetector(["pubspec.yaml"]),
271
+ async spawn(root) {
272
+ const proc = launchLSP("dart", ["language-server", "--protocol=lsp"], { cwd: root });
273
+ return { process: proc };
274
+ },
275
+ };
276
+ export const LuaServer = {
277
+ id: "lua",
278
+ name: "Lua Language Server",
279
+ extensions: [".lua"],
280
+ root: createRootDetector([".luarc.json", ".luacheckrc"]),
281
+ async spawn(root) {
282
+ const proc = launchLSP("lua-language-server", [], { cwd: root });
283
+ return { process: proc };
284
+ },
285
+ };
286
+ export const CppServer = {
287
+ id: "cpp",
288
+ name: "clangd",
289
+ extensions: [".c", ".cpp", ".cc", ".cxx", ".h", ".hpp"],
290
+ root: createRootDetector([
291
+ "compile_commands.json",
292
+ ".clangd",
293
+ "CMakeLists.txt",
294
+ "Makefile",
295
+ ]),
296
+ async spawn(root) {
297
+ const proc = launchLSP("clangd", ["--background-index"], { cwd: root });
298
+ return { process: proc };
299
+ },
300
+ };
301
+ export const ZigServer = {
302
+ id: "zig",
303
+ name: "ZLS",
304
+ extensions: [".zig", ".zon"],
305
+ root: createRootDetector(["build.zig"]),
306
+ async spawn(root) {
307
+ const proc = launchLSP("zls", [], { cwd: root });
308
+ return { process: proc };
309
+ },
310
+ };
311
+ export const HaskellServer = {
312
+ id: "haskell",
313
+ name: "Haskell Language Server",
314
+ extensions: [".hs", ".lhs"],
315
+ root: createRootDetector(["stack.yaml", "cabal.project", "*.cabal"]),
316
+ async spawn(root) {
317
+ const proc = launchLSP("haskell-language-server-wrapper", ["--lsp"], { cwd: root });
318
+ return { process: proc };
319
+ },
320
+ };
321
+ export const ElixirServer = {
322
+ id: "elixir",
323
+ name: "ElixirLS",
324
+ extensions: [".ex", ".exs"],
325
+ root: createRootDetector(["mix.exs"]),
326
+ async spawn(root) {
327
+ const proc = launchLSP("elixir-ls", [], { cwd: root });
328
+ return { process: proc };
329
+ },
330
+ };
331
+ export const GleamServer = {
332
+ id: "gleam",
333
+ name: "Gleam LSP",
334
+ extensions: [".gleam"],
335
+ root: createRootDetector(["gleam.toml"]),
336
+ async spawn(root) {
337
+ const proc = launchLSP("gleam", ["lsp"], { cwd: root });
338
+ return { process: proc };
339
+ },
340
+ };
341
+ export const OCamlServer = {
342
+ id: "ocaml",
343
+ name: "ocamllsp",
344
+ extensions: [".ml", ".mli"],
345
+ root: createRootDetector(["dune-project", "opam"]),
346
+ async spawn(root) {
347
+ const proc = launchLSP("ocamllsp", [], { cwd: root });
348
+ return { process: proc };
349
+ },
350
+ };
351
+ export const ClojureServer = {
352
+ id: "clojure",
353
+ name: "Clojure LSP",
354
+ extensions: [".clj", ".cljs", ".cljc", ".edn"],
355
+ root: createRootDetector(["deps.edn", "project.clj"]),
356
+ async spawn(root) {
357
+ const proc = launchLSP("clojure-lsp", [], { cwd: root });
358
+ return { process: proc };
359
+ },
360
+ };
361
+ export const TerraformServer = {
362
+ id: "terraform",
363
+ name: "Terraform LSP",
364
+ extensions: [".tf", ".tfvars"],
365
+ root: createRootDetector([".terraform.lock.hcl"]),
366
+ async spawn(root) {
367
+ const proc = launchLSP("terraform-ls", ["serve"], { cwd: root });
368
+ return { process: proc };
369
+ },
370
+ };
371
+ export const NixServer = {
372
+ id: "nix",
373
+ name: "nixd",
374
+ extensions: [".nix"],
375
+ root: createRootDetector(["flake.nix"]),
376
+ async spawn(root) {
377
+ const proc = launchLSP("nixd", [], { cwd: root });
378
+ return { process: proc };
379
+ },
380
+ };
381
+ export const BashServer = {
382
+ id: "bash",
383
+ name: "Bash Language Server",
384
+ extensions: [".sh", ".bash", ".zsh"],
385
+ root: async () => process.cwd(),
386
+ async spawn() {
387
+ const proc = launchViaPackageManager("bash-language-server", ["start"], {});
388
+ return { process: proc };
389
+ },
390
+ };
391
+ export const DockerServer = {
392
+ id: "docker",
393
+ name: "Dockerfile Language Server",
394
+ extensions: [".dockerfile", "Dockerfile"],
395
+ root: async () => process.cwd(),
396
+ async spawn() {
397
+ const proc = launchViaPackageManager("dockerfile-language-server-nodejs", ["--stdio"], {});
398
+ return { process: proc };
399
+ },
400
+ };
401
+ export const YamlServer = {
402
+ id: "yaml",
403
+ name: "YAML Language Server",
404
+ extensions: [".yaml", ".yml"],
405
+ root: async () => process.cwd(),
406
+ async spawn() {
407
+ const proc = launchViaPackageManager("yaml-language-server", ["--stdio"], {});
408
+ return { process: proc };
409
+ },
410
+ };
411
+ export const JsonServer = {
412
+ id: "json",
413
+ name: "VSCode JSON Language Server",
414
+ extensions: [".json", ".jsonc"],
415
+ root: async () => process.cwd(),
416
+ async spawn() {
417
+ const proc = launchViaPackageManager("vscode-json-languageserver", ["--stdio"], {});
418
+ return { process: proc };
419
+ },
420
+ };
421
+ export const PrismaServer = {
422
+ id: "prisma",
423
+ name: "Prisma Language Server",
424
+ extensions: [".prisma"],
425
+ root: createRootDetector(["prisma/schema.prisma"]),
426
+ async spawn(root) {
427
+ const proc = launchViaPackageManager("@prisma/language-server", ["--stdio"], { cwd: root });
428
+ return { process: proc };
429
+ },
430
+ };
431
+ // --- Web Framework & Styling Servers ---
432
+ export const VueServer = {
433
+ id: "vue",
434
+ name: "Vue Language Server",
435
+ extensions: [".vue"],
436
+ root: createRootDetector(["package-lock.json", "bun.lockb", "bun.lock", "pnpm-lock.yaml", "yarn.lock"]),
437
+ async spawn(root) {
438
+ const proc = launchViaPackageManager("@vue/language-server", ["--stdio"], { cwd: root });
439
+ return { process: proc };
440
+ },
441
+ };
442
+ export const SvelteServer = {
443
+ id: "svelte",
444
+ name: "Svelte Language Server",
445
+ extensions: [".svelte"],
446
+ root: createRootDetector(["package-lock.json", "bun.lockb", "bun.lock", "pnpm-lock.yaml", "yarn.lock"]),
447
+ async spawn(root) {
448
+ const proc = launchViaPackageManager("svelte-language-server", ["--stdio"], { cwd: root });
449
+ return { process: proc };
450
+ },
451
+ };
452
+ export const ESLintServer = {
453
+ id: "eslint",
454
+ name: "ESLint Language Server",
455
+ extensions: [".js", ".jsx", ".ts", ".tsx", ".vue", ".svelte"],
456
+ root: createRootDetector([
457
+ ".eslintrc",
458
+ ".eslintrc.json",
459
+ ".eslintrc.js",
460
+ "eslint.config.js",
461
+ "eslint.config.mjs",
462
+ "package.json",
463
+ ]),
464
+ async spawn(root) {
465
+ const proc = launchViaPackageManager("vscode-eslint", ["--stdio"], { cwd: root });
466
+ return { process: proc };
467
+ },
468
+ };
469
+ export const CssServer = {
470
+ id: "css",
471
+ name: "CSS Language Server",
472
+ extensions: [".css", ".scss", ".sass", ".less"],
473
+ root: async () => process.cwd(),
474
+ async spawn() {
475
+ const proc = launchViaPackageManager("vscode-css-languageserver", ["--stdio"], {});
476
+ return { process: proc };
477
+ },
478
+ };
479
+ // --- Registry ---
480
+ export const LSP_SERVERS = [
481
+ TypeScriptServer,
482
+ PythonServer,
483
+ GoServer,
484
+ RustServer,
485
+ RubyServer,
486
+ PHPServer,
487
+ CSharpServer,
488
+ FSharpServer,
489
+ JavaServer,
490
+ KotlinServer,
491
+ SwiftServer,
492
+ DartServer,
493
+ LuaServer,
494
+ CppServer,
495
+ ZigServer,
496
+ HaskellServer,
497
+ ElixirServer,
498
+ GleamServer,
499
+ OCamlServer,
500
+ ClojureServer,
501
+ TerraformServer,
502
+ NixServer,
503
+ BashServer,
504
+ DockerServer,
505
+ YamlServer,
506
+ JsonServer,
507
+ PrismaServer,
508
+ // Web frameworks & styling
509
+ VueServer,
510
+ SvelteServer,
511
+ ESLintServer,
512
+ CssServer,
513
+ ];
514
+ /**
515
+ * Get server for a file extension
516
+ */
517
+ export function getServerForExtension(ext) {
518
+ return LSP_SERVERS.find((server) => server.extensions.includes(ext));
519
+ }
520
+ /**
521
+ * Get server by ID
522
+ */
523
+ export function getServerById(id) {
524
+ return LSP_SERVERS.find((server) => server.id === id);
525
+ }
526
+ /**
527
+ * Get all servers for a file (may have multiple matches)
528
+ */
529
+ export function getServersForFile(filePath) {
530
+ const ext = path.extname(filePath).toLowerCase();
531
+ return LSP_SERVERS.filter((server) => server.extensions.includes(ext));
532
+ }
@@ -0,0 +1,10 @@
1
+ /**
2
+ * LSP Module for pi-lens
3
+ *
4
+ * Multi-language LSP client supporting 40+ languages.
5
+ */
6
+ export * from "./client.js";
7
+ export * from "./server.js";
8
+ export * from "./language.js";
9
+ export * from "./launch.js";
10
+ export * from "./index.js";
@@ -0,0 +1,11 @@
1
+ /**
2
+ * LSP Module for pi-lens
3
+ *
4
+ * Multi-language LSP client supporting 40+ languages.
5
+ */
6
+
7
+ export * from "./client.js";
8
+ export * from "./server.js";
9
+ export * from "./language.js";
10
+ export * from "./launch.js";
11
+ export * from "./index.js";
@@ -0,0 +1,48 @@
1
+ /**
2
+ * Path utilities for pi-lens
3
+ *
4
+ * Handles cross-platform path normalization, particularly
5
+ * Windows case-insensitivity issues when using paths as Map keys.
6
+ */
7
+ import { fileURLToPath, pathToFileURL } from "url";
8
+ /**
9
+ * Normalize a file path for consistent Map key usage.
10
+ * On Windows: converts to lowercase (case-insensitive filesystem)
11
+ * On other platforms: returns path as-is (case-sensitive filesystem)
12
+ *
13
+ * This ensures that "C:\foo.ts" and "c:\foo.ts" resolve to the same key.
14
+ */
15
+ export function normalizeFilePath(filePath) {
16
+ if (process.platform === "win32") {
17
+ return filePath.toLowerCase();
18
+ }
19
+ return filePath;
20
+ }
21
+ /**
22
+ * Convert a file:// URI to a normalized path.
23
+ * Handles URL decoding and Windows drive letter normalization.
24
+ */
25
+ export function uriToPath(uri) {
26
+ try {
27
+ const filePath = fileURLToPath(uri);
28
+ return normalizeFilePath(filePath);
29
+ }
30
+ catch {
31
+ // Not a valid file:// URI, treat as plain path
32
+ return normalizeFilePath(uri);
33
+ }
34
+ }
35
+ /**
36
+ * Convert a path to a file:// URI.
37
+ * Does NOT normalize the path - URIs preserve original casing.
38
+ */
39
+ export function pathToUri(filePath) {
40
+ return pathToFileURL(filePath).href;
41
+ }
42
+ /**
43
+ * Normalize a Map key lookup for file paths.
44
+ * Use this when getting/setting values in Maps that use file paths as keys.
45
+ */
46
+ export function normalizeMapKey(filePath) {
47
+ return normalizeFilePath(filePath);
48
+ }
@@ -0,0 +1,52 @@
1
+ /**
2
+ * Path utilities for pi-lens
3
+ *
4
+ * Handles cross-platform path normalization, particularly
5
+ * Windows case-insensitivity issues when using paths as Map keys.
6
+ */
7
+
8
+ import { fileURLToPath, pathToFileURL } from "url";
9
+
10
+ /**
11
+ * Normalize a file path for consistent Map key usage.
12
+ * On Windows: converts to lowercase (case-insensitive filesystem)
13
+ * On other platforms: returns path as-is (case-sensitive filesystem)
14
+ *
15
+ * This ensures that "C:\foo.ts" and "c:\foo.ts" resolve to the same key.
16
+ */
17
+ export function normalizeFilePath(filePath: string): string {
18
+ if (process.platform === "win32") {
19
+ return filePath.toLowerCase();
20
+ }
21
+ return filePath;
22
+ }
23
+
24
+ /**
25
+ * Convert a file:// URI to a normalized path.
26
+ * Handles URL decoding and Windows drive letter normalization.
27
+ */
28
+ export function uriToPath(uri: string): string {
29
+ try {
30
+ const filePath = fileURLToPath(uri);
31
+ return normalizeFilePath(filePath);
32
+ } catch {
33
+ // Not a valid file:// URI, treat as plain path
34
+ return normalizeFilePath(uri);
35
+ }
36
+ }
37
+
38
+ /**
39
+ * Convert a path to a file:// URI.
40
+ * Does NOT normalize the path - URIs preserve original casing.
41
+ */
42
+ export function pathToUri(filePath: string): string {
43
+ return pathToFileURL(filePath).href;
44
+ }
45
+
46
+ /**
47
+ * Normalize a Map key lookup for file paths.
48
+ * Use this when getting/setting values in Maps that use file paths as keys.
49
+ */
50
+ export function normalizeMapKey(filePath: string): string {
51
+ return normalizeFilePath(filePath);
52
+ }