@lyse-labs/lyse 0.1.0-alpha.2

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 (510) hide show
  1. package/LICENSE +45 -0
  2. package/PRIVACY.md +181 -0
  3. package/README.md +34 -0
  4. package/SECURITY.md +43 -0
  5. package/dist/bench/evidence-pack/anti-dummy.d.ts +11 -0
  6. package/dist/bench/evidence-pack/anti-dummy.js +20 -0
  7. package/dist/bench/evidence-pack/anti-dummy.js.map +1 -0
  8. package/dist/bench/evidence-pack/builder.d.ts +14 -0
  9. package/dist/bench/evidence-pack/builder.js +77 -0
  10. package/dist/bench/evidence-pack/builder.js.map +1 -0
  11. package/dist/bench/evidence-pack/histograms.d.ts +2 -0
  12. package/dist/bench/evidence-pack/histograms.js +131 -0
  13. package/dist/bench/evidence-pack/histograms.js.map +1 -0
  14. package/dist/bench/evidence-pack/manifest-detector.d.ts +2 -0
  15. package/dist/bench/evidence-pack/manifest-detector.js +65 -0
  16. package/dist/bench/evidence-pack/manifest-detector.js.map +1 -0
  17. package/dist/bench/evidence-pack/package-json-digest.d.ts +2 -0
  18. package/dist/bench/evidence-pack/package-json-digest.js +51 -0
  19. package/dist/bench/evidence-pack/package-json-digest.js.map +1 -0
  20. package/dist/bench/evidence-pack/sampler.d.ts +2 -0
  21. package/dist/bench/evidence-pack/sampler.js +140 -0
  22. package/dist/bench/evidence-pack/sampler.js.map +1 -0
  23. package/dist/bench/evidence-pack/types.d.ts +127 -0
  24. package/dist/bench/evidence-pack/types.js +2 -0
  25. package/dist/bench/evidence-pack/types.js.map +1 -0
  26. package/dist/bench/evidence-pack/verifier-corpus.d.ts +5 -0
  27. package/dist/bench/evidence-pack/verifier-corpus.js +13 -0
  28. package/dist/bench/evidence-pack/verifier-corpus.js.map +1 -0
  29. package/dist/bench/taxonomy-loader.d.ts +20 -0
  30. package/dist/bench/taxonomy-loader.js +28 -0
  31. package/dist/bench/taxonomy-loader.js.map +1 -0
  32. package/dist/bench/taxonomy.v3.json +20 -0
  33. package/dist/cli/__tests__/version-migration.test.d.ts +1 -0
  34. package/dist/cli/__tests__/version-migration.test.js +69 -0
  35. package/dist/cli/__tests__/version-migration.test.js.map +1 -0
  36. package/dist/cli/output/__tests__/eslint-style.test.d.ts +1 -0
  37. package/dist/cli/output/__tests__/eslint-style.test.js +205 -0
  38. package/dist/cli/output/__tests__/eslint-style.test.js.map +1 -0
  39. package/dist/cli/output/__tests__/limit.test.d.ts +1 -0
  40. package/dist/cli/output/__tests__/limit.test.js +65 -0
  41. package/dist/cli/output/__tests__/limit.test.js.map +1 -0
  42. package/dist/cli/output/eslint-style.d.ts +18 -0
  43. package/dist/cli/output/eslint-style.js +42 -0
  44. package/dist/cli/output/eslint-style.js.map +1 -0
  45. package/dist/cli/output/limit.d.ts +5 -0
  46. package/dist/cli/output/limit.js +32 -0
  47. package/dist/cli/output/limit.js.map +1 -0
  48. package/dist/cli/output/score-gauge.d.ts +4 -0
  49. package/dist/cli/output/score-gauge.js +15 -0
  50. package/dist/cli/output/score-gauge.js.map +1 -0
  51. package/dist/cli/version-migration.d.ts +18 -0
  52. package/dist/cli/version-migration.js +49 -0
  53. package/dist/cli/version-migration.js.map +1 -0
  54. package/dist/cli.d.ts +2 -0
  55. package/dist/cli.js +856 -0
  56. package/dist/cli.js.map +1 -0
  57. package/dist/codemods/diff.d.ts +17 -0
  58. package/dist/codemods/diff.js +77 -0
  59. package/dist/codemods/diff.js.map +1 -0
  60. package/dist/codemods/git-helpers.d.ts +10 -0
  61. package/dist/codemods/git-helpers.js +75 -0
  62. package/dist/codemods/git-helpers.js.map +1 -0
  63. package/dist/codemods/index.d.ts +27 -0
  64. package/dist/codemods/index.js +40 -0
  65. package/dist/codemods/index.js.map +1 -0
  66. package/dist/codemods/naming-component-pascalcase.d.ts +13 -0
  67. package/dist/codemods/naming-component-pascalcase.js +106 -0
  68. package/dist/codemods/naming-component-pascalcase.js.map +1 -0
  69. package/dist/codemods/naming-hook-prefix.d.ts +13 -0
  70. package/dist/codemods/naming-hook-prefix.js +86 -0
  71. package/dist/codemods/naming-hook-prefix.js.map +1 -0
  72. package/dist/codemods/safety.d.ts +50 -0
  73. package/dist/codemods/safety.js +109 -0
  74. package/dist/codemods/safety.js.map +1 -0
  75. package/dist/codemods/safety.test.d.ts +1 -0
  76. package/dist/codemods/safety.test.js +154 -0
  77. package/dist/codemods/safety.test.js.map +1 -0
  78. package/dist/codemods/shadow-native.d.ts +2 -0
  79. package/dist/codemods/shadow-native.js +103 -0
  80. package/dist/codemods/shadow-native.js.map +1 -0
  81. package/dist/codemods/tokens-color.d.ts +2 -0
  82. package/dist/codemods/tokens-color.js +69 -0
  83. package/dist/codemods/tokens-color.js.map +1 -0
  84. package/dist/codemods/tokens-spacing.d.ts +2 -0
  85. package/dist/codemods/tokens-spacing.js +66 -0
  86. package/dist/codemods/tokens-spacing.js.map +1 -0
  87. package/dist/commands/__tests__/email-prompt.test.d.ts +1 -0
  88. package/dist/commands/__tests__/email-prompt.test.js +126 -0
  89. package/dist/commands/__tests__/email-prompt.test.js.map +1 -0
  90. package/dist/commands/__tests__/explain-score.test.d.ts +1 -0
  91. package/dist/commands/__tests__/explain-score.test.js +88 -0
  92. package/dist/commands/__tests__/explain-score.test.js.map +1 -0
  93. package/dist/commands/__tests__/feedback.test.d.ts +1 -0
  94. package/dist/commands/__tests__/feedback.test.js +186 -0
  95. package/dist/commands/__tests__/feedback.test.js.map +1 -0
  96. package/dist/commands/audit-flags.d.ts +49 -0
  97. package/dist/commands/audit-flags.js +17 -0
  98. package/dist/commands/audit-flags.js.map +1 -0
  99. package/dist/commands/audit-pipeline.d.ts +31 -0
  100. package/dist/commands/audit-pipeline.js +342 -0
  101. package/dist/commands/audit-pipeline.js.map +1 -0
  102. package/dist/commands/bench-pack.d.ts +5 -0
  103. package/dist/commands/bench-pack.js +61 -0
  104. package/dist/commands/bench-pack.js.map +1 -0
  105. package/dist/commands/ci-setup.d.ts +9 -0
  106. package/dist/commands/ci-setup.js +42 -0
  107. package/dist/commands/ci-setup.js.map +1 -0
  108. package/dist/commands/email-prompt.d.ts +36 -0
  109. package/dist/commands/email-prompt.js +160 -0
  110. package/dist/commands/email-prompt.js.map +1 -0
  111. package/dist/commands/explain-score.d.ts +35 -0
  112. package/dist/commands/explain-score.js +137 -0
  113. package/dist/commands/explain-score.js.map +1 -0
  114. package/dist/commands/explain.d.ts +6 -0
  115. package/dist/commands/explain.js +93 -0
  116. package/dist/commands/explain.js.map +1 -0
  117. package/dist/commands/feedback.d.ts +31 -0
  118. package/dist/commands/feedback.js +155 -0
  119. package/dist/commands/feedback.js.map +1 -0
  120. package/dist/commands/fix.d.ts +48 -0
  121. package/dist/commands/fix.js +191 -0
  122. package/dist/commands/fix.js.map +1 -0
  123. package/dist/commands/init-detect.d.ts +20 -0
  124. package/dist/commands/init-detect.js +124 -0
  125. package/dist/commands/init-detect.js.map +1 -0
  126. package/dist/commands/init-write-agents-md.d.ts +14 -0
  127. package/dist/commands/init-write-agents-md.js +71 -0
  128. package/dist/commands/init-write-agents-md.js.map +1 -0
  129. package/dist/commands/init-write-lyse-md.d.ts +14 -0
  130. package/dist/commands/init-write-lyse-md.js +253 -0
  131. package/dist/commands/init-write-lyse-md.js.map +1 -0
  132. package/dist/commands/init.d.ts +8 -0
  133. package/dist/commands/init.js +147 -0
  134. package/dist/commands/init.js.map +1 -0
  135. package/dist/commands/mcp-entry.d.ts +33 -0
  136. package/dist/commands/mcp-entry.js +47 -0
  137. package/dist/commands/mcp-entry.js.map +1 -0
  138. package/dist/commands/mcp-setup.d.ts +10 -0
  139. package/dist/commands/mcp-setup.js +74 -0
  140. package/dist/commands/mcp-setup.js.map +1 -0
  141. package/dist/commands/share.d.ts +4 -0
  142. package/dist/commands/share.js +60 -0
  143. package/dist/commands/share.js.map +1 -0
  144. package/dist/commands/telemetry.d.ts +4 -0
  145. package/dist/commands/telemetry.js +27 -0
  146. package/dist/commands/telemetry.js.map +1 -0
  147. package/dist/commands/templates/lyse-workflow.yml.template +30 -0
  148. package/dist/config/schema.d.ts +158 -0
  149. package/dist/config/schema.js +136 -0
  150. package/dist/config/schema.js.map +1 -0
  151. package/dist/credentials/keychain.d.ts +20 -0
  152. package/dist/credentials/keychain.js +35 -0
  153. package/dist/credentials/keychain.js.map +1 -0
  154. package/dist/credentials/keychain.test.d.ts +1 -0
  155. package/dist/credentials/keychain.test.js +32 -0
  156. package/dist/credentials/keychain.test.js.map +1 -0
  157. package/dist/credentials/paths.d.ts +1 -0
  158. package/dist/credentials/paths.js +6 -0
  159. package/dist/credentials/paths.js.map +1 -0
  160. package/dist/credentials/store.d.ts +17 -0
  161. package/dist/credentials/store.js +60 -0
  162. package/dist/credentials/store.js.map +1 -0
  163. package/dist/credentials/store.test.d.ts +1 -0
  164. package/dist/credentials/store.test.js +48 -0
  165. package/dist/credentials/store.test.js.map +1 -0
  166. package/dist/detection/from-filesystem.d.ts +2 -0
  167. package/dist/detection/from-filesystem.js +26 -0
  168. package/dist/detection/from-filesystem.js.map +1 -0
  169. package/dist/detection/from-git.d.ts +2 -0
  170. package/dist/detection/from-git.js +53 -0
  171. package/dist/detection/from-git.js.map +1 -0
  172. package/dist/detection/from-package-json.d.ts +2 -0
  173. package/dist/detection/from-package-json.js +194 -0
  174. package/dist/detection/from-package-json.js.map +1 -0
  175. package/dist/detection/pre-flight.d.ts +5 -0
  176. package/dist/detection/pre-flight.js +47 -0
  177. package/dist/detection/pre-flight.js.map +1 -0
  178. package/dist/detection/types.d.ts +27 -0
  179. package/dist/detection/types.js +2 -0
  180. package/dist/detection/types.js.map +1 -0
  181. package/dist/entitlement/check.d.ts +7 -0
  182. package/dist/entitlement/check.js +37 -0
  183. package/dist/entitlement/check.js.map +1 -0
  184. package/dist/entitlement/index.d.ts +1 -0
  185. package/dist/entitlement/index.js +2 -0
  186. package/dist/entitlement/index.js.map +1 -0
  187. package/dist/entitlement/keys.d.ts +10 -0
  188. package/dist/entitlement/keys.js +14 -0
  189. package/dist/entitlement/keys.js.map +1 -0
  190. package/dist/history/ndjson-store.d.ts +70 -0
  191. package/dist/history/ndjson-store.js +102 -0
  192. package/dist/history/ndjson-store.js.map +1 -0
  193. package/dist/identity/index.d.ts +1 -0
  194. package/dist/identity/index.js +2 -0
  195. package/dist/identity/index.js.map +1 -0
  196. package/dist/identity/repo-bucket.d.ts +22 -0
  197. package/dist/identity/repo-bucket.js +78 -0
  198. package/dist/identity/repo-bucket.js.map +1 -0
  199. package/dist/index.d.ts +1 -0
  200. package/dist/index.js +15 -0
  201. package/dist/index.js.map +1 -0
  202. package/dist/llm/layer4-stage.d.ts +18 -0
  203. package/dist/llm/layer4-stage.js +12 -0
  204. package/dist/llm/layer4-stage.js.map +1 -0
  205. package/dist/loaders/components.d.ts +26 -0
  206. package/dist/loaders/components.js +285 -0
  207. package/dist/loaders/components.js.map +1 -0
  208. package/dist/loaders/stories.d.ts +2 -0
  209. package/dist/loaders/stories.js +231 -0
  210. package/dist/loaders/stories.js.map +1 -0
  211. package/dist/loaders/token-graph.d.ts +39 -0
  212. package/dist/loaders/token-graph.js +146 -0
  213. package/dist/loaders/token-graph.js.map +1 -0
  214. package/dist/loaders/tokens.d.ts +2 -0
  215. package/dist/loaders/tokens.js +521 -0
  216. package/dist/loaders/tokens.js.map +1 -0
  217. package/dist/mcp/_find-root.d.ts +12 -0
  218. package/dist/mcp/_find-root.js +28 -0
  219. package/dist/mcp/_find-root.js.map +1 -0
  220. package/dist/mcp/server.d.ts +1 -0
  221. package/dist/mcp/server.js +42 -0
  222. package/dist/mcp/server.js.map +1 -0
  223. package/dist/mcp/tools/audit-file.d.ts +25 -0
  224. package/dist/mcp/tools/audit-file.js +147 -0
  225. package/dist/mcp/tools/audit-file.js.map +1 -0
  226. package/dist/mcp/tools/suggest-fix.d.ts +13 -0
  227. package/dist/mcp/tools/suggest-fix.js +100 -0
  228. package/dist/mcp/tools/suggest-fix.js.map +1 -0
  229. package/dist/menu/action-menu.d.ts +6 -0
  230. package/dist/menu/action-menu.js +15 -0
  231. package/dist/menu/action-menu.js.map +1 -0
  232. package/dist/menu/prompts.d.ts +7 -0
  233. package/dist/menu/prompts.js +30 -0
  234. package/dist/menu/prompts.js.map +1 -0
  235. package/dist/menu/repl.d.ts +17 -0
  236. package/dist/menu/repl.js +77 -0
  237. package/dist/menu/repl.js.map +1 -0
  238. package/dist/parsers/css-in-js.d.ts +2 -0
  239. package/dist/parsers/css-in-js.js +69 -0
  240. package/dist/parsers/css-in-js.js.map +1 -0
  241. package/dist/parsers/css.d.ts +2 -0
  242. package/dist/parsers/css.js +32 -0
  243. package/dist/parsers/css.js.map +1 -0
  244. package/dist/parsers/scss-transform.d.ts +25 -0
  245. package/dist/parsers/scss-transform.js +55 -0
  246. package/dist/parsers/scss-transform.js.map +1 -0
  247. package/dist/parsers/tailwind-v4.d.ts +8 -0
  248. package/dist/parsers/tailwind-v4.js +90 -0
  249. package/dist/parsers/tailwind-v4.js.map +1 -0
  250. package/dist/parsers/ts-morph-project.d.ts +16 -0
  251. package/dist/parsers/ts-morph-project.js +77 -0
  252. package/dist/parsers/ts-morph-project.js.map +1 -0
  253. package/dist/parsers/ts.d.ts +2 -0
  254. package/dist/parsers/ts.js +61 -0
  255. package/dist/parsers/ts.js.map +1 -0
  256. package/dist/reliability/__tests__/types.test.d.ts +1 -0
  257. package/dist/reliability/__tests__/types.test.js +14 -0
  258. package/dist/reliability/__tests__/types.test.js.map +1 -0
  259. package/dist/reliability/catalogue/__tests__/promotion.test.d.ts +1 -0
  260. package/dist/reliability/catalogue/__tests__/promotion.test.js +23 -0
  261. package/dist/reliability/catalogue/__tests__/promotion.test.js.map +1 -0
  262. package/dist/reliability/catalogue/__tests__/sub-axes.test.d.ts +1 -0
  263. package/dist/reliability/catalogue/__tests__/sub-axes.test.js +27 -0
  264. package/dist/reliability/catalogue/__tests__/sub-axes.test.js.map +1 -0
  265. package/dist/reliability/catalogue/promotion.d.ts +8 -0
  266. package/dist/reliability/catalogue/promotion.js +25 -0
  267. package/dist/reliability/catalogue/promotion.js.map +1 -0
  268. package/dist/reliability/catalogue/sub-axes.d.ts +3 -0
  269. package/dist/reliability/catalogue/sub-axes.js +18 -0
  270. package/dist/reliability/catalogue/sub-axes.js.map +1 -0
  271. package/dist/reliability/confidence/__tests__/manifest-loader.test.d.ts +1 -0
  272. package/dist/reliability/confidence/__tests__/manifest-loader.test.js +103 -0
  273. package/dist/reliability/confidence/__tests__/manifest-loader.test.js.map +1 -0
  274. package/dist/reliability/confidence/bundled-manifest.d.ts +2 -0
  275. package/dist/reliability/confidence/bundled-manifest.js +8 -0
  276. package/dist/reliability/confidence/bundled-manifest.js.map +1 -0
  277. package/dist/reliability/confidence/index.d.ts +3 -0
  278. package/dist/reliability/confidence/index.js +3 -0
  279. package/dist/reliability/confidence/index.js.map +1 -0
  280. package/dist/reliability/confidence/manifest-loader.d.ts +8 -0
  281. package/dist/reliability/confidence/manifest-loader.js +53 -0
  282. package/dist/reliability/confidence/manifest-loader.js.map +1 -0
  283. package/dist/reliability/feedback/__tests__/interactive.test.d.ts +1 -0
  284. package/dist/reliability/feedback/__tests__/interactive.test.js +85 -0
  285. package/dist/reliability/feedback/__tests__/interactive.test.js.map +1 -0
  286. package/dist/reliability/feedback/__tests__/repo-bucket.test.d.ts +1 -0
  287. package/dist/reliability/feedback/__tests__/repo-bucket.test.js +18 -0
  288. package/dist/reliability/feedback/__tests__/repo-bucket.test.js.map +1 -0
  289. package/dist/reliability/feedback/__tests__/sender.test.d.ts +1 -0
  290. package/dist/reliability/feedback/__tests__/sender.test.js +101 -0
  291. package/dist/reliability/feedback/__tests__/sender.test.js.map +1 -0
  292. package/dist/reliability/feedback/interactive.d.ts +20 -0
  293. package/dist/reliability/feedback/interactive.js +150 -0
  294. package/dist/reliability/feedback/interactive.js.map +1 -0
  295. package/dist/reliability/feedback/repo-bucket.d.ts +2 -0
  296. package/dist/reliability/feedback/repo-bucket.js +9 -0
  297. package/dist/reliability/feedback/repo-bucket.js.map +1 -0
  298. package/dist/reliability/feedback/sender.d.ts +16 -0
  299. package/dist/reliability/feedback/sender.js +28 -0
  300. package/dist/reliability/feedback/sender.js.map +1 -0
  301. package/dist/reliability/index.d.ts +1 -0
  302. package/dist/reliability/index.js +2 -0
  303. package/dist/reliability/index.js.map +1 -0
  304. package/dist/reliability/llm-eval/__tests__/budget.test.d.ts +1 -0
  305. package/dist/reliability/llm-eval/__tests__/budget.test.js +39 -0
  306. package/dist/reliability/llm-eval/__tests__/budget.test.js.map +1 -0
  307. package/dist/reliability/llm-eval/budget.d.ts +14 -0
  308. package/dist/reliability/llm-eval/budget.js +43 -0
  309. package/dist/reliability/llm-eval/budget.js.map +1 -0
  310. package/dist/reliability/score/__tests__/formula-v1.test.d.ts +1 -0
  311. package/dist/reliability/score/__tests__/formula-v1.test.js +29 -0
  312. package/dist/reliability/score/__tests__/formula-v1.test.js.map +1 -0
  313. package/dist/reliability/score/formula-v1.d.ts +13 -0
  314. package/dist/reliability/score/formula-v1.js +19 -0
  315. package/dist/reliability/score/formula-v1.js.map +1 -0
  316. package/dist/reliability/score/version-pin.d.ts +1 -0
  317. package/dist/reliability/score/version-pin.js +2 -0
  318. package/dist/reliability/score/version-pin.js.map +1 -0
  319. package/dist/reliability/score/weight.d.ts +1 -0
  320. package/dist/reliability/score/weight.js +5 -0
  321. package/dist/reliability/score/weight.js.map +1 -0
  322. package/dist/reliability/types.d.ts +39 -0
  323. package/dist/reliability/types.js +2 -0
  324. package/dist/reliability/types.js.map +1 -0
  325. package/dist/reporters/coverage-footer.d.ts +2 -0
  326. package/dist/reporters/coverage-footer.js +7 -0
  327. package/dist/reporters/coverage-footer.js.map +1 -0
  328. package/dist/reporters/json.d.ts +5 -0
  329. package/dist/reporters/json.js +51 -0
  330. package/dist/reporters/json.js.map +1 -0
  331. package/dist/reporters/markdown.d.ts +7 -0
  332. package/dist/reporters/markdown.js +35 -0
  333. package/dist/reporters/markdown.js.map +1 -0
  334. package/dist/reporters/sarif.d.ts +5 -0
  335. package/dist/reporters/sarif.js +109 -0
  336. package/dist/reporters/sarif.js.map +1 -0
  337. package/dist/reporters/terminal-format.d.ts +34 -0
  338. package/dist/reporters/terminal-format.js +97 -0
  339. package/dist/reporters/terminal-format.js.map +1 -0
  340. package/dist/reporters/terminal.d.ts +4 -0
  341. package/dist/reporters/terminal.js +201 -0
  342. package/dist/reporters/terminal.js.map +1 -0
  343. package/dist/rule-runner.d.ts +8 -0
  344. package/dist/rule-runner.js +22 -0
  345. package/dist/rule-runner.js.map +1 -0
  346. package/dist/rules/_codemod-adapter.d.ts +16 -0
  347. package/dist/rules/_codemod-adapter.js +49 -0
  348. package/dist/rules/_codemod-adapter.js.map +1 -0
  349. package/dist/rules/_exclude.d.ts +11 -0
  350. package/dist/rules/_exclude.js +17 -0
  351. package/dist/rules/_exclude.js.map +1 -0
  352. package/dist/rules/_function-body-analysis.d.ts +82 -0
  353. package/dist/rules/_function-body-analysis.js +379 -0
  354. package/dist/rules/_function-body-analysis.js.map +1 -0
  355. package/dist/rules/_rule-module.d.ts +31 -0
  356. package/dist/rules/_rule-module.js +32 -0
  357. package/dist/rules/_rule-module.js.map +1 -0
  358. package/dist/rules/_skip-context.d.ts +36 -0
  359. package/dist/rules/_skip-context.js +128 -0
  360. package/dist/rules/_skip-context.js.map +1 -0
  361. package/dist/rules/a11y-essentials.d.ts +1 -0
  362. package/dist/rules/a11y-essentials.js +140 -0
  363. package/dist/rules/a11y-essentials.js.map +1 -0
  364. package/dist/rules/ai-surface-agents-md-quality.d.ts +22 -0
  365. package/dist/rules/ai-surface-agents-md-quality.js +233 -0
  366. package/dist/rules/ai-surface-agents-md-quality.js.map +1 -0
  367. package/dist/rules/ai-surface-component-manifest-json.d.ts +19 -0
  368. package/dist/rules/ai-surface-component-manifest-json.js +232 -0
  369. package/dist/rules/ai-surface-component-manifest-json.js.map +1 -0
  370. package/dist/rules/ai-surface-ds-index-exported.d.ts +19 -0
  371. package/dist/rules/ai-surface-ds-index-exported.js +239 -0
  372. package/dist/rules/ai-surface-ds-index-exported.js.map +1 -0
  373. package/dist/rules/components-shadow-native.d.ts +2 -0
  374. package/dist/rules/components-shadow-native.js +118 -0
  375. package/dist/rules/components-shadow-native.js.map +1 -0
  376. package/dist/rules/manifest.d.ts +5 -0
  377. package/dist/rules/manifest.js +20 -0
  378. package/dist/rules/manifest.js.map +1 -0
  379. package/dist/rules/naming-component-pascalcase.d.ts +7 -0
  380. package/dist/rules/naming-component-pascalcase.js +197 -0
  381. package/dist/rules/naming-component-pascalcase.js.map +1 -0
  382. package/dist/rules/naming-hook-prefix.d.ts +6 -0
  383. package/dist/rules/naming-hook-prefix.js +185 -0
  384. package/dist/rules/naming-hook-prefix.js.map +1 -0
  385. package/dist/rules/pack-loader.d.ts +30 -0
  386. package/dist/rules/pack-loader.js +60 -0
  387. package/dist/rules/pack-loader.js.map +1 -0
  388. package/dist/rules/pack-validator.d.ts +8 -0
  389. package/dist/rules/pack-validator.js +97 -0
  390. package/dist/rules/pack-validator.js.map +1 -0
  391. package/dist/rules/registry.d.ts +3 -0
  392. package/dist/rules/registry.js +28 -0
  393. package/dist/rules/registry.js.map +1 -0
  394. package/dist/rules/storybook-coverage.d.ts +1 -0
  395. package/dist/rules/storybook-coverage.js +49 -0
  396. package/dist/rules/storybook-coverage.js.map +1 -0
  397. package/dist/rules/templates/_regex-utils.d.ts +5 -0
  398. package/dist/rules/templates/_regex-utils.js +8 -0
  399. package/dist/rules/templates/_regex-utils.js.map +1 -0
  400. package/dist/rules/templates/a11y-jsx-template.d.ts +15 -0
  401. package/dist/rules/templates/a11y-jsx-template.js +69 -0
  402. package/dist/rules/templates/a11y-jsx-template.js.map +1 -0
  403. package/dist/rules/templates/css-property-token-compliance.d.ts +18 -0
  404. package/dist/rules/templates/css-property-token-compliance.js +56 -0
  405. package/dist/rules/templates/css-property-token-compliance.js.map +1 -0
  406. package/dist/rules/templates/import-source-restriction.d.ts +18 -0
  407. package/dist/rules/templates/import-source-restriction.js +60 -0
  408. package/dist/rules/templates/import-source-restriction.js.map +1 -0
  409. package/dist/rules/templates/js-call-token-compliance.d.ts +18 -0
  410. package/dist/rules/templates/js-call-token-compliance.js +61 -0
  411. package/dist/rules/templates/js-call-token-compliance.js.map +1 -0
  412. package/dist/rules/templates/js-prop-token-compliance.d.ts +21 -0
  413. package/dist/rules/templates/js-prop-token-compliance.js +56 -0
  414. package/dist/rules/templates/js-prop-token-compliance.js.map +1 -0
  415. package/dist/rules/templates/naming-convention.d.ts +18 -0
  416. package/dist/rules/templates/naming-convention.js +76 -0
  417. package/dist/rules/templates/naming-convention.js.map +1 -0
  418. package/dist/rules/templates/registry.d.ts +5 -0
  419. package/dist/rules/templates/registry.js +30 -0
  420. package/dist/rules/templates/registry.js.map +1 -0
  421. package/dist/rules/templates/storybook-coverage-template.d.ts +15 -0
  422. package/dist/rules/templates/storybook-coverage-template.js +58 -0
  423. package/dist/rules/templates/storybook-coverage-template.js.map +1 -0
  424. package/dist/rules/templates/tailwind-utility-class-compliance.d.ts +15 -0
  425. package/dist/rules/templates/tailwind-utility-class-compliance.js +70 -0
  426. package/dist/rules/templates/tailwind-utility-class-compliance.js.map +1 -0
  427. package/dist/rules/templates/types.d.ts +16 -0
  428. package/dist/rules/templates/types.js +2 -0
  429. package/dist/rules/templates/types.js.map +1 -0
  430. package/dist/rules/tokens-description-coverage.d.ts +15 -0
  431. package/dist/rules/tokens-description-coverage.js +193 -0
  432. package/dist/rules/tokens-description-coverage.js.map +1 -0
  433. package/dist/rules/tokens-dtcg-conformance.d.ts +47 -0
  434. package/dist/rules/tokens-dtcg-conformance.js +363 -0
  435. package/dist/rules/tokens-dtcg-conformance.js.map +1 -0
  436. package/dist/rules/tokens-no-hardcoded-color.d.ts +18 -0
  437. package/dist/rules/tokens-no-hardcoded-color.js +436 -0
  438. package/dist/rules/tokens-no-hardcoded-color.js.map +1 -0
  439. package/dist/rules/tokens-no-hardcoded-spacing.d.ts +8 -0
  440. package/dist/rules/tokens-no-hardcoded-spacing.js +193 -0
  441. package/dist/rules/tokens-no-hardcoded-spacing.js.map +1 -0
  442. package/dist/scorer.d.ts +38 -0
  443. package/dist/scorer.js +126 -0
  444. package/dist/scorer.js.map +1 -0
  445. package/dist/share/clipboard.d.ts +26 -0
  446. package/dist/share/clipboard.js +76 -0
  447. package/dist/share/clipboard.js.map +1 -0
  448. package/dist/suppression/__tests__/inline.test.d.ts +1 -0
  449. package/dist/suppression/__tests__/inline.test.js +25 -0
  450. package/dist/suppression/__tests__/inline.test.js.map +1 -0
  451. package/dist/suppression/__tests__/lyseignore.test.d.ts +1 -0
  452. package/dist/suppression/__tests__/lyseignore.test.js +32 -0
  453. package/dist/suppression/__tests__/lyseignore.test.js.map +1 -0
  454. package/dist/suppression/defaults.d.ts +1 -0
  455. package/dist/suppression/defaults.js +14 -0
  456. package/dist/suppression/defaults.js.map +1 -0
  457. package/dist/suppression/inline.d.ts +7 -0
  458. package/dist/suppression/inline.js +32 -0
  459. package/dist/suppression/inline.js.map +1 -0
  460. package/dist/suppression/lyseignore.d.ts +2 -0
  461. package/dist/suppression/lyseignore.js +22 -0
  462. package/dist/suppression/lyseignore.js.map +1 -0
  463. package/dist/telemetry/consent.d.ts +48 -0
  464. package/dist/telemetry/consent.js +139 -0
  465. package/dist/telemetry/consent.js.map +1 -0
  466. package/dist/telemetry/index.d.ts +2 -0
  467. package/dist/telemetry/index.js +3 -0
  468. package/dist/telemetry/index.js.map +1 -0
  469. package/dist/telemetry/local-log.d.ts +64 -0
  470. package/dist/telemetry/local-log.js +123 -0
  471. package/dist/telemetry/local-log.js.map +1 -0
  472. package/dist/tokens/dtcg-model.d.ts +53 -0
  473. package/dist/tokens/dtcg-model.js +18 -0
  474. package/dist/tokens/dtcg-model.js.map +1 -0
  475. package/dist/tokens/normalizer.d.ts +17 -0
  476. package/dist/tokens/normalizer.js +360 -0
  477. package/dist/tokens/normalizer.js.map +1 -0
  478. package/dist/types.d.ts +300 -0
  479. package/dist/types.js +4 -0
  480. package/dist/types.js.map +1 -0
  481. package/dist/util/git.d.ts +6 -0
  482. package/dist/util/git.js +40 -0
  483. package/dist/util/git.js.map +1 -0
  484. package/dist/util/git.test.d.ts +5 -0
  485. package/dist/util/git.test.js +42 -0
  486. package/dist/util/git.test.js.map +1 -0
  487. package/dist/util/gitignore.d.ts +9 -0
  488. package/dist/util/gitignore.js +39 -0
  489. package/dist/util/gitignore.js.map +1 -0
  490. package/dist/util/hash-deps.d.ts +6 -0
  491. package/dist/util/hash-deps.js +24 -0
  492. package/dist/util/hash-deps.js.map +1 -0
  493. package/dist/util/spinner.d.ts +41 -0
  494. package/dist/util/spinner.js +126 -0
  495. package/dist/util/spinner.js.map +1 -0
  496. package/dist/util/with-spinner.d.ts +11 -0
  497. package/dist/util/with-spinner.js +19 -0
  498. package/dist/util/with-spinner.js.map +1 -0
  499. package/dist/walker.d.ts +14 -0
  500. package/dist/walker.js +91 -0
  501. package/dist/walker.js.map +1 -0
  502. package/package.json +83 -0
  503. package/rules-manifest.json +289 -0
  504. package/schemas/v1/lyse-config.json +82 -0
  505. package/schemas/v1/lyse-event.json +68 -0
  506. package/schemas/v1/lyse-license.json +19 -0
  507. package/schemas/v1/lyse-llm-payload.json +53 -0
  508. package/schemas/v1/lyse-llm-response.json +45 -0
  509. package/schemas/v1/lyse-result.json +98 -0
  510. package/schemas/v1/lyse-rules.json +42 -0
@@ -0,0 +1,109 @@
1
+ import { ruleMap as ruleRegistry } from "../rules/registry.js";
2
+ /** An empty token map — the safe fallback when a repo has no discoverable tokens. */
3
+ function emptyTokenMap() {
4
+ return {
5
+ colors: new Map(),
6
+ spacing: new Map(),
7
+ typography: new Map(),
8
+ radii: new Map(),
9
+ shadows: new Map(),
10
+ motion: new Map(),
11
+ breakpoints: new Map(),
12
+ zIndex: new Map(),
13
+ opacity: new Map(),
14
+ borderWidth: new Map(),
15
+ source: "mixed",
16
+ };
17
+ }
18
+ /**
19
+ * Build the ClassifyContext that confidence classification needs from audit
20
+ * output. Single source of truth for the token fallback + component-set
21
+ * derivation, shared by `fix` and the post-audit menu count.
22
+ */
23
+ export function buildClassifyContext(findings, tokens, config, repoRoot) {
24
+ return {
25
+ tokens: tokens ?? emptyTokenMap(),
26
+ components: new Set(findings.filter((f) => f.axis === "components").map((f) => f.ruleId)),
27
+ config,
28
+ ...(repoRoot !== undefined ? { repoRoot } : {}),
29
+ };
30
+ }
31
+ /**
32
+ * Count findings that `lyse fix` would auto-apply at the default "high"
33
+ * confidence floor: those whose rule has a codemod AND classify as high
34
+ * confidence. Mirrors `runFix`'s default filter chain (high floor, no
35
+ * `--rule` filter) so the post-audit menu shows a count consistent with what
36
+ * `fix` actually does. `repoRoot` must match `runFix`'s `cwd` — some rules
37
+ * downgrade confidence based on it (e.g. token-definition files), so passing a
38
+ * different value here would desync the menu count from the real fix count.
39
+ */
40
+ export function countAutoFixable(findings, tokens, config, repoRoot) {
41
+ const fixable = findings.filter((f) => !!ruleRegistry.get(f.ruleId)?.applyCodemod);
42
+ const ctx = buildClassifyContext(findings, tokens, config, repoRoot);
43
+ return groupByConfidence(fixable, ctx).high.length;
44
+ }
45
+ /**
46
+ * Dispatcher that delegates classifyConfidence to the rule that owns the finding.
47
+ * Returns "low" (safe default) for unknown rules or rules without classifyConfidence.
48
+ *
49
+ * Open-Closed Principle: each rule owns its confidence classification logic.
50
+ * This function is a thin dispatch layer, not a policy layer.
51
+ */
52
+ export function classifyConfidence(finding, ctx) {
53
+ const rule = ruleRegistry.get(finding.ruleId);
54
+ if (!rule?.classifyConfidence) {
55
+ return "low";
56
+ }
57
+ return rule.classifyConfidence(finding, ctx);
58
+ }
59
+ /**
60
+ * Group findings by confidence level, augmenting each with a `confidence` field.
61
+ * Returns three buckets: high, medium, low (always present, may be empty).
62
+ */
63
+ export function groupByConfidence(findings, ctx) {
64
+ const groups = {
65
+ high: [],
66
+ medium: [],
67
+ low: [],
68
+ };
69
+ for (const f of findings) {
70
+ const c = classifyConfidence(f, ctx);
71
+ groups[c].push({ ...f, confidence: c });
72
+ }
73
+ return groups;
74
+ }
75
+ /**
76
+ * Populate `Finding.confidence` on every finding in an AuditResult by running
77
+ * each through the owning rule's `classifyConfidence`. The audit pipeline emits
78
+ * findings without a confidence field (rules don't know their own classification
79
+ * at emit time — the context needed for classification is repo-wide, not
80
+ * rule-local). Downstream CLI consumers (score gauge, ESLint-style tag,
81
+ * post-audit menu) all read `finding.confidence`, so this helper closes the
82
+ * gap once per run instead of having every consumer recompute it.
83
+ *
84
+ * Returns a NEW result with new finding objects (no in-place mutation) so
85
+ * cached or reused references stay untouched.
86
+ */
87
+ export function populateConfidence(result, ctx) {
88
+ return {
89
+ ...result,
90
+ findings: result.findings.map((f) => ({
91
+ ...f,
92
+ confidence: classifyConfidence(f, ctx),
93
+ })),
94
+ };
95
+ }
96
+ /**
97
+ * Group findings by ruleId.
98
+ * Returns a Map where each key is a ruleId and value is the array of findings for that rule.
99
+ */
100
+ export function groupByRule(findings) {
101
+ const groups = new Map();
102
+ for (const f of findings) {
103
+ const arr = groups.get(f.ruleId) ?? [];
104
+ arr.push(f);
105
+ groups.set(f.ruleId, arr);
106
+ }
107
+ return groups;
108
+ }
109
+ //# sourceMappingURL=safety.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"safety.js","sourceRoot":"","sources":["../../src/codemods/safety.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,154 @@
1
+ import { describe, it, expect } from "vitest";
2
+ import { countAutoFixable, buildClassifyContext, populateConfidence } from "./safety.js";
3
+ function tokenMapWith(colors) {
4
+ return {
5
+ colors: new Map(Object.entries(colors)),
6
+ spacing: new Map(),
7
+ typography: new Map(),
8
+ radii: new Map(),
9
+ shadows: new Map(),
10
+ motion: new Map(),
11
+ breakpoints: new Map(),
12
+ zIndex: new Map(),
13
+ opacity: new Map(),
14
+ borderWidth: new Map(),
15
+ source: "mixed",
16
+ };
17
+ }
18
+ function colorFinding(value, file = "src/Button.tsx") {
19
+ return {
20
+ ruleId: "tokens/no-hardcoded-color",
21
+ axis: "tokens",
22
+ severity: "warning",
23
+ location: { file, line: 1, column: 1 },
24
+ message: `Hardcoded color value: ${value}`,
25
+ };
26
+ }
27
+ const EMPTY_CONFIG = {};
28
+ describe("countAutoFixable", () => {
29
+ it("counts a hardcoded color mapping to exactly one token as auto-fixable (high)", () => {
30
+ const tokens = tokenMapWith({ "#fff": ["color.white"] });
31
+ expect(countAutoFixable([colorFinding("#fff")], tokens, EMPTY_CONFIG)).toBe(1);
32
+ });
33
+ it("ignores a color with no matching token (low confidence)", () => {
34
+ const tokens = tokenMapWith({ "#000": ["color.black"] });
35
+ expect(countAutoFixable([colorFinding("#fff")], tokens, EMPTY_CONFIG)).toBe(0);
36
+ });
37
+ it("ignores findings whose rule has no codemod", () => {
38
+ const a11y = {
39
+ ruleId: "a11y/essentials",
40
+ axis: "a11y",
41
+ severity: "warning",
42
+ location: { file: "src/x.tsx", line: 1, column: 1 },
43
+ message: "Anchors must have content",
44
+ };
45
+ expect(countAutoFixable([a11y], tokenMapWith({}), EMPTY_CONFIG)).toBe(0);
46
+ });
47
+ it("falls back to an empty token map when tokens is null (no crash, count 0)", () => {
48
+ expect(countAutoFixable([colorFinding("#fff")], null, EMPTY_CONFIG)).toBe(0);
49
+ });
50
+ });
51
+ describe("buildClassifyContext", () => {
52
+ it("derives the component-name set from components-axis findings only", () => {
53
+ const findings = [
54
+ {
55
+ ruleId: "naming/component-pascalcase",
56
+ axis: "components",
57
+ severity: "warning",
58
+ location: { file: "a.tsx", line: 1, column: 1 },
59
+ message: "x",
60
+ },
61
+ colorFinding("#fff", "b.tsx"),
62
+ ];
63
+ const ctx = buildClassifyContext(findings, null, EMPTY_CONFIG);
64
+ expect(ctx.components.has("naming/component-pascalcase")).toBe(true);
65
+ expect(ctx.components.has("tokens/no-hardcoded-color")).toBe(false);
66
+ });
67
+ it("omits repoRoot when not provided (exactOptionalPropertyTypes-safe)", () => {
68
+ const ctx = buildClassifyContext([], null, EMPTY_CONFIG);
69
+ expect("repoRoot" in ctx).toBe(false);
70
+ });
71
+ it("threads repoRoot into the context when provided (keeps menu count in sync with fix)", () => {
72
+ // Some rules downgrade confidence based on repoRoot (e.g. token-definition
73
+ // files). The menu must pass the same repoRoot as `runFix` or its count
74
+ // would desync; guard that the helper forwards it.
75
+ const ctx = buildClassifyContext([], null, EMPTY_CONFIG, "/repo/root");
76
+ expect(ctx.repoRoot).toBe("/repo/root");
77
+ });
78
+ });
79
+ function makeAuditResult(findings) {
80
+ return {
81
+ schemaVersion: 2,
82
+ rulesVersion: "test",
83
+ toolVersion: "test",
84
+ scoringVersion: "scoring-v1",
85
+ repoRoot: "/repo",
86
+ timestamp: "2026-01-01T00:00:00.000Z",
87
+ stack: [],
88
+ finalScore: 100,
89
+ tier: "excellent",
90
+ axes: [],
91
+ findings,
92
+ };
93
+ }
94
+ describe("populateConfidence", () => {
95
+ // Bug fix (#16, #17): the audit pipeline emits Finding objects without a
96
+ // `confidence` field, so the CLI's score-gauge experimental counter and the
97
+ // ESLint-style "EXP" tag were both inert. populateConfidence runs each
98
+ // finding through the owning rule's classifyConfidence so downstream
99
+ // consumers see a populated value.
100
+ it("sets confidence on every finding (high when token maps cleanly)", () => {
101
+ const tokens = tokenMapWith({ "#fff": ["color.white"] });
102
+ const result = makeAuditResult([colorFinding("#fff")]);
103
+ const ctx = buildClassifyContext(result.findings, tokens, EMPTY_CONFIG, "/repo");
104
+ const populated = populateConfidence(result, ctx);
105
+ expect(populated.findings[0]?.confidence).toBe("high");
106
+ });
107
+ it("sets confidence='low' when no token matches (drives the EXP tag + experimental counter)", () => {
108
+ const tokens = tokenMapWith({ "#000": ["color.black"] });
109
+ const result = makeAuditResult([colorFinding("#fff")]);
110
+ const ctx = buildClassifyContext(result.findings, tokens, EMPTY_CONFIG, "/repo");
111
+ const populated = populateConfidence(result, ctx);
112
+ expect(populated.findings[0]?.confidence).toBe("low");
113
+ });
114
+ it("sets confidence on every finding even when the rule has no classifyConfidence (defaults to low)", () => {
115
+ // Rules without a classifyConfidence implementation get "low" via the
116
+ // safe-default dispatch in classifyConfidence(). Either way the field
117
+ // must be present on every emitted finding so the renderer + score gauge
118
+ // are not silently inert.
119
+ const findings = [
120
+ {
121
+ ruleId: "unknown-rule-id",
122
+ axis: "a11y",
123
+ severity: "warning",
124
+ location: { file: "x.tsx", line: 1, column: 1 },
125
+ message: "n/a",
126
+ },
127
+ ];
128
+ const result = makeAuditResult(findings);
129
+ const ctx = buildClassifyContext(findings, null, EMPTY_CONFIG);
130
+ const populated = populateConfidence(result, ctx);
131
+ expect(populated.findings[0]?.confidence).toBeDefined();
132
+ expect(populated.findings[0]?.confidence).toBe("low");
133
+ });
134
+ it("returns a new result + new finding objects (no in-place mutation)", () => {
135
+ const result = makeAuditResult([colorFinding("#fff")]);
136
+ const ctx = buildClassifyContext(result.findings, null, EMPTY_CONFIG);
137
+ const populated = populateConfidence(result, ctx);
138
+ expect(populated).not.toBe(result);
139
+ expect(populated.findings).not.toBe(result.findings);
140
+ expect(populated.findings[0]).not.toBe(result.findings[0]);
141
+ // The input result's finding must remain untouched.
142
+ expect(result.findings[0]?.confidence).toBeUndefined();
143
+ });
144
+ it("preserves non-finding AuditResult fields verbatim", () => {
145
+ const result = makeAuditResult([colorFinding("#fff")]);
146
+ const ctx = buildClassifyContext(result.findings, null, EMPTY_CONFIG);
147
+ const populated = populateConfidence(result, ctx);
148
+ expect(populated.finalScore).toBe(result.finalScore);
149
+ expect(populated.tier).toBe(result.tier);
150
+ expect(populated.toolVersion).toBe(result.toolVersion);
151
+ expect(populated.repoRoot).toBe(result.repoRoot);
152
+ });
153
+ });
154
+ //# sourceMappingURL=safety.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"safety.test.js","sourceRoot":"","sources":["../../src/codemods/safety.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import type { CodemodInput, CodemodResult } from "./index.js";
2
+ export declare function fixShadowNative(input: CodemodInput): CodemodResult;
@@ -0,0 +1,103 @@
1
+ import { singleLineDiff, prependLineDiff } from "./diff.js";
2
+ const NATIVE_TO_DS = {
3
+ button: "Button",
4
+ a: "Link",
5
+ input: "Input",
6
+ select: "Select",
7
+ textarea: "Textarea",
8
+ };
9
+ export function fixShadowNative(input) {
10
+ const ruleId = "components/no-native-shadows";
11
+ const { source, path, finding, ctx } = input;
12
+ if (!ctx.componentsModule) {
13
+ return {
14
+ patch: null,
15
+ confidence: 0,
16
+ alternatives: [],
17
+ rationale: "No componentsModule configured — cannot suggest a replacement.",
18
+ rule_id: ruleId,
19
+ schema_version: "1.0.0",
20
+ };
21
+ }
22
+ // Extract native tag from finding message format:
23
+ // "Native <button> used where <Button> from @acme/ui is available"
24
+ const tagMatch = finding.message.match(/Native <(\w+)>/);
25
+ if (!tagMatch) {
26
+ return {
27
+ patch: null,
28
+ confidence: 0,
29
+ alternatives: [],
30
+ rationale: "Could not parse native tag from finding message.",
31
+ rule_id: ruleId,
32
+ schema_version: "1.0.0",
33
+ };
34
+ }
35
+ const nativeTag = tagMatch[1].toLowerCase();
36
+ const dsName = NATIVE_TO_DS[nativeTag];
37
+ if (!dsName) {
38
+ return {
39
+ patch: null,
40
+ confidence: 0,
41
+ alternatives: [],
42
+ rationale: `No DS equivalent mapped for <${nativeTag}>.`,
43
+ rule_id: ruleId,
44
+ schema_version: "1.0.0",
45
+ };
46
+ }
47
+ const sourceLines = source.split("\n");
48
+ const lineIdx = finding.location.line - 1;
49
+ const sourceLine = sourceLines[lineIdx] ?? "";
50
+ if (!sourceLine.includes(`<${nativeTag}`)) {
51
+ return {
52
+ patch: null,
53
+ confidence: 0,
54
+ alternatives: [],
55
+ rationale: `Native <${nativeTag}> not found on line ${finding.location.line}.`,
56
+ rule_id: ruleId,
57
+ schema_version: "1.0.0",
58
+ };
59
+ }
60
+ // Check if the DS component is already imported from componentsModule
61
+ const importedAlready = source.split("\n").some((l) => {
62
+ return ((l.includes(`from "${ctx.componentsModule}"`) ||
63
+ l.includes(`from '${ctx.componentsModule}'`)) &&
64
+ new RegExp(`\\b${dsName}\\b`).test(l));
65
+ });
66
+ const oldFragment = `<${nativeTag}`;
67
+ const newFragment = `<${dsName}`;
68
+ const oldClosing = `</${nativeTag}>`;
69
+ // Check if the closing tag is on the same line — affects confidence
70
+ const closingOnSameLine = sourceLine.includes(oldClosing);
71
+ const confidence = closingOnSameLine ? 0.85 : 0.5;
72
+ const tagPatch = singleLineDiff(path, source, finding.location.line, oldFragment, newFragment);
73
+ if (importedAlready) {
74
+ return {
75
+ patch: tagPatch,
76
+ confidence,
77
+ alternatives: [],
78
+ rationale: closingOnSameLine
79
+ ? null
80
+ : `Closing </${nativeTag}> is not on the same line; the diff replaces the opening tag only. Verify the closing tag manually.`,
81
+ rule_id: ruleId,
82
+ schema_version: "1.0.0",
83
+ };
84
+ }
85
+ // Need to inject an import — find the line of the last import
86
+ let lastImportLine = 0;
87
+ for (let i = 0; i < sourceLines.length; i++) {
88
+ if (/^import\s/.test(sourceLines[i] ?? ""))
89
+ lastImportLine = i + 1;
90
+ }
91
+ const importPatch = prependLineDiff(path, source, `import { ${dsName} } from "${ctx.componentsModule}";`, lastImportLine);
92
+ return {
93
+ patch: importPatch + tagPatch,
94
+ confidence: confidence - 0.1, // chained patches are slightly less safe
95
+ alternatives: [],
96
+ rationale: closingOnSameLine
97
+ ? null
98
+ : `Closing </${nativeTag}> is not on the same line; the diff replaces the opening tag only. Verify the closing tag manually.`,
99
+ rule_id: ruleId,
100
+ schema_version: "1.0.0",
101
+ };
102
+ }
103
+ //# sourceMappingURL=shadow-native.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shadow-native.js","sourceRoot":"","sources":["../../src/codemods/shadow-native.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import type { CodemodInput, CodemodResult } from "./index.js";
2
+ export declare function fixHardcodedColor(input: CodemodInput): CodemodResult;
@@ -0,0 +1,69 @@
1
+ import { singleLineDiff } from "./diff.js";
2
+ const COLOR_VALUE_RE = /(#[0-9a-fA-F]{3,8}|rgb[a]?\([^)]+\)|hsl[a]?\([^)]+\)|oklch\([^)]+\))/;
3
+ export function fixHardcodedColor(input) {
4
+ const ruleId = "tokens/no-hardcoded-color";
5
+ const { source, path, finding, ctx } = input;
6
+ if (!ctx.tokens || ctx.tokens.colors.size === 0) {
7
+ return {
8
+ patch: null,
9
+ confidence: 0,
10
+ alternatives: [],
11
+ rationale: "No tokens loaded — cannot suggest a replacement.",
12
+ rule_id: ruleId,
13
+ schema_version: "1.0.0",
14
+ };
15
+ }
16
+ // Extract the hardcoded value from the finding's source line
17
+ const sourceLines = source.split("\n");
18
+ const sourceLine = sourceLines[finding.location.line - 1] ?? "";
19
+ const match = sourceLine.match(COLOR_VALUE_RE);
20
+ if (!match) {
21
+ return {
22
+ patch: null,
23
+ confidence: 0,
24
+ alternatives: [],
25
+ rationale: `Could not extract a hardcoded color from line ${finding.location.line}.`,
26
+ rule_id: ruleId,
27
+ schema_version: "1.0.0",
28
+ };
29
+ }
30
+ const value = match[0].toLowerCase();
31
+ const candidates = ctx.tokens.colors.get(value);
32
+ if (!candidates || candidates.length === 0) {
33
+ return {
34
+ patch: null,
35
+ confidence: 0,
36
+ alternatives: [],
37
+ rationale: `Color value ${match[0]} is not in the project's token map. Add the token or remove the hardcoded value.`,
38
+ rule_id: ruleId,
39
+ schema_version: "1.0.0",
40
+ };
41
+ }
42
+ if (candidates.length > 1) {
43
+ // Multiple candidates: produce alternatives but no primary patch
44
+ const alternatives = candidates.map((tokenName) => {
45
+ const patch = singleLineDiff(path, source, finding.location.line, match[0], `var(--color-${tokenName})`);
46
+ return { patch, rationale: `Replace with token ${tokenName}` };
47
+ });
48
+ return {
49
+ patch: null,
50
+ confidence: 0,
51
+ alternatives,
52
+ rationale: `Color ${match[0]} maps to multiple tokens (${candidates.join(", ")}). Pick one explicitly.`,
53
+ rule_id: ruleId,
54
+ schema_version: "1.0.0",
55
+ };
56
+ }
57
+ // Exactly one candidate
58
+ const tokenName = candidates[0];
59
+ const patch = singleLineDiff(path, source, finding.location.line, match[0], `var(--color-${tokenName})`);
60
+ return {
61
+ patch,
62
+ confidence: 0.95,
63
+ alternatives: [],
64
+ rationale: null,
65
+ rule_id: ruleId,
66
+ schema_version: "1.0.0",
67
+ };
68
+ }
69
+ //# sourceMappingURL=tokens-color.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tokens-color.js","sourceRoot":"","sources":["../../src/codemods/tokens-color.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import type { CodemodInput, CodemodResult } from "./index.js";
2
+ export declare function fixHardcodedSpacing(input: CodemodInput): CodemodResult;
@@ -0,0 +1,66 @@
1
+ import { singleLineDiff } from "./diff.js";
2
+ const SPACING_VALUE_RE = /(\d+(\.\d+)?)(px|rem|em)\b/;
3
+ export function fixHardcodedSpacing(input) {
4
+ const ruleId = "tokens/no-hardcoded-spacing";
5
+ const { source, path, finding, ctx } = input;
6
+ if (!ctx.tokens || ctx.tokens.spacing.size === 0) {
7
+ return {
8
+ patch: null,
9
+ confidence: 0,
10
+ alternatives: [],
11
+ rationale: "No spacing tokens loaded — cannot suggest a replacement.",
12
+ rule_id: ruleId,
13
+ schema_version: "1.0.0",
14
+ };
15
+ }
16
+ const sourceLines = source.split("\n");
17
+ const sourceLine = sourceLines[finding.location.line - 1] ?? "";
18
+ const match = sourceLine.match(SPACING_VALUE_RE);
19
+ if (!match) {
20
+ return {
21
+ patch: null,
22
+ confidence: 0,
23
+ alternatives: [],
24
+ rationale: `Could not extract a hardcoded spacing value from line ${finding.location.line}.`,
25
+ rule_id: ruleId,
26
+ schema_version: "1.0.0",
27
+ };
28
+ }
29
+ const numericValue = match[1];
30
+ const candidates = ctx.tokens.spacing.get(numericValue);
31
+ if (!candidates || candidates.length === 0) {
32
+ return {
33
+ patch: null,
34
+ confidence: 0,
35
+ alternatives: [],
36
+ rationale: `Value ${match[0]} is not in the spacing token scale.`,
37
+ rule_id: ruleId,
38
+ schema_version: "1.0.0",
39
+ };
40
+ }
41
+ if (candidates.length > 1) {
42
+ const alternatives = candidates.map((tokenName) => {
43
+ const patch = singleLineDiff(path, source, finding.location.line, match[0], `var(--spacing-${tokenName})`);
44
+ return { patch, rationale: `Replace with token ${tokenName}` };
45
+ });
46
+ return {
47
+ patch: null,
48
+ confidence: 0,
49
+ alternatives,
50
+ rationale: `Value ${match[0]} maps to multiple tokens (${candidates.join(", ")}).`,
51
+ rule_id: ruleId,
52
+ schema_version: "1.0.0",
53
+ };
54
+ }
55
+ const tokenName = candidates[0];
56
+ const patch = singleLineDiff(path, source, finding.location.line, match[0], `var(--spacing-${tokenName})`);
57
+ return {
58
+ patch,
59
+ confidence: 0.95,
60
+ alternatives: [],
61
+ rationale: null,
62
+ rule_id: ruleId,
63
+ schema_version: "1.0.0",
64
+ };
65
+ }
66
+ //# sourceMappingURL=tokens-spacing.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tokens-spacing.js","sourceRoot":"","sources":["../../src/codemods/tokens-spacing.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export {};