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,508 @@
1
+ /**
2
+ * Formatter Definitions for pi-lens
3
+ *
4
+ * Auto-detects formatters based on:
5
+ * - Config files (biome.json, .prettierrc, etc.)
6
+ * - Dependencies (package.json, requirements.txt, etc.)
7
+ * - Binary availability (which/where)
8
+ *
9
+ * Inspired by OpenCode's formatter.ts pattern
10
+ */
11
+
12
+ import * as fs from "node:fs/promises";
13
+ import * as path from "node:path";
14
+ import { safeSpawn } from "./safe-spawn.js";
15
+
16
+ // --- Types ---
17
+
18
+ export interface FormatterInfo {
19
+ name: string;
20
+ command: string[]; // Command with $FILE placeholder
21
+ extensions: string[];
22
+ /** Detect if this formatter should be used for a project */
23
+ detect(cwd: string): Promise<boolean>;
24
+ }
25
+
26
+ export interface FormatterResult {
27
+ success: boolean;
28
+ changed: boolean;
29
+ error?: string;
30
+ }
31
+
32
+ // --- Utility Functions ---
33
+
34
+ async function fileExists(filePath: string): Promise<boolean> {
35
+ try {
36
+ await fs.access(filePath);
37
+ return true;
38
+ } catch {
39
+ return false;
40
+ }
41
+ }
42
+
43
+ async function findUp(
44
+ targets: string[],
45
+ startDir: string,
46
+ stopDir: string = path.parse(startDir).root,
47
+ ): Promise<string[]> {
48
+ const found: string[] = [];
49
+ let currentDir = startDir;
50
+
51
+ while (currentDir !== stopDir) {
52
+ for (const target of targets) {
53
+ const checkPath = path.join(currentDir, target);
54
+ if (await fileExists(checkPath)) {
55
+ found.push(checkPath);
56
+ }
57
+ }
58
+ const parent = path.dirname(currentDir);
59
+ if (parent === currentDir) break;
60
+ currentDir = parent;
61
+ }
62
+
63
+ return found;
64
+ }
65
+
66
+ async function readJson(filePath: string): Promise<unknown> {
67
+ try {
68
+ const content = await fs.readFile(filePath, "utf-8");
69
+ return JSON.parse(content);
70
+ } catch {
71
+ return {};
72
+ }
73
+ }
74
+
75
+ async function which(command: string): Promise<string | null> {
76
+ const result = safeSpawn(
77
+ process.platform === "win32" ? "where" : "which",
78
+ [command],
79
+ { timeout: 5000 },
80
+ );
81
+ if (result.error || result.status !== 0) return null;
82
+ return result.stdout?.trim().split("\n")[0] ?? null;
83
+ }
84
+
85
+ // --- Formatter Definitions ---
86
+
87
+ export const biomeFormatter: FormatterInfo = {
88
+ name: "biome",
89
+ command: ["npx", "@biomejs/biome", "format", "--write", "$FILE"],
90
+ extensions: [
91
+ ".js",
92
+ ".jsx",
93
+ ".mjs",
94
+ ".cjs",
95
+ ".ts",
96
+ ".tsx",
97
+ ".mts",
98
+ ".cts",
99
+ ".json",
100
+ ".jsonc",
101
+ ".css",
102
+ ".scss",
103
+ ".sass",
104
+ ".vue",
105
+ ".svelte",
106
+ ".html",
107
+ ".htm",
108
+ ],
109
+ async detect(cwd: string) {
110
+ const configs = ["biome.json", "biome.jsonc"];
111
+ const found = await findUp(configs, cwd);
112
+ if (found.length > 0) return true;
113
+
114
+ // Also check if biome is in package.json devDependencies
115
+ const pkgPath = path.join(cwd, "package.json");
116
+ if (await fileExists(pkgPath)) {
117
+ const pkg = (await readJson(pkgPath)) as {
118
+ devDependencies?: Record<string, string>;
119
+ };
120
+ if (pkg.devDependencies?.["@biomejs/biome"]) return true;
121
+ }
122
+
123
+ return false;
124
+ },
125
+ };
126
+
127
+ export const prettierFormatter: FormatterInfo = {
128
+ name: "prettier",
129
+ command: ["npx", "prettier", "--write", "$FILE"],
130
+ extensions: [
131
+ ".js",
132
+ ".jsx",
133
+ ".mjs",
134
+ ".cjs",
135
+ ".ts",
136
+ ".tsx",
137
+ ".mts",
138
+ ".cts",
139
+ ".json",
140
+ ".jsonc",
141
+ ".css",
142
+ ".scss",
143
+ ".sass",
144
+ ".less",
145
+ ".vue",
146
+ ".svelte",
147
+ ".html",
148
+ ".htm",
149
+ ".md",
150
+ ".mdx",
151
+ ".yaml",
152
+ ".yml",
153
+ ".graphql",
154
+ ".gql",
155
+ ],
156
+ async detect(cwd: string) {
157
+ // Check for prettier config files
158
+ const configs = [
159
+ ".prettierrc",
160
+ ".prettierrc.json",
161
+ ".prettierrc.yml",
162
+ ".prettierrc.yaml",
163
+ ".prettierrc.js",
164
+ ".prettierrc.cjs",
165
+ "prettier.config.js",
166
+ "prettier.config.cjs",
167
+ ];
168
+ const found = await findUp(configs, cwd);
169
+ if (found.length > 0) return true;
170
+
171
+ // Check package.json
172
+ const pkgPath = path.join(cwd, "package.json");
173
+ if (await fileExists(pkgPath)) {
174
+ const pkg = (await readJson(pkgPath)) as {
175
+ devDependencies?: Record<string, string>;
176
+ dependencies?: Record<string, string>;
177
+ prettier?: unknown;
178
+ };
179
+ if (pkg.devDependencies?.prettier || pkg.dependencies?.prettier) {
180
+ return true;
181
+ }
182
+ // Also check if "prettier" field exists in package.json
183
+ if (pkg.prettier !== undefined) return true;
184
+ }
185
+
186
+ return false;
187
+ },
188
+ };
189
+
190
+ export const ruffFormatter: FormatterInfo = {
191
+ name: "ruff",
192
+ command: ["ruff", "format", "$FILE"],
193
+ extensions: [".py", ".pyi"],
194
+ async detect(cwd: string) {
195
+ // Check for ruff config
196
+ const configs = ["pyproject.toml", "ruff.toml", ".ruff.toml"];
197
+ const found = await findUp(configs, cwd);
198
+
199
+ for (const configPath of found) {
200
+ if (configPath.endsWith("pyproject.toml")) {
201
+ const content = await fs.readFile(configPath, "utf-8");
202
+ if (content.includes("[tool.ruff]")) return true;
203
+ } else {
204
+ return true; // ruff.toml or .ruff.toml found
205
+ }
206
+ }
207
+
208
+ // Check if ruff in requirements
209
+ const deps = ["requirements.txt", "pyproject.toml", "Pipfile"];
210
+ for (const dep of deps) {
211
+ const depPath = path.join(cwd, dep);
212
+ if (await fileExists(depPath)) {
213
+ const content = await fs.readFile(depPath, "utf-8");
214
+ if (content.includes("ruff")) return true;
215
+ }
216
+ }
217
+
218
+ // Check if ruff binary available and no other Python formatter detected
219
+ const hasRuff = (await which("ruff")) !== null;
220
+ if (hasRuff) {
221
+ // Prefer ruff if no black config found
222
+ const blackFound = await findUp(["pyproject.toml"], cwd);
223
+ for (const p of blackFound) {
224
+ const content = await fs.readFile(p, "utf-8");
225
+ if (content.includes("[tool.black]")) return false; // Prefer black if configured
226
+ }
227
+ return true;
228
+ }
229
+
230
+ return false;
231
+ },
232
+ };
233
+
234
+ export const blackFormatter: FormatterInfo = {
235
+ name: "black",
236
+ command: ["black", "$FILE"],
237
+ extensions: [".py", ".pyi"],
238
+ async detect(cwd: string) {
239
+ // Check for black config in pyproject.toml
240
+ const configs = ["pyproject.toml"];
241
+ const found = await findUp(configs, cwd);
242
+ for (const configPath of found) {
243
+ const content = await fs.readFile(configPath, "utf-8");
244
+ if (content.includes("[tool.black]")) return true;
245
+ }
246
+
247
+ // Check if black in requirements
248
+ const deps = ["requirements.txt", "pyproject.toml", "Pipfile"];
249
+ for (const dep of deps) {
250
+ const depPath = path.join(cwd, dep);
251
+ if (await fileExists(depPath)) {
252
+ const content = await fs.readFile(depPath, "utf-8");
253
+ if (content.toLowerCase().includes("black")) return true;
254
+ }
255
+ }
256
+
257
+ return false;
258
+ },
259
+ };
260
+
261
+ export const gofmtFormatter: FormatterInfo = {
262
+ name: "gofmt",
263
+ command: ["gofmt", "-w", "$FILE"],
264
+ extensions: [".go"],
265
+ async detect(_cwd: string) {
266
+ return (await which("gofmt")) !== null;
267
+ },
268
+ };
269
+
270
+ export const rustfmtFormatter: FormatterInfo = {
271
+ name: "rustfmt",
272
+ command: ["rustfmt", "$FILE"],
273
+ extensions: [".rs"],
274
+ async detect(_cwd: string) {
275
+ return (await which("rustfmt")) !== null;
276
+ },
277
+ };
278
+
279
+ export const zigFormatter: FormatterInfo = {
280
+ name: "zig",
281
+ command: ["zig", "fmt", "$FILE"],
282
+ extensions: [".zig", ".zon"],
283
+ async detect(_cwd: string) {
284
+ return (await which("zig")) !== null;
285
+ },
286
+ };
287
+
288
+ export const dartFormatter: FormatterInfo = {
289
+ name: "dart",
290
+ command: ["dart", "format", "$FILE"],
291
+ extensions: [".dart"],
292
+ async detect(_cwd: string) {
293
+ return (await which("dart")) !== null;
294
+ },
295
+ };
296
+
297
+ export const shfmtFormatter: FormatterInfo = {
298
+ name: "shfmt",
299
+ command: ["shfmt", "-w", "$FILE"],
300
+ extensions: [".sh", ".bash"],
301
+ async detect(_cwd: string) {
302
+ return (await which("shfmt")) !== null;
303
+ },
304
+ };
305
+
306
+ export const nixfmtFormatter: FormatterInfo = {
307
+ name: "nixfmt",
308
+ command: ["nixfmt", "$FILE"],
309
+ extensions: [".nix"],
310
+ async detect(_cwd: string) {
311
+ return (await which("nixfmt")) !== null;
312
+ },
313
+ };
314
+
315
+ export const mixFormatter: FormatterInfo = {
316
+ name: "mix",
317
+ command: ["mix", "format", "$FILE"],
318
+ extensions: [".ex", ".exs", ".eex", ".heex", ".leex"],
319
+ async detect(_cwd: string) {
320
+ return (await which("mix")) !== null;
321
+ },
322
+ };
323
+
324
+ export const ocamlformatFormatter: FormatterInfo = {
325
+ name: "ocamlformat",
326
+ command: ["ocamlformat", "-i", "$FILE"],
327
+ extensions: [".ml", ".mli"],
328
+ async detect(cwd: string) {
329
+ const hasBinary = (await which("ocamlformat")) !== null;
330
+ if (!hasBinary) return false;
331
+ const configs = [".ocamlformat"];
332
+ const found = await findUp(configs, cwd);
333
+ return found.length > 0;
334
+ },
335
+ };
336
+
337
+ export const clangFormatFormatter: FormatterInfo = {
338
+ name: "clang-format",
339
+ command: ["clang-format", "-i", "$FILE"],
340
+ extensions: [".c", ".cc", ".cpp", ".cxx", ".h", ".hpp", ".ino"],
341
+ async detect(cwd: string) {
342
+ const hasBinary = (await which("clang-format")) !== null;
343
+ if (!hasBinary) return false;
344
+ const configs = [".clang-format", "_clang-format"];
345
+ const found = await findUp(configs, cwd);
346
+ return found.length > 0;
347
+ },
348
+ };
349
+
350
+ export const ktlintFormatter: FormatterInfo = {
351
+ name: "ktlint",
352
+ command: ["ktlint", "-F", "$FILE"],
353
+ extensions: [".kt", ".kts"],
354
+ async detect(_cwd: string) {
355
+ return (await which("ktlint")) !== null;
356
+ },
357
+ };
358
+
359
+ export const terraformFormatter: FormatterInfo = {
360
+ name: "terraform",
361
+ command: ["terraform", "fmt", "$FILE"],
362
+ extensions: [".tf", ".tfvars"],
363
+ async detect(_cwd: string) {
364
+ return (await which("terraform")) !== null;
365
+ },
366
+ };
367
+
368
+ // --- Registry ---
369
+
370
+ const ALL_FORMATTERS: FormatterInfo[] = [
371
+ biomeFormatter,
372
+ prettierFormatter,
373
+ ruffFormatter,
374
+ blackFormatter,
375
+ gofmtFormatter,
376
+ rustfmtFormatter,
377
+ zigFormatter,
378
+ dartFormatter,
379
+ shfmtFormatter,
380
+ nixfmtFormatter,
381
+ mixFormatter,
382
+ ocamlformatFormatter,
383
+ clangFormatFormatter,
384
+ ktlintFormatter,
385
+ terraformFormatter,
386
+ ];
387
+
388
+ // Cache for detection results - stores array of enabled formatter names per cwd+ext
389
+ const detectionCache = new Map<string, Map<string, string[]>>();
390
+
391
+ // --- Public API ---
392
+
393
+ export async function getFormattersForFile(
394
+ filePath: string,
395
+ cwd: string,
396
+ ): Promise<FormatterInfo[]> {
397
+ const ext = path.extname(filePath).toLowerCase();
398
+ const cacheKey = `${cwd}:${ext}`;
399
+
400
+ // Check cache
401
+ let cached = detectionCache.get(cwd);
402
+ if (!cached) {
403
+ cached = new Map();
404
+ detectionCache.set(cwd, cached);
405
+ }
406
+
407
+ if (cached.has(cacheKey)) {
408
+ const enabledNames = cached.get(cacheKey);
409
+ if (!enabledNames || enabledNames.length === 0) return [];
410
+ // Return cached formatters by name (preserves priority order)
411
+ return ALL_FORMATTERS.filter((f) => enabledNames.includes(f.name));
412
+ }
413
+
414
+ // Detect formatters for this extension
415
+ const matching = ALL_FORMATTERS.filter((f) => f.extensions.includes(ext));
416
+ const enabled: FormatterInfo[] = [];
417
+
418
+ // Check for Biome first (preferred default)
419
+ const biomeFormatter = matching.find((f) => f.name === "biome");
420
+ let biomeEnabled = false;
421
+ if (biomeFormatter) {
422
+ try {
423
+ biomeEnabled = await biomeFormatter.detect(cwd);
424
+ if (biomeEnabled) {
425
+ enabled.push(biomeFormatter);
426
+ }
427
+ } catch (err) {
428
+ console.error(
429
+ `[format] Detection failed for ${biomeFormatter.name}:`,
430
+ err,
431
+ );
432
+ }
433
+ }
434
+
435
+ // If Biome is enabled, skip Prettier for overlapping extensions
436
+ // (Biome is the preferred default, Prettier is fallback)
437
+ const skipPrettier = biomeEnabled;
438
+
439
+ for (const formatter of matching) {
440
+ // Skip Biome (already checked above)
441
+ if (formatter.name === "biome") continue;
442
+
443
+ // Skip Prettier if Biome is enabled (prevents race condition)
444
+ if (skipPrettier && formatter.name === "prettier") continue;
445
+
446
+ try {
447
+ const isEnabled = await formatter.detect(cwd);
448
+ if (isEnabled) {
449
+ enabled.push(formatter);
450
+ }
451
+ } catch (err) {
452
+ // Detection failed, skip this formatter
453
+ console.error(`[format] Detection failed for ${formatter.name}:`, err);
454
+ }
455
+ }
456
+
457
+ // Store the list of enabled formatter names in cache
458
+ const enabledNames = enabled.map((f) => f.name);
459
+ cached.set(cacheKey, enabledNames);
460
+ return enabled;
461
+ }
462
+
463
+ export function clearFormatterCache(): void {
464
+ detectionCache.clear();
465
+ }
466
+
467
+ export async function formatFile(
468
+ filePath: string,
469
+ formatter: FormatterInfo,
470
+ ): Promise<FormatterResult> {
471
+ try {
472
+ const absolutePath = path.resolve(filePath);
473
+ const contentBefore = await fs.readFile(absolutePath, "utf-8");
474
+
475
+ // Replace $FILE placeholder
476
+ const cmd = formatter.command.map((c) => c.replace("$FILE", absolutePath));
477
+
478
+ // Run formatter
479
+ const result = safeSpawn(cmd[0], cmd.slice(1), { timeout: 15000 });
480
+
481
+ if (result.error) {
482
+ return {
483
+ success: false,
484
+ changed: false,
485
+ error: result.error.message,
486
+ };
487
+ }
488
+
489
+ // Check if content changed
490
+ const contentAfter = await fs.readFile(absolutePath, "utf-8");
491
+ const changed = contentBefore !== contentAfter;
492
+
493
+ return {
494
+ success: true,
495
+ changed,
496
+ };
497
+ } catch (err) {
498
+ return {
499
+ success: false,
500
+ changed: false,
501
+ error: err instanceof Error ? err.message : String(err),
502
+ };
503
+ }
504
+ }
505
+
506
+ export function listAllFormatters(): string[] {
507
+ return ALL_FORMATTERS.map((f) => f.name);
508
+ }
@@ -6,9 +6,9 @@
6
6
  * Requires: gopls (go install golang.org/x/tools/gopls@latest)
7
7
  * Docs: https://pkg.go.dev/golang.org/x/tools/gopls
8
8
  */
9
- import { spawnSync } from "node:child_process";
10
9
  import * as fs from "node:fs";
11
10
  import * as path from "node:path";
11
+ import { safeSpawn } from "./safe-spawn.js";
12
12
  // --- Common install paths ---
13
13
  const GO_WINDOWS_PATHS = [
14
14
  "C:\\Program Files\\Go\\bin\\go.exe",
@@ -48,10 +48,8 @@ export class GoClient {
48
48
  }
49
49
  else {
50
50
  // Relative (PATH) - try running it
51
- const result = spawnSync(p, ["version"], {
52
- encoding: "utf-8",
51
+ const result = safeSpawn(p, ["version"], {
53
52
  timeout: 3000,
54
- shell: true,
55
53
  });
56
54
  if (!result.error && result.status === 0) {
57
55
  this.goPath = p;
@@ -83,10 +81,8 @@ export class GoClient {
83
81
  isGoplsAvailable() {
84
82
  if (this.goplsAvailable !== null)
85
83
  return this.goplsAvailable;
86
- const result = spawnSync("gopls", ["version"], {
87
- encoding: "utf-8",
84
+ const result = safeSpawn("gopls", ["version"], {
88
85
  timeout: 5000,
89
- shell: true,
90
86
  });
91
87
  this.goplsAvailable = !result.error && result.status === 0;
92
88
  if (this.goplsAvailable) {
@@ -114,12 +110,9 @@ export class GoClient {
114
110
  const fileName = path.basename(absolutePath);
115
111
  try {
116
112
  // Run go vet on the specific file
117
- const goCmd = goExe.includes(" ") ? `"${goExe}"` : goExe;
118
- const result = spawnSync(goCmd, ["vet", fileName], {
119
- encoding: "utf-8",
113
+ const result = safeSpawn(goExe, ["vet", fileName], {
120
114
  timeout: 15000,
121
115
  cwd: dir,
122
- shell: true,
123
116
  });
124
117
  const output = (result.stderr || "") + (result.stdout || "");
125
118
  return this.parseOutput(output, absolutePath);
@@ -136,11 +129,9 @@ export class GoClient {
136
129
  if (!this.isGoAvailable())
137
130
  return [];
138
131
  try {
139
- const result = spawnSync("go", ["build", "./..."], {
140
- encoding: "utf-8",
132
+ const result = safeSpawn("go", ["build", "./..."], {
141
133
  timeout: 30000,
142
134
  cwd,
143
- shell: true,
144
135
  });
145
136
  const output = (result.stderr || "") + (result.stdout || "");
146
137
  return this.parseOutput(output, cwd);
@@ -10,6 +10,7 @@
10
10
  import { spawnSync } from "node:child_process";
11
11
  import * as fs from "node:fs";
12
12
  import * as path from "node:path";
13
+ import { safeSpawn } from "./safe-spawn.js";
13
14
 
14
15
  // --- Types ---
15
16
 
@@ -71,10 +72,8 @@ export class GoClient {
71
72
  }
72
73
  } else {
73
74
  // Relative (PATH) - try running it
74
- const result = spawnSync(p, ["version"], {
75
- encoding: "utf-8",
75
+ const result = safeSpawn(p, ["version"], {
76
76
  timeout: 3000,
77
- shell: true,
78
77
  });
79
78
  if (!result.error && result.status === 0) {
80
79
  this.goPath = p;
@@ -107,10 +106,8 @@ export class GoClient {
107
106
  isGoplsAvailable(): boolean {
108
107
  if (this.goplsAvailable !== null) return this.goplsAvailable;
109
108
 
110
- const result = spawnSync("gopls", ["version"], {
111
- encoding: "utf-8",
109
+ const result = safeSpawn("gopls", ["version"], {
112
110
  timeout: 5000,
113
- shell: true,
114
111
  });
115
112
 
116
113
  this.goplsAvailable = !result.error && result.status === 0;
@@ -142,12 +139,9 @@ export class GoClient {
142
139
 
143
140
  try {
144
141
  // Run go vet on the specific file
145
- const goCmd = goExe.includes(" ") ? `"${goExe}"` : goExe;
146
- const result = spawnSync(goCmd, ["vet", fileName], {
147
- encoding: "utf-8",
142
+ const result = safeSpawn(goExe, ["vet", fileName], {
148
143
  timeout: 15000,
149
144
  cwd: dir,
150
- shell: true,
151
145
  });
152
146
 
153
147
  const output = (result.stderr || "") + (result.stdout || "");
@@ -165,11 +159,9 @@ export class GoClient {
165
159
  if (!this.isGoAvailable()) return [];
166
160
 
167
161
  try {
168
- const result = spawnSync("go", ["build", "./..."], {
169
- encoding: "utf-8",
162
+ const result = safeSpawn("go", ["build", "./..."], {
170
163
  timeout: 30000,
171
164
  cwd,
172
- shell: true,
173
165
  });
174
166
 
175
167
  const output = (result.stderr || "") + (result.stdout || "");