codesift-mcp 0.3.0 → 0.5.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 (543) hide show
  1. package/README.md +215 -23
  2. package/dist/cache/hono-cache.d.ts +50 -0
  3. package/dist/cache/hono-cache.d.ts.map +1 -0
  4. package/dist/cache/hono-cache.js +132 -0
  5. package/dist/cache/hono-cache.js.map +1 -0
  6. package/dist/cli/help.d.ts.map +1 -1
  7. package/dist/cli/help.js +8 -6
  8. package/dist/cli/help.js.map +1 -1
  9. package/dist/cli/platform.d.ts.map +1 -1
  10. package/dist/cli/platform.js +12 -14
  11. package/dist/cli/platform.js.map +1 -1
  12. package/dist/cli/setup.d.ts +1 -1
  13. package/dist/cli/setup.d.ts.map +1 -1
  14. package/dist/cli/setup.js +27 -3
  15. package/dist/cli/setup.js.map +1 -1
  16. package/dist/formatters-shortening.d.ts +13 -0
  17. package/dist/formatters-shortening.d.ts.map +1 -1
  18. package/dist/formatters-shortening.js +131 -0
  19. package/dist/formatters-shortening.js.map +1 -1
  20. package/dist/formatters.d.ts +38 -0
  21. package/dist/formatters.d.ts.map +1 -1
  22. package/dist/formatters.js +521 -0
  23. package/dist/formatters.js.map +1 -1
  24. package/dist/instructions.d.ts +1 -1
  25. package/dist/instructions.d.ts.map +1 -1
  26. package/dist/instructions.js +39 -38
  27. package/dist/instructions.js.map +1 -1
  28. package/dist/lsp/lsp-servers.d.ts.map +1 -1
  29. package/dist/lsp/lsp-servers.js +5 -0
  30. package/dist/lsp/lsp-servers.js.map +1 -1
  31. package/dist/lsp/lsp-tools.d.ts.map +1 -1
  32. package/dist/lsp/lsp-tools.js +1 -0
  33. package/dist/lsp/lsp-tools.js.map +1 -1
  34. package/dist/parser/astro-template.d.ts +47 -0
  35. package/dist/parser/astro-template.d.ts.map +1 -0
  36. package/dist/parser/astro-template.js +171 -0
  37. package/dist/parser/astro-template.js.map +1 -0
  38. package/dist/parser/extractors/_shared.d.ts +4 -0
  39. package/dist/parser/extractors/_shared.d.ts.map +1 -1
  40. package/dist/parser/extractors/_shared.js +8 -0
  41. package/dist/parser/extractors/_shared.js.map +1 -1
  42. package/dist/parser/extractors/astro.d.ts +4 -5
  43. package/dist/parser/extractors/astro.d.ts.map +1 -1
  44. package/dist/parser/extractors/astro.js +102 -26
  45. package/dist/parser/extractors/astro.js.map +1 -1
  46. package/dist/parser/extractors/gradle-kts.d.ts +4 -0
  47. package/dist/parser/extractors/gradle-kts.d.ts.map +1 -0
  48. package/dist/parser/extractors/gradle-kts.js +246 -0
  49. package/dist/parser/extractors/gradle-kts.js.map +1 -0
  50. package/dist/parser/extractors/hono-inline-analyzer.d.ts +34 -0
  51. package/dist/parser/extractors/hono-inline-analyzer.d.ts.map +1 -0
  52. package/dist/parser/extractors/hono-inline-analyzer.js +465 -0
  53. package/dist/parser/extractors/hono-inline-analyzer.js.map +1 -0
  54. package/dist/parser/extractors/hono-model.d.ts +196 -0
  55. package/dist/parser/extractors/hono-model.d.ts.map +1 -0
  56. package/dist/parser/extractors/hono-model.js +10 -0
  57. package/dist/parser/extractors/hono-model.js.map +1 -0
  58. package/dist/parser/extractors/hono.d.ts +118 -0
  59. package/dist/parser/extractors/hono.d.ts.map +1 -0
  60. package/dist/parser/extractors/hono.js +1527 -0
  61. package/dist/parser/extractors/hono.js.map +1 -0
  62. package/dist/parser/extractors/kotlin.d.ts +4 -0
  63. package/dist/parser/extractors/kotlin.d.ts.map +1 -0
  64. package/dist/parser/extractors/kotlin.js +521 -0
  65. package/dist/parser/extractors/kotlin.js.map +1 -0
  66. package/dist/parser/extractors/php.d.ts +22 -0
  67. package/dist/parser/extractors/php.d.ts.map +1 -0
  68. package/dist/parser/extractors/php.js +334 -0
  69. package/dist/parser/extractors/php.js.map +1 -0
  70. package/dist/parser/extractors/python.d.ts.map +1 -1
  71. package/dist/parser/extractors/python.js +234 -11
  72. package/dist/parser/extractors/python.js.map +1 -1
  73. package/dist/parser/extractors/sql.d.ts +33 -0
  74. package/dist/parser/extractors/sql.d.ts.map +1 -0
  75. package/dist/parser/extractors/sql.js +506 -0
  76. package/dist/parser/extractors/sql.js.map +1 -0
  77. package/dist/parser/extractors/typescript.d.ts.map +1 -1
  78. package/dist/parser/extractors/typescript.js +209 -3
  79. package/dist/parser/extractors/typescript.js.map +1 -1
  80. package/dist/parser/languages/tree-sitter-javascript.wasm +0 -0
  81. package/dist/parser/languages/tree-sitter-kotlin.wasm +0 -0
  82. package/dist/parser/languages/tree-sitter-php.wasm +0 -0
  83. package/dist/parser/languages/tree-sitter-php_only.wasm +0 -0
  84. package/dist/parser/languages/tree-sitter-python.wasm +0 -0
  85. package/dist/parser/parse-cache.d.ts +39 -0
  86. package/dist/parser/parse-cache.d.ts.map +1 -0
  87. package/dist/parser/parse-cache.js +87 -0
  88. package/dist/parser/parse-cache.js.map +1 -0
  89. package/dist/parser/parser-manager.d.ts +32 -0
  90. package/dist/parser/parser-manager.d.ts.map +1 -1
  91. package/dist/parser/parser-manager.js +93 -3
  92. package/dist/parser/parser-manager.js.map +1 -1
  93. package/dist/parser/symbol-extractor.d.ts.map +1 -1
  94. package/dist/parser/symbol-extractor.js +16 -0
  95. package/dist/parser/symbol-extractor.js.map +1 -1
  96. package/dist/register-tools.d.ts +38 -2
  97. package/dist/register-tools.d.ts.map +1 -1
  98. package/dist/register-tools.js +2444 -195
  99. package/dist/register-tools.js.map +1 -1
  100. package/dist/search/reranker.js +1 -1
  101. package/dist/search/reranker.js.map +1 -1
  102. package/dist/search/tool-ranker.d.ts +90 -0
  103. package/dist/search/tool-ranker.d.ts.map +1 -0
  104. package/dist/search/tool-ranker.js +420 -0
  105. package/dist/search/tool-ranker.js.map +1 -0
  106. package/dist/server-helpers.d.ts.map +1 -1
  107. package/dist/server-helpers.js +11 -0
  108. package/dist/server-helpers.js.map +1 -1
  109. package/dist/server.js +47 -14
  110. package/dist/server.js.map +1 -1
  111. package/dist/storage/index-store.d.ts +15 -1
  112. package/dist/storage/index-store.d.ts.map +1 -1
  113. package/dist/storage/index-store.js +27 -1
  114. package/dist/storage/index-store.js.map +1 -1
  115. package/dist/storage/session-state.d.ts +1 -1
  116. package/dist/storage/session-state.d.ts.map +1 -1
  117. package/dist/storage/session-state.js +6 -4
  118. package/dist/storage/session-state.js.map +1 -1
  119. package/dist/storage/usage-tracker.d.ts.map +1 -1
  120. package/dist/storage/usage-tracker.js +4 -1
  121. package/dist/storage/usage-tracker.js.map +1 -1
  122. package/dist/tools/agent-config-tools.d.ts +24 -0
  123. package/dist/tools/agent-config-tools.d.ts.map +1 -0
  124. package/dist/tools/agent-config-tools.js +119 -0
  125. package/dist/tools/agent-config-tools.js.map +1 -0
  126. package/dist/tools/architecture-tools.d.ts +23 -0
  127. package/dist/tools/architecture-tools.d.ts.map +1 -0
  128. package/dist/tools/architecture-tools.js +140 -0
  129. package/dist/tools/architecture-tools.js.map +1 -0
  130. package/dist/tools/astro-actions.d.ts +54 -0
  131. package/dist/tools/astro-actions.d.ts.map +1 -0
  132. package/dist/tools/astro-actions.js +561 -0
  133. package/dist/tools/astro-actions.js.map +1 -0
  134. package/dist/tools/astro-audit.d.ts +87 -0
  135. package/dist/tools/astro-audit.d.ts.map +1 -0
  136. package/dist/tools/astro-audit.js +345 -0
  137. package/dist/tools/astro-audit.js.map +1 -0
  138. package/dist/tools/astro-config.d.ts +33 -0
  139. package/dist/tools/astro-config.d.ts.map +1 -0
  140. package/dist/tools/astro-config.js +260 -0
  141. package/dist/tools/astro-config.js.map +1 -0
  142. package/dist/tools/astro-content-collections.d.ts +44 -0
  143. package/dist/tools/astro-content-collections.d.ts.map +1 -0
  144. package/dist/tools/astro-content-collections.js +630 -0
  145. package/dist/tools/astro-content-collections.js.map +1 -0
  146. package/dist/tools/astro-islands.d.ts +63 -0
  147. package/dist/tools/astro-islands.d.ts.map +1 -0
  148. package/dist/tools/astro-islands.js +255 -0
  149. package/dist/tools/astro-islands.js.map +1 -0
  150. package/dist/tools/astro-migration.d.ts +31 -0
  151. package/dist/tools/astro-migration.d.ts.map +1 -0
  152. package/dist/tools/astro-migration.js +378 -0
  153. package/dist/tools/astro-migration.js.map +1 -0
  154. package/dist/tools/astro-routes.d.ts +49 -0
  155. package/dist/tools/astro-routes.d.ts.map +1 -0
  156. package/dist/tools/astro-routes.js +119 -0
  157. package/dist/tools/astro-routes.js.map +1 -0
  158. package/dist/tools/async-correctness.d.ts +26 -0
  159. package/dist/tools/async-correctness.d.ts.map +1 -0
  160. package/dist/tools/async-correctness.js +166 -0
  161. package/dist/tools/async-correctness.js.map +1 -0
  162. package/dist/tools/audit-tools.d.ts +38 -0
  163. package/dist/tools/audit-tools.d.ts.map +1 -0
  164. package/dist/tools/audit-tools.js +248 -0
  165. package/dist/tools/audit-tools.js.map +1 -0
  166. package/dist/tools/celery-tools.d.ts +38 -0
  167. package/dist/tools/celery-tools.d.ts.map +1 -0
  168. package/dist/tools/celery-tools.js +154 -0
  169. package/dist/tools/celery-tools.js.map +1 -0
  170. package/dist/tools/clone-tools.js +1 -1
  171. package/dist/tools/clone-tools.js.map +1 -1
  172. package/dist/tools/complexity-tools.d.ts +4 -0
  173. package/dist/tools/complexity-tools.d.ts.map +1 -1
  174. package/dist/tools/complexity-tools.js +78 -4
  175. package/dist/tools/complexity-tools.js.map +1 -1
  176. package/dist/tools/compose-tools.d.ts +60 -0
  177. package/dist/tools/compose-tools.d.ts.map +1 -0
  178. package/dist/tools/compose-tools.js +203 -0
  179. package/dist/tools/compose-tools.js.map +1 -0
  180. package/dist/tools/coupling-tools.d.ts +50 -0
  181. package/dist/tools/coupling-tools.d.ts.map +1 -0
  182. package/dist/tools/coupling-tools.js +262 -0
  183. package/dist/tools/coupling-tools.js.map +1 -0
  184. package/dist/tools/dependency-audit-tools.d.ts +65 -0
  185. package/dist/tools/dependency-audit-tools.d.ts.map +1 -0
  186. package/dist/tools/dependency-audit-tools.js +553 -0
  187. package/dist/tools/dependency-audit-tools.js.map +1 -0
  188. package/dist/tools/django-settings.d.ts +22 -0
  189. package/dist/tools/django-settings.d.ts.map +1 -0
  190. package/dist/tools/django-settings.js +301 -0
  191. package/dist/tools/django-settings.js.map +1 -0
  192. package/dist/tools/django-view-security-tools.d.ts +32 -0
  193. package/dist/tools/django-view-security-tools.d.ts.map +1 -0
  194. package/dist/tools/django-view-security-tools.js +184 -0
  195. package/dist/tools/django-view-security-tools.js.map +1 -0
  196. package/dist/tools/fastapi-depends.d.ts +63 -0
  197. package/dist/tools/fastapi-depends.d.ts.map +1 -0
  198. package/dist/tools/fastapi-depends.js +191 -0
  199. package/dist/tools/fastapi-depends.js.map +1 -0
  200. package/dist/tools/frequency-tools.js +1 -1
  201. package/dist/tools/frequency-tools.js.map +1 -1
  202. package/dist/tools/graph-tools.d.ts +8 -2
  203. package/dist/tools/graph-tools.d.ts.map +1 -1
  204. package/dist/tools/graph-tools.js +44 -3
  205. package/dist/tools/graph-tools.js.map +1 -1
  206. package/dist/tools/hilt-tools.d.ts +55 -0
  207. package/dist/tools/hilt-tools.d.ts.map +1 -0
  208. package/dist/tools/hilt-tools.js +258 -0
  209. package/dist/tools/hilt-tools.js.map +1 -0
  210. package/dist/tools/hono-analyze-app.d.ts +48 -0
  211. package/dist/tools/hono-analyze-app.d.ts.map +1 -0
  212. package/dist/tools/hono-analyze-app.js +94 -0
  213. package/dist/tools/hono-analyze-app.js.map +1 -0
  214. package/dist/tools/hono-api-contract.d.ts +22 -0
  215. package/dist/tools/hono-api-contract.d.ts.map +1 -0
  216. package/dist/tools/hono-api-contract.js +112 -0
  217. package/dist/tools/hono-api-contract.js.map +1 -0
  218. package/dist/tools/hono-conditional-middleware.d.ts +27 -0
  219. package/dist/tools/hono-conditional-middleware.d.ts.map +1 -0
  220. package/dist/tools/hono-conditional-middleware.js +62 -0
  221. package/dist/tools/hono-conditional-middleware.js.map +1 -0
  222. package/dist/tools/hono-context-flow.d.ts +24 -0
  223. package/dist/tools/hono-context-flow.d.ts.map +1 -0
  224. package/dist/tools/hono-context-flow.js +70 -0
  225. package/dist/tools/hono-context-flow.js.map +1 -0
  226. package/dist/tools/hono-dead-routes.d.ts +26 -0
  227. package/dist/tools/hono-dead-routes.d.ts.map +1 -0
  228. package/dist/tools/hono-dead-routes.js +102 -0
  229. package/dist/tools/hono-dead-routes.js.map +1 -0
  230. package/dist/tools/hono-entry-resolver.d.ts +27 -0
  231. package/dist/tools/hono-entry-resolver.d.ts.map +1 -0
  232. package/dist/tools/hono-entry-resolver.js +31 -0
  233. package/dist/tools/hono-entry-resolver.js.map +1 -0
  234. package/dist/tools/hono-env-regression.d.ts +29 -0
  235. package/dist/tools/hono-env-regression.d.ts.map +1 -0
  236. package/dist/tools/hono-env-regression.js +157 -0
  237. package/dist/tools/hono-env-regression.js.map +1 -0
  238. package/dist/tools/hono-inline-analyze.d.ts +31 -0
  239. package/dist/tools/hono-inline-analyze.d.ts.map +1 -0
  240. package/dist/tools/hono-inline-analyze.js +59 -0
  241. package/dist/tools/hono-inline-analyze.js.map +1 -0
  242. package/dist/tools/hono-middleware-chain.d.ts +40 -0
  243. package/dist/tools/hono-middleware-chain.d.ts.map +1 -0
  244. package/dist/tools/hono-middleware-chain.js +121 -0
  245. package/dist/tools/hono-middleware-chain.js.map +1 -0
  246. package/dist/tools/hono-modules.d.ts +22 -0
  247. package/dist/tools/hono-modules.d.ts.map +1 -0
  248. package/dist/tools/hono-modules.js +118 -0
  249. package/dist/tools/hono-modules.js.map +1 -0
  250. package/dist/tools/hono-response-types.d.ts +37 -0
  251. package/dist/tools/hono-response-types.d.ts.map +1 -0
  252. package/dist/tools/hono-response-types.js +76 -0
  253. package/dist/tools/hono-response-types.js.map +1 -0
  254. package/dist/tools/hono-rpc-types.d.ts +21 -0
  255. package/dist/tools/hono-rpc-types.d.ts.map +1 -0
  256. package/dist/tools/hono-rpc-types.js +49 -0
  257. package/dist/tools/hono-rpc-types.js.map +1 -0
  258. package/dist/tools/hono-security.d.ts +31 -0
  259. package/dist/tools/hono-security.d.ts.map +1 -0
  260. package/dist/tools/hono-security.js +269 -0
  261. package/dist/tools/hono-security.js.map +1 -0
  262. package/dist/tools/hono-visualize.d.ts +13 -0
  263. package/dist/tools/hono-visualize.d.ts.map +1 -0
  264. package/dist/tools/hono-visualize.js +64 -0
  265. package/dist/tools/hono-visualize.js.map +1 -0
  266. package/dist/tools/hotspot-tools.d.ts.map +1 -1
  267. package/dist/tools/hotspot-tools.js +9 -7
  268. package/dist/tools/hotspot-tools.js.map +1 -1
  269. package/dist/tools/index-tools.d.ts +17 -0
  270. package/dist/tools/index-tools.d.ts.map +1 -1
  271. package/dist/tools/index-tools.js +210 -10
  272. package/dist/tools/index-tools.js.map +1 -1
  273. package/dist/tools/kotlin-tools.d.ts +142 -0
  274. package/dist/tools/kotlin-tools.d.ts.map +1 -0
  275. package/dist/tools/kotlin-tools.js +572 -0
  276. package/dist/tools/kotlin-tools.js.map +1 -0
  277. package/dist/tools/legacy-hono-conventions.d.ts +14 -0
  278. package/dist/tools/legacy-hono-conventions.d.ts.map +1 -0
  279. package/dist/tools/legacy-hono-conventions.js +152 -0
  280. package/dist/tools/legacy-hono-conventions.js.map +1 -0
  281. package/dist/tools/migration-lint-tools.d.ts +26 -0
  282. package/dist/tools/migration-lint-tools.d.ts.map +1 -0
  283. package/dist/tools/migration-lint-tools.js +247 -0
  284. package/dist/tools/migration-lint-tools.js.map +1 -0
  285. package/dist/tools/model-tools.d.ts +30 -0
  286. package/dist/tools/model-tools.d.ts.map +1 -0
  287. package/dist/tools/model-tools.js +145 -0
  288. package/dist/tools/model-tools.js.map +1 -0
  289. package/dist/tools/nest-ext-tools.d.ts +207 -0
  290. package/dist/tools/nest-ext-tools.d.ts.map +1 -0
  291. package/dist/tools/nest-ext-tools.js +752 -0
  292. package/dist/tools/nest-ext-tools.js.map +1 -0
  293. package/dist/tools/nest-tools.d.ts +198 -0
  294. package/dist/tools/nest-tools.d.ts.map +1 -0
  295. package/dist/tools/nest-tools.js +1142 -0
  296. package/dist/tools/nest-tools.js.map +1 -0
  297. package/dist/tools/nextjs-api-contract-readers.d.ts +14 -0
  298. package/dist/tools/nextjs-api-contract-readers.d.ts.map +1 -0
  299. package/dist/tools/nextjs-api-contract-readers.js +204 -0
  300. package/dist/tools/nextjs-api-contract-readers.js.map +1 -0
  301. package/dist/tools/nextjs-api-contract-tools.d.ts +57 -0
  302. package/dist/tools/nextjs-api-contract-tools.d.ts.map +1 -0
  303. package/dist/tools/nextjs-api-contract-tools.js +144 -0
  304. package/dist/tools/nextjs-api-contract-tools.js.map +1 -0
  305. package/dist/tools/nextjs-boundary-tools.d.ts +39 -0
  306. package/dist/tools/nextjs-boundary-tools.d.ts.map +1 -0
  307. package/dist/tools/nextjs-boundary-tools.js +152 -0
  308. package/dist/tools/nextjs-boundary-tools.js.map +1 -0
  309. package/dist/tools/nextjs-component-readers.d.ts +101 -0
  310. package/dist/tools/nextjs-component-readers.d.ts.map +1 -0
  311. package/dist/tools/nextjs-component-readers.js +287 -0
  312. package/dist/tools/nextjs-component-readers.js.map +1 -0
  313. package/dist/tools/nextjs-component-tools.d.ts +51 -0
  314. package/dist/tools/nextjs-component-tools.d.ts.map +1 -0
  315. package/dist/tools/nextjs-component-tools.js +212 -0
  316. package/dist/tools/nextjs-component-tools.js.map +1 -0
  317. package/dist/tools/nextjs-data-flow-tools.d.ts +42 -0
  318. package/dist/tools/nextjs-data-flow-tools.d.ts.map +1 -0
  319. package/dist/tools/nextjs-data-flow-tools.js +158 -0
  320. package/dist/tools/nextjs-data-flow-tools.js.map +1 -0
  321. package/dist/tools/nextjs-framework-audit-tools.d.ts +60 -0
  322. package/dist/tools/nextjs-framework-audit-tools.d.ts.map +1 -0
  323. package/dist/tools/nextjs-framework-audit-tools.js +394 -0
  324. package/dist/tools/nextjs-framework-audit-tools.js.map +1 -0
  325. package/dist/tools/nextjs-link-tools.d.ts +41 -0
  326. package/dist/tools/nextjs-link-tools.d.ts.map +1 -0
  327. package/dist/tools/nextjs-link-tools.js +157 -0
  328. package/dist/tools/nextjs-link-tools.js.map +1 -0
  329. package/dist/tools/nextjs-metadata-tools.d.ts +74 -0
  330. package/dist/tools/nextjs-metadata-tools.d.ts.map +1 -0
  331. package/dist/tools/nextjs-metadata-tools.js +252 -0
  332. package/dist/tools/nextjs-metadata-tools.js.map +1 -0
  333. package/dist/tools/nextjs-middleware-coverage-tools.d.ts +41 -0
  334. package/dist/tools/nextjs-middleware-coverage-tools.d.ts.map +1 -0
  335. package/dist/tools/nextjs-middleware-coverage-tools.js +88 -0
  336. package/dist/tools/nextjs-middleware-coverage-tools.js.map +1 -0
  337. package/dist/tools/nextjs-route-readers.d.ts +81 -0
  338. package/dist/tools/nextjs-route-readers.d.ts.map +1 -0
  339. package/dist/tools/nextjs-route-readers.js +340 -0
  340. package/dist/tools/nextjs-route-readers.js.map +1 -0
  341. package/dist/tools/nextjs-route-tools.d.ts +36 -0
  342. package/dist/tools/nextjs-route-tools.d.ts.map +1 -0
  343. package/dist/tools/nextjs-route-tools.js +175 -0
  344. package/dist/tools/nextjs-route-tools.js.map +1 -0
  345. package/dist/tools/nextjs-security-readers.d.ts +22 -0
  346. package/dist/tools/nextjs-security-readers.d.ts.map +1 -0
  347. package/dist/tools/nextjs-security-readers.js +318 -0
  348. package/dist/tools/nextjs-security-readers.js.map +1 -0
  349. package/dist/tools/nextjs-security-scoring.d.ts +15 -0
  350. package/dist/tools/nextjs-security-scoring.d.ts.map +1 -0
  351. package/dist/tools/nextjs-security-scoring.js +65 -0
  352. package/dist/tools/nextjs-security-scoring.js.map +1 -0
  353. package/dist/tools/nextjs-security-tools.d.ts +75 -0
  354. package/dist/tools/nextjs-security-tools.d.ts.map +1 -0
  355. package/dist/tools/nextjs-security-tools.js +153 -0
  356. package/dist/tools/nextjs-security-tools.js.map +1 -0
  357. package/dist/tools/nextjs-tools.d.ts +15 -0
  358. package/dist/tools/nextjs-tools.d.ts.map +1 -0
  359. package/dist/tools/nextjs-tools.js +15 -0
  360. package/dist/tools/nextjs-tools.js.map +1 -0
  361. package/dist/tools/outline-tools.d.ts.map +1 -1
  362. package/dist/tools/outline-tools.js +20 -0
  363. package/dist/tools/outline-tools.js.map +1 -1
  364. package/dist/tools/pattern-tools.d.ts +8 -0
  365. package/dist/tools/pattern-tools.d.ts.map +1 -1
  366. package/dist/tools/pattern-tools.js +651 -3
  367. package/dist/tools/pattern-tools.js.map +1 -1
  368. package/dist/tools/perf-tools.d.ts +32 -0
  369. package/dist/tools/perf-tools.d.ts.map +1 -0
  370. package/dist/tools/perf-tools.js +227 -0
  371. package/dist/tools/perf-tools.js.map +1 -0
  372. package/dist/tools/php-tools.d.ts +185 -0
  373. package/dist/tools/php-tools.d.ts.map +1 -0
  374. package/dist/tools/php-tools.js +645 -0
  375. package/dist/tools/php-tools.js.map +1 -0
  376. package/dist/tools/plan-turn-tools.d.ts +89 -0
  377. package/dist/tools/plan-turn-tools.d.ts.map +1 -0
  378. package/dist/tools/plan-turn-tools.js +508 -0
  379. package/dist/tools/plan-turn-tools.js.map +1 -0
  380. package/dist/tools/prisma-schema-tools.d.ts +44 -0
  381. package/dist/tools/prisma-schema-tools.d.ts.map +1 -0
  382. package/dist/tools/prisma-schema-tools.js +358 -0
  383. package/dist/tools/prisma-schema-tools.js.map +1 -0
  384. package/dist/tools/project-tools.d.ts +116 -7
  385. package/dist/tools/project-tools.d.ts.map +1 -1
  386. package/dist/tools/project-tools.js +595 -218
  387. package/dist/tools/project-tools.js.map +1 -1
  388. package/dist/tools/pydantic-models.d.ts +46 -0
  389. package/dist/tools/pydantic-models.d.ts.map +1 -0
  390. package/dist/tools/pydantic-models.js +249 -0
  391. package/dist/tools/pydantic-models.js.map +1 -0
  392. package/dist/tools/pyproject-tools.d.ts +23 -0
  393. package/dist/tools/pyproject-tools.d.ts.map +1 -0
  394. package/dist/tools/pyproject-tools.js +133 -0
  395. package/dist/tools/pyproject-tools.js.map +1 -0
  396. package/dist/tools/pytest-tools.d.ts +20 -0
  397. package/dist/tools/pytest-tools.d.ts.map +1 -0
  398. package/dist/tools/pytest-tools.js +106 -0
  399. package/dist/tools/pytest-tools.js.map +1 -0
  400. package/dist/tools/python-audit.d.ts +40 -0
  401. package/dist/tools/python-audit.d.ts.map +1 -0
  402. package/dist/tools/python-audit.js +244 -0
  403. package/dist/tools/python-audit.js.map +1 -0
  404. package/dist/tools/python-callers.d.ts +28 -0
  405. package/dist/tools/python-callers.d.ts.map +1 -0
  406. package/dist/tools/python-callers.js +110 -0
  407. package/dist/tools/python-callers.js.map +1 -0
  408. package/dist/tools/python-circular-imports.d.ts +19 -0
  409. package/dist/tools/python-circular-imports.d.ts.map +1 -0
  410. package/dist/tools/python-circular-imports.js +126 -0
  411. package/dist/tools/python-circular-imports.js.map +1 -0
  412. package/dist/tools/python-constants-tools.d.ts +44 -0
  413. package/dist/tools/python-constants-tools.d.ts.map +1 -0
  414. package/dist/tools/python-constants-tools.js +525 -0
  415. package/dist/tools/python-constants-tools.js.map +1 -0
  416. package/dist/tools/python-deps-analyzer.d.ts +46 -0
  417. package/dist/tools/python-deps-analyzer.d.ts.map +1 -0
  418. package/dist/tools/python-deps-analyzer.js +227 -0
  419. package/dist/tools/python-deps-analyzer.js.map +1 -0
  420. package/dist/tools/query-tools.d.ts +23 -0
  421. package/dist/tools/query-tools.d.ts.map +1 -0
  422. package/dist/tools/query-tools.js +256 -0
  423. package/dist/tools/query-tools.js.map +1 -0
  424. package/dist/tools/react-tools.d.ts +263 -0
  425. package/dist/tools/react-tools.d.ts.map +1 -0
  426. package/dist/tools/react-tools.js +839 -0
  427. package/dist/tools/react-tools.js.map +1 -0
  428. package/dist/tools/report-tools.js +47 -0
  429. package/dist/tools/report-tools.js.map +1 -1
  430. package/dist/tools/review-diff-tools.d.ts +5 -4
  431. package/dist/tools/review-diff-tools.d.ts.map +1 -1
  432. package/dist/tools/review-diff-tools.js +157 -66
  433. package/dist/tools/review-diff-tools.js.map +1 -1
  434. package/dist/tools/room-tools.d.ts +36 -0
  435. package/dist/tools/room-tools.d.ts.map +1 -0
  436. package/dist/tools/room-tools.js +147 -0
  437. package/dist/tools/room-tools.js.map +1 -0
  438. package/dist/tools/route-tools.d.ts +27 -1
  439. package/dist/tools/route-tools.d.ts.map +1 -1
  440. package/dist/tools/route-tools.js +744 -18
  441. package/dist/tools/route-tools.js.map +1 -1
  442. package/dist/tools/ruff-tools.d.ts +32 -0
  443. package/dist/tools/ruff-tools.d.ts.map +1 -0
  444. package/dist/tools/ruff-tools.js +114 -0
  445. package/dist/tools/ruff-tools.js.map +1 -0
  446. package/dist/tools/search-ranker.d.ts.map +1 -1
  447. package/dist/tools/search-ranker.js +7 -0
  448. package/dist/tools/search-ranker.js.map +1 -1
  449. package/dist/tools/search-tools.d.ts +3 -2
  450. package/dist/tools/search-tools.d.ts.map +1 -1
  451. package/dist/tools/search-tools.js +16 -3
  452. package/dist/tools/search-tools.js.map +1 -1
  453. package/dist/tools/serialization-tools.d.ts +24 -0
  454. package/dist/tools/serialization-tools.d.ts.map +1 -0
  455. package/dist/tools/serialization-tools.js +156 -0
  456. package/dist/tools/serialization-tools.js.map +1 -0
  457. package/dist/tools/sql-tools.d.ts +274 -0
  458. package/dist/tools/sql-tools.d.ts.map +1 -0
  459. package/dist/tools/sql-tools.js +1160 -0
  460. package/dist/tools/sql-tools.js.map +1 -0
  461. package/dist/tools/status-tools.d.ts +10 -0
  462. package/dist/tools/status-tools.d.ts.map +1 -0
  463. package/dist/tools/status-tools.js +32 -0
  464. package/dist/tools/status-tools.js.map +1 -0
  465. package/dist/tools/symbol-tools.d.ts +19 -0
  466. package/dist/tools/symbol-tools.d.ts.map +1 -1
  467. package/dist/tools/symbol-tools.js +75 -4
  468. package/dist/tools/symbol-tools.js.map +1 -1
  469. package/dist/tools/taint-tools.d.ts +43 -0
  470. package/dist/tools/taint-tools.d.ts.map +1 -0
  471. package/dist/tools/taint-tools.js +922 -0
  472. package/dist/tools/taint-tools.js.map +1 -0
  473. package/dist/tools/test-impact-tools.d.ts +29 -0
  474. package/dist/tools/test-impact-tools.d.ts.map +1 -0
  475. package/dist/tools/test-impact-tools.js +156 -0
  476. package/dist/tools/test-impact-tools.js.map +1 -0
  477. package/dist/tools/typecheck-tools.d.ts +39 -0
  478. package/dist/tools/typecheck-tools.d.ts.map +1 -0
  479. package/dist/tools/typecheck-tools.js +191 -0
  480. package/dist/tools/typecheck-tools.js.map +1 -0
  481. package/dist/tools/wiring-tools.d.ts +19 -0
  482. package/dist/tools/wiring-tools.d.ts.map +1 -0
  483. package/dist/tools/wiring-tools.js +147 -0
  484. package/dist/tools/wiring-tools.js.map +1 -0
  485. package/dist/types.d.ts +9 -1
  486. package/dist/types.d.ts.map +1 -1
  487. package/dist/utils/framework-detect.d.ts +18 -2
  488. package/dist/utils/framework-detect.d.ts.map +1 -1
  489. package/dist/utils/framework-detect.js +150 -3
  490. package/dist/utils/framework-detect.js.map +1 -1
  491. package/dist/utils/import-graph.d.ts +42 -0
  492. package/dist/utils/import-graph.d.ts.map +1 -1
  493. package/dist/utils/import-graph.js +248 -9
  494. package/dist/utils/import-graph.js.map +1 -1
  495. package/dist/utils/language-detect.d.ts +21 -0
  496. package/dist/utils/language-detect.d.ts.map +1 -0
  497. package/dist/utils/language-detect.js +183 -0
  498. package/dist/utils/language-detect.js.map +1 -0
  499. package/dist/utils/nextjs-ast-readers.d.ts +44 -0
  500. package/dist/utils/nextjs-ast-readers.d.ts.map +1 -0
  501. package/dist/utils/nextjs-ast-readers.js +341 -0
  502. package/dist/utils/nextjs-ast-readers.js.map +1 -0
  503. package/dist/utils/nextjs-audit-cache.d.ts +51 -0
  504. package/dist/utils/nextjs-audit-cache.d.ts.map +1 -0
  505. package/dist/utils/nextjs-audit-cache.js +116 -0
  506. package/dist/utils/nextjs-audit-cache.js.map +1 -0
  507. package/dist/utils/nextjs-metadata-readers.d.ts +65 -0
  508. package/dist/utils/nextjs-metadata-readers.d.ts.map +1 -0
  509. package/dist/utils/nextjs-metadata-readers.js +447 -0
  510. package/dist/utils/nextjs-metadata-readers.js.map +1 -0
  511. package/dist/utils/nextjs.d.ts +42 -0
  512. package/dist/utils/nextjs.d.ts.map +1 -0
  513. package/dist/utils/nextjs.js +284 -0
  514. package/dist/utils/nextjs.js.map +1 -0
  515. package/dist/utils/python-import-resolver.d.ts +42 -0
  516. package/dist/utils/python-import-resolver.d.ts.map +1 -0
  517. package/dist/utils/python-import-resolver.js +101 -0
  518. package/dist/utils/python-import-resolver.js.map +1 -0
  519. package/dist/utils/python-imports.d.ts +28 -0
  520. package/dist/utils/python-imports.d.ts.map +1 -0
  521. package/dist/utils/python-imports.js +117 -0
  522. package/dist/utils/python-imports.js.map +1 -0
  523. package/dist/utils/react-alias.d.ts +15 -0
  524. package/dist/utils/react-alias.d.ts.map +1 -0
  525. package/dist/utils/react-alias.js +31 -0
  526. package/dist/utils/react-alias.js.map +1 -0
  527. package/dist/utils/test-file.d.ts.map +1 -1
  528. package/dist/utils/test-file.js +7 -0
  529. package/dist/utils/test-file.js.map +1 -1
  530. package/dist/utils/walk.d.ts +22 -0
  531. package/dist/utils/walk.d.ts.map +1 -1
  532. package/dist/utils/walk.js +70 -2
  533. package/dist/utils/walk.js.map +1 -1
  534. package/package.json +4 -3
  535. package/rules/codesift.md +71 -5
  536. package/rules/codesift.mdc +71 -5
  537. package/rules/codex.md +71 -5
  538. package/rules/gemini.md +71 -5
  539. package/src/parser/languages/tree-sitter-javascript.wasm +0 -0
  540. package/src/parser/languages/tree-sitter-kotlin.wasm +0 -0
  541. package/src/parser/languages/tree-sitter-php.wasm +0 -0
  542. package/src/parser/languages/tree-sitter-php_only.wasm +0 -0
  543. package/src/parser/languages/tree-sitter-python.wasm +0 -0
@@ -1,11 +1,187 @@
1
+ import { readFile } from "node:fs/promises";
2
+ import { join } from "node:path";
1
3
  import { getCodeIndex } from "./index-tools.js";
2
4
  import { isTestFileStrict as isTestFile } from "../utils/test-file.js";
3
5
  // Built-in patterns inspired by CQ checklist + common React/TS anti-patterns
4
- const BUILTIN_PATTERNS = {
6
+ // Exported for direct regex testing in unit tests.
7
+ export const BUILTIN_PATTERNS = {
5
8
  "useEffect-no-cleanup": {
6
9
  regex: /useEffect\s*\(\s*(?:async\s*)?\(\)\s*=>\s*\{(?:(?!return\s*\(\s*\)\s*=>|return\s+\(\)\s*=>|return\s*\(\s*\)\s*\{|return\s+function)[\s\S])*\}\s*,/,
7
10
  description: "useEffect without cleanup return — potential memory leak (CQ22)",
8
11
  },
12
+ // --- React anti-patterns (Wave 2) ---
13
+ "hook-in-condition": {
14
+ regex: /\b(?:if|for|while|switch)\s*\([^)]*\)\s*\{[\s\S]{0,500}?\buse[A-Z]\w*\s*\(/,
15
+ description: "React hook called inside if/for/while/switch — violates Rule of Hooks",
16
+ },
17
+ "useEffect-async": {
18
+ regex: /useEffect\s*\(\s*async\s+(?:function\b|\(|[a-z_$])/,
19
+ description: "async function directly in useEffect — use inner async wrapper (CQ22)",
20
+ },
21
+ "useEffect-object-dep": {
22
+ regex: /useEffect\s*\([\s\S]*?,\s*\[[^\]]*[{[]/,
23
+ description: "Object/array literal in useEffect dependency array — causes infinite re-renders",
24
+ },
25
+ "missing-display-name": {
26
+ regex: /(?:React\.)?(?:memo|forwardRef)\s*\((?:(?!displayName)[\s\S]){0,500}$/,
27
+ description: "React.memo/forwardRef without displayName nearby — harder to debug in DevTools",
28
+ },
29
+ "index-as-key": {
30
+ regex: /\.map\s*\(\s*\(\s*\w+\s*,\s*(index|idx|i)\b[^)]*\)\s*=>[\s\S]{0,400}?key\s*=\s*\{?\s*\1\b/,
31
+ description: "Array index used as React key — causes incorrect reconciliation on reorder",
32
+ },
33
+ "inline-handler": {
34
+ regex: /\bon[A-Z]\w*\s*=\s*\{\s*(?:\([^)]*\)|[a-z_$][\w$]*)\s*=>/,
35
+ description: "Inline arrow function in JSX event handler — creates new reference every render (memoization killer)",
36
+ },
37
+ "conditional-render-hook": {
38
+ regex: /\breturn\s+[^;{]*;\s*\n[\s\S]*?\buse[A-Z]\w*\s*\(/,
39
+ description: "React hook called after early return — violates Rule of Hooks",
40
+ },
41
+ // --- React anti-patterns (Wave 4b — additional) ---
42
+ "dangerously-set-html": {
43
+ regex: /dangerouslySetInnerHTML\s*=\s*\{/,
44
+ description: "dangerouslySetInnerHTML used — XSS risk unless content is sanitized (CQ24)",
45
+ },
46
+ "direct-dom-access": {
47
+ regex: /\bdocument\.(getElementById|querySelector|querySelectorAll|getElementsBy)\s*\(/,
48
+ description: "Direct DOM access in React component — use useRef instead (breaks SSR, bypasses virtual DOM)",
49
+ },
50
+ "unstable-default-value": {
51
+ regex: /(?:function\s+[A-Z]\w*|const\s+[A-Z]\w*\s*=\s*(?:\([^)]*\)|[^=]*)\s*=>)\s*[\s\S]{0,100}(?:\{\s*[^}]*=\s*\[\s*\]|\{\s*[^}]*=\s*\{\s*\})/,
52
+ description: "Default prop value [] or {} in component params — creates new reference every render, breaks memo/PureComponent",
53
+ },
54
+ "jsx-falsy-and": {
55
+ regex: /\{\s*(?:count|length|size|num|total|amount)\s*&&\s*</,
56
+ description: "Numeric variable used with && in JSX — renders '0' on screen when falsy. Use ternary or Boolean() (React gotcha)",
57
+ },
58
+ "nested-component-def": {
59
+ regex: /(?:function\s+[A-Z]\w*\s*\([^)]*\)\s*\{|const\s+[A-Z]\w*\s*=\s*(?:\([^)]*\)|[\w$]*)\s*=>\s*\{)[\s\S]{0,1500}?\n\s{2,}(?:function\s+[A-Z]\w*\s*\(|const\s+[A-Z]\w*\s*=\s*\()/,
60
+ description: "Component defined inside another component — remounts on every parent render, loses all state. Hoist to module level.",
61
+ },
62
+ "usecallback-no-deps": {
63
+ regex: /use(?:Callback|Memo)\s*\([\s\S]*?\)\s*\)\s*[;,]/,
64
+ description: "useCallback/useMemo with only one argument (no dependency array) — useless memoization, value recreated every render",
65
+ },
66
+ // --- React 19 features (Tier 4 — Item 19) ---
67
+ "react19-use-without-suspense": {
68
+ regex: /\buse\s*\(\s*[a-zA-Z_$][\w$]*\s*\)/,
69
+ description: "React 19 use(promise) — must be wrapped in <Suspense> or it throws. Verify Suspense boundary exists in parent.",
70
+ fileIncludePattern: /\.(tsx|jsx)$/,
71
+ },
72
+ "react19-server-action-not-async": {
73
+ regex: /^[\s\S]{0,200}["']use server["'][\s\S]{0,500}?\bexport\s+(?!async)function\s+\w+\s*\(/m,
74
+ description: "React 19 Server Action: function in 'use server' file must be async (returns Promise). Synchronous functions break the action contract.",
75
+ fileIncludePattern: /\.(tsx|jsx|ts|js)$/,
76
+ },
77
+ "react19-form-action-non-function": {
78
+ regex: /<form\s+[^>]*\baction\s*=\s*["'][^"']/,
79
+ description: "React 19 form action prop should be a function (Server Action), not a string URL. Use <form action={serverAction}> for progressive enhancement.",
80
+ fileIncludePattern: /\.(tsx|jsx)$/,
81
+ },
82
+ "react19-useoptimistic-no-transition": {
83
+ regex: /\buseOptimistic\s*\([\s\S]{0,300}?(?!useTransition|startTransition)/,
84
+ description: "React 19 useOptimistic should be paired with useTransition/startTransition for non-urgent updates. Without it, optimistic updates can be interrupted.",
85
+ fileIncludePattern: /\.(tsx|jsx)$/,
86
+ },
87
+ // --- oxlint-inspired React rules (April 2026) ---
88
+ "hook-usestate-destructure": {
89
+ regex: /(?:^|\n)\s*useState\s*(?:<[^>]+>)?\s*\([^)]*\)\s*;/,
90
+ description: "useState() called without destructuring [value, setter] — value is inaccessible. Use: const [value, setValue] = useState(initial). (oxlint react/hook-use-state)",
91
+ fileIncludePattern: /\.(tsx|jsx|ts)$/,
92
+ },
93
+ "prefer-function-component": {
94
+ regex: /class\s+\w+\s+extends\s+(?:React\.)?(?:Component|PureComponent)\b/,
95
+ description: "Class component could be a function component — class components lack hook support, are harder to tree-shake, and React Compiler cannot optimize them. (oxlint react/prefer-function-component)",
96
+ fileIncludePattern: /\.(tsx|jsx)$/,
97
+ },
98
+ // --- React Compiler bailout patterns (GA v1.0, Oct 2025 — Next.js 16 stable) ---
99
+ "compiler-side-effect-in-render": {
100
+ regex: /\b(?:console\.(?:log|warn|error|info)\s*\(|Math\.random\s*\(|Date\.now\s*\(|document\.(?:getElementById|querySelector|createElement)\s*\()/,
101
+ description: "Side effect in render body — React Compiler silently skips memoization. Move to useEffect or event handler.",
102
+ fileIncludePattern: /\.(tsx|jsx)$/,
103
+ },
104
+ "compiler-ref-read-in-render": {
105
+ regex: /(?:^|\n)\s*(?:const|let|var)\s+\w+\s*=\s*\w+Ref\.current\b/,
106
+ description: "Reading ref.current during render — React Compiler cannot track ref mutations. Read refs in useEffect or event handlers only.",
107
+ fileIncludePattern: /\.(tsx|jsx)$/,
108
+ },
109
+ "compiler-prop-mutation": {
110
+ regex: /\bprops\.\w+\.(?:push|pop|shift|unshift|splice|sort|reverse|fill)\s*\(/,
111
+ description: "Mutating props object — breaks React Compiler immutability assumption. Clone before mutating: [...props.items, newItem].",
112
+ fileIncludePattern: /\.(tsx|jsx)$/,
113
+ },
114
+ "compiler-state-mutation": {
115
+ regex: /(?:^|\n)\s*\w+\.(?:push|pop|shift|unshift|splice|sort|reverse|fill)\s*\([\s\S]{0,200}?set[A-Z]\w*\s*\(\s*\w+\s*\)/,
116
+ description: "Direct state mutation then setState with same reference — React Compiler assumes immutable updates. Use spread: setItems([...items, newItem]).",
117
+ fileIncludePattern: /\.(tsx|jsx)$/,
118
+ },
119
+ "compiler-try-catch-bailout": {
120
+ regex: /(?:function\s+[A-Z]\w*|const\s+[A-Z]\w*\s*=)[\s\S]{0,300}?\btry\s*\{[\s\S]{0,500}?\bcatch\s*\(/,
121
+ description: "try/catch in component body — React Compiler may silently bail out (known issue #35644). Move error handling to useEffect or extract to a hook.",
122
+ fileIncludePattern: /\.(tsx|jsx)$/,
123
+ },
124
+ "compiler-redundant-memo": {
125
+ regex: /\b(?:React\.)?memo\s*\(\s*(?:function\s+[A-Z]|(?:\([^)]*\)|[A-Z]\w*)\s*=>)/,
126
+ description: "React.memo wrapping — React Compiler auto-memoizes, making manual memo redundant. Safe to remove after compiler adoption.",
127
+ fileIncludePattern: /\.(tsx|jsx)$/,
128
+ },
129
+ "compiler-redundant-usecallback": {
130
+ regex: /\buseCallback\s*\(\s*(?:\([^)]*\)|[a-z_$][\w$]*)\s*=>/,
131
+ description: "useCallback wrapping — React Compiler auto-memoizes callbacks, making manual useCallback redundant. Safe to remove after compiler adoption.",
132
+ fileIncludePattern: /\.(tsx|jsx)$/,
133
+ },
134
+ // --- RSC boundary serializability (Tier 4 — Item 18) ---
135
+ "rsc-non-serializable-prop": {
136
+ // Detects patterns like onClick={fn} or callback={handler} on JSX elements
137
+ // when the file has "use client" directive or imports from a client component.
138
+ // Heuristic: prop=function-reference (not arrow inline, that's caught elsewhere).
139
+ regex: /\b(?:onClick|onChange|onSubmit|onError|callback|handler|render)\s*=\s*\{\s*[a-z_$][\w$]*\s*\}/,
140
+ description: "Function passed as prop across RSC boundary — must be a Server Action ('use server') or component must be Client Component ('use client'). Functions are not serializable.",
141
+ fileIncludePattern: /\.(tsx|jsx)$/,
142
+ },
143
+ "rsc-date-prop": {
144
+ regex: /\b\w+\s*=\s*\{\s*new\s+Date\s*\(/,
145
+ description: "Date object passed as prop — Date is serializable in JSON but loses prototype across RSC boundary. Use ISO string + parse on client side.",
146
+ fileIncludePattern: /\.(tsx|jsx)$/,
147
+ },
148
+ // --- useEffect pain points (37% of devs struggle — State of React 2025) ---
149
+ "useEffect-missing-cleanup": {
150
+ regex: /useEffect\s*\(\s*(?:\([^)]*\)|[a-z_$][\w$]*)\s*=>[\s\S]{0,800}?(?:addEventListener|setInterval|setTimeout|subscribe|on\s*\()(?:(?!return\s*(?:\(\s*\)\s*=>|function))[\s\S]){0,800}\}\s*,/,
151
+ description: "useEffect with addEventListener/setInterval/subscribe but no cleanup return — memory leak. Return a cleanup function that removes the listener/clears the interval.",
152
+ fileIncludePattern: /\.(tsx|jsx)$/,
153
+ },
154
+ "useEffect-setstate-loop": {
155
+ regex: /useEffect\s*\(\s*(?:\([^)]*\)|[a-z_$][\w$]*)\s*=>[\s\S]{0,500}?set([A-Z]\w*)\s*\([\s\S]{0,300}?\[\s*[\s\S]*?\b\1\b/i,
156
+ description: "setState inside useEffect with same state variable in dependency array — causes infinite render loop. Either remove from deps or use functional updater: setState(prev => ...).",
157
+ fileIncludePattern: /\.(tsx|jsx)$/,
158
+ },
159
+ "useEffect-missing-deps-identifier": {
160
+ // Heuristic: useEffect with empty dep array [] but body references an
161
+ // identifier that looks like a prop/state (lowerCamelCase, 2+ chars).
162
+ // Subset of eslint-react-hooks/exhaustive-deps — catches the common
163
+ // "empty deps but reads mutable value" bug without full scope analysis.
164
+ regex: /useEffect\s*\(\s*\([^)]*\)\s*=>\s*\{[\s\S]{0,400}?\b(?:props\.\w+|[a-z][a-zA-Z]*(?:\.\w+)?)[\s\S]{0,400}?\}\s*,\s*\[\s*\]\s*\)/,
165
+ description: "useEffect with empty deps array [] reads props/state identifiers in body — likely missing dependencies. If intentional, add // eslint-disable-next-line react-hooks/exhaustive-deps with reason.",
166
+ fileIncludePattern: /\.(tsx|jsx)$/,
167
+ },
168
+ // --- Next.js 16 cache patterns ---
169
+ "nextjs-use-cache-without-tag": {
170
+ regex: /['"]use cache['"](?:(?!cacheTag\s*\()[\s\S]){0,1000}$/,
171
+ description: "Next.js 16 'use cache' directive without cacheTag() call — cache entry is hard to invalidate. Add cacheTag('name') for targeted revalidation.",
172
+ fileIncludePattern: /\.(tsx|jsx|ts)$/,
173
+ },
174
+ "nextjs-revalidatetag-deprecated": {
175
+ regex: /\brevalidateTag\s*\(\s*['"][^'"]+['"]\s*\)/,
176
+ description: "Next.js 16: revalidateTag() without cacheLife profile (second argument). Single-arg form deprecated — add cacheLife profile.",
177
+ fileIncludePattern: /\.(tsx|jsx|ts)$/,
178
+ },
179
+ // --- TanStack Query patterns ---
180
+ "tanstack-missing-invalidation": {
181
+ regex: /\buseMutation\s*\((?:(?!invalidateQueries|invalidateQuery)[\s\S]){0,800}\}\s*\)/,
182
+ description: "useMutation without invalidateQueries in onSuccess/onSettled — stale data remains in cache after mutation. Add queryClient.invalidateQueries() on success.",
183
+ fileIncludePattern: /\.(tsx|jsx|ts)$/,
184
+ },
9
185
  "empty-catch": {
10
186
  regex: /catch\s*\([^)]*\)\s*\{\s*\}/,
11
187
  description: "Empty catch block — swallowed error (CQ8)",
@@ -38,6 +214,427 @@ const BUILTIN_PATTERNS = {
38
214
  regex: /\/\/\s*(TODO|FIXME|HACK|XXX|TEMP|TEMPORARY)\b|\/\/\s*(Phase|Step|Stage)\s*\d|\/\/\s*(placeholder|stub|dummy)\b|throw new Error\(['"]not implemented['"]\)|console\.(log|warn)\(['"]TODO\b/i,
39
215
  description: "Scaffolding markers: TODO/FIXME/HACK, Phase/Step markers, placeholder stubs, not-implemented throws (tech debt)",
40
216
  },
217
+ // Kotlin anti-patterns
218
+ "runblocking-in-coroutine": {
219
+ regex: /suspend\s+fun[\s\S]{0,500}runBlocking\s*[\({]/,
220
+ description: "runBlocking inside suspend function — deadlock risk (Kotlin coroutines)",
221
+ },
222
+ "globalscope-launch": {
223
+ regex: /GlobalScope\.(launch|async)\s*[\({]/,
224
+ description: "GlobalScope.launch/async — lifecycle leak, use structured concurrency (Kotlin)",
225
+ },
226
+ "data-class-mutable": {
227
+ regex: /data\s+class\s+\w+\([^)]*\bvar\s+/,
228
+ description: "data class with var property — breaks hashCode/equals contract (Kotlin)",
229
+ },
230
+ "lateinit-no-check": {
231
+ regex: /lateinit\s+var\s+(\w+)/,
232
+ description: "lateinit var without isInitialized check — UninitializedPropertyAccessException risk (Kotlin)",
233
+ },
234
+ "empty-when-branch": {
235
+ regex: /when\s*\([^)]*\)\s*\{[\s\S]*?->\s*\{\s*\}/,
236
+ description: "Empty when branch — swallowed case (Kotlin)",
237
+ },
238
+ "mutable-shared-state": {
239
+ regex: /(?:companion\s+object|object\s+\w+)\s*\{[\s\S]*?\bvar\s+/,
240
+ description: "Mutable var inside object/companion — thread-unsafe shared state (Kotlin)",
241
+ },
242
+ // Kotest anti-patterns — require include_tests=true to surface
243
+ "kotest-missing-assertion": {
244
+ regex: /\btest\s*\(\s*"[^"]*"\s*\)\s*\{(?:(?!\bshould(?:Be|NotBe|Throw|Contain|Match|HaveSize)\b|\bshould\s*\{|\bshouldBe\b|\bassertSoftly\b|\bassertThat\b|\bassertTrue\b|\bassertFalse\b|\bassertEquals\b|\bexpect\s*\(|\bverify\s*\()[\s\S])*?\}/,
245
+ description: "Kotest test block without any shouldBe/shouldThrow/assertSoftly/assertEquals — missing assertion",
246
+ },
247
+ "kotest-mixed-styles": {
248
+ regex: /(?:\bFunSpec\s*\([\s\S]*?(?:\bDescribeSpec|\bStringSpec|\bBehaviorSpec|\bShouldSpec|\bWordSpec|\bFeatureSpec|\bExpectSpec)\s*\()|(?:(?:\bDescribeSpec|\bStringSpec|\bBehaviorSpec|\bShouldSpec|\bWordSpec|\bFeatureSpec|\bExpectSpec)\s*\([\s\S]*?\bFunSpec\s*\()/,
249
+ description: "Multiple Kotest spec styles (e.g. FunSpec + DescribeSpec) in same file — inconsistent test layout",
250
+ },
251
+ // Jetpack Compose anti-patterns
252
+ "compose-missing-remember": {
253
+ regex: /(?<!\bremember\s*\{[^}]{0,60})\b(?:mutableStateOf|mutableStateListOf|mutableIntStateOf|derivedStateOf)\s*(?:<[^>]*>)?\s*\(/,
254
+ description: "mutableStateOf/derivedStateOf without remember — state resets every recomposition (Compose)",
255
+ },
256
+ "compose-unstable-lambda": {
257
+ regex: /@Composable[\s\S]{0,2000}?\bon[A-Z]\w*\s*:\s*\([^)]*\)\s*->\s*Unit/,
258
+ description: "Event callback param with function type — unstable, causes child recomposition every frame unless caller uses remember (Compose)",
259
+ },
260
+ "compose-side-effect-in-composition": {
261
+ regex: /@Composable[\s\S]{0,1000}?(?:\bcoroutineScope\s*\{|\bviewModelScope\.launch|\bGlobalScope\.launch)/,
262
+ description: "Coroutine launch in @Composable body — use LaunchedEffect/rememberCoroutineScope instead (Compose)",
263
+ },
264
+ // PHP anti-patterns
265
+ "sql-injection-php": {
266
+ regex: /\$_(?:GET|POST|REQUEST)\[[^\]]+\][\s\S]{0,200}?(?:->query\(|->execute\(|createCommand\()/,
267
+ description: "User input from $_GET/$_POST flowing into SQL query without sanitization (PHP)",
268
+ },
269
+ "xss-php": {
270
+ regex: /echo\s+\$_(?:GET|POST|REQUEST)\[|print\s+\$_(?:GET|POST|REQUEST)\[/,
271
+ description: "Unescaped user input echoed to output — XSS risk (PHP). Use htmlspecialchars()",
272
+ },
273
+ "eval-php": {
274
+ regex: /\beval\s*\(/,
275
+ description: "eval() usage — code injection risk (PHP)",
276
+ },
277
+ "exec-php": {
278
+ regex: /\b(?:exec|system|passthru|shell_exec|popen|proc_open)\s*\(/,
279
+ description: "Shell command execution — command injection risk (PHP)",
280
+ },
281
+ "unserialize-php": {
282
+ regex: /\bunserialize\s*\(\s*\$_(?:GET|POST|REQUEST|COOKIE)/,
283
+ description: "unserialize() on user input — deserialization attack risk (PHP)",
284
+ },
285
+ "file-include-var": {
286
+ regex: /(?:require|include)(?:_once)?\s*\(?\s*\$(?!this)/,
287
+ description: "require/include with variable — file inclusion risk (PHP)",
288
+ },
289
+ "unescaped-yii-view": {
290
+ regex: /<\?=\s*\$(?!this->(?:render|beginBlock|endBlock))(?!.*?Html::encode)/,
291
+ description: "Yii2 view outputs variable without Html::encode() — XSS risk",
292
+ },
293
+ "raw-query-yii": {
294
+ regex: /createCommand\s*\(\s*["'][^"']*\$\{?\w+/,
295
+ description: "Yii2 createCommand with string interpolation — SQL injection risk",
296
+ },
297
+ // NestJS anti-patterns
298
+ "nest-circular-inject": {
299
+ regex: /@Inject\s*\(\s*forwardRef\s*\(/,
300
+ description: "Circular dependency via forwardRef — restructure module boundaries (NestJS)",
301
+ },
302
+ "nest-catch-all-filter": {
303
+ regex: /@Catch\s*\(\s*\)/,
304
+ description: "@Catch() with no argument — catches all exceptions indiscriminately (NestJS)",
305
+ },
306
+ "nest-request-scope": {
307
+ regex: /scope:\s*Scope\.REQUEST/,
308
+ description: "Request-scoped provider — performance overhead, breaks singleton assumptions (NestJS)",
309
+ },
310
+ "nest-raw-exception": {
311
+ regex: /throw\s+new\s+Error\s*\(/,
312
+ description: "Raw Error thrown instead of NestJS HttpException/BadRequestException (NestJS)",
313
+ },
314
+ "nest-any-guard-return": {
315
+ regex: /canActivate[\s\S]{0,100}return\s+true\s*;/,
316
+ description: "Guard always returns true — security no-op (NestJS)",
317
+ },
318
+ "nest-service-locator": {
319
+ regex: /moduleRef\s*\.\s*(?:get|resolve)\s*\(/,
320
+ description: "Service locator via ModuleRef.get/resolve — use constructor injection instead (NestJS)",
321
+ },
322
+ "nest-direct-env": {
323
+ regex: /process\.env\.\w+/,
324
+ description: "Direct process.env access — use ConfigService for type-safe config (NestJS)",
325
+ },
326
+ // Wave 2 anti-patterns
327
+ "nest-graphql-no-auth": {
328
+ // R-7 fix: restrict to .resolver.ts files (via fileIncludePattern) to avoid
329
+ // false positives on REST @Query() params. Regex checks for @Resolver + @Query/@Mutation
330
+ // present AND no @UseGuards anywhere in the matched span (capped at 2000 chars to avoid
331
+ // catastrophic backtracking — O(n) since the negation only runs once per symbol source).
332
+ regex: /^(?![\s\S]*@UseGuards)[\s\S]*@Resolver\s*\([\s\S]{0,500}?@(?:Query|Mutation)\s*\(/,
333
+ description: "GraphQL resolver with @Query/@Mutation but no @UseGuards in file — likely unprotected (NestJS)",
334
+ fileIncludePattern: /\.resolver\.[jt]sx?$/,
335
+ },
336
+ "nest-eager-relation": {
337
+ regex: /@(?:OneToMany|ManyToOne|OneToOne|ManyToMany)\s*\(\s*\(\)\s*=>\s*\w+[\s\S]{0,200}\beager:\s*true/,
338
+ description: "TypeORM relation with { eager: true } — auto-loads joins on every query (NestJS)",
339
+ },
340
+ // Wave 3: nestjs-doctor rule parity batch (15 rules)
341
+ // --- Security (5 rules) ---
342
+ "nest-typeorm-synchronize-prod": {
343
+ regex: /synchronize:\s*true(?![\s\S]{0,100}NODE_ENV\s*!==\s*['"`]production)/,
344
+ description: "TypeORM synchronize: true — schema auto-sync in production drops/recreates tables (NestJS)",
345
+ fileIncludePattern: /\.(ts|js)$/,
346
+ },
347
+ "nest-exposed-stack-trace": {
348
+ regex: /\.stack\s*(?:,|\)|\}|\n)/,
349
+ description: "Error.stack exposed in response/log — leaks internal paths and line numbers (NestJS security)",
350
+ fileIncludePattern: /\.(controller|filter|interceptor)\.[jt]sx?$/,
351
+ },
352
+ "nest-raw-entity-response": {
353
+ regex: /return\s+(?:await\s+)?this\.\w+Repository\.find/,
354
+ description: "Raw entity returned from controller — bypasses @Exclude/@Transform, leaks internal fields (NestJS)",
355
+ fileIncludePattern: /\.controller\.[jt]sx?$/,
356
+ },
357
+ "nest-cors-wildcard": {
358
+ regex: /(?:cors:\s*(?:true|\{\s*origin:\s*['"`]\*['"`])|enableCors\s*\(\s*\{\s*origin:\s*['"`]\*['"`])/,
359
+ description: "CORS wildcard origin — allows any site to make credentialed requests (NestJS security)",
360
+ },
361
+ "nest-disabled-csrf": {
362
+ regex: /csrf:\s*false|csrfProtection.*disabled/i,
363
+ description: "CSRF protection disabled — forms vulnerable to cross-site request forgery (NestJS)",
364
+ },
365
+ // --- Correctness (5 rules) ---
366
+ "nest-missing-guard-method": {
367
+ regex: /implements\s+(?:Can(?:Activate|Load)|NestGuard)(?:\s*\{(?![\s\S]{0,500}(?:canActivate|canLoad)\s*\())/,
368
+ description: "Guard class implements CanActivate/CanLoad but missing the required method (NestJS)",
369
+ fileIncludePattern: /\.guard\.[jt]sx?$/,
370
+ },
371
+ "nest-missing-pipe-transform": {
372
+ regex: /implements\s+PipeTransform(?:\s*\{(?![\s\S]{0,500}transform\s*\())/,
373
+ description: "Pipe class implements PipeTransform but missing transform() method (NestJS)",
374
+ fileIncludePattern: /\.pipe\.[jt]sx?$/,
375
+ },
376
+ "nest-missing-filter-catch": {
377
+ regex: /implements\s+ExceptionFilter(?:\s*\{(?![\s\S]{0,500}catch\s*\())/,
378
+ description: "Exception filter class implements ExceptionFilter but missing catch() method (NestJS)",
379
+ fileIncludePattern: /\.filter\.[jt]sx?$/,
380
+ },
381
+ "nest-missing-interceptor-intercept": {
382
+ regex: /implements\s+NestInterceptor(?:\s*\{(?![\s\S]{0,500}intercept\s*\())/,
383
+ description: "Interceptor class implements NestInterceptor but missing intercept() method (NestJS)",
384
+ fileIncludePattern: /\.interceptor\.[jt]sx?$/,
385
+ },
386
+ "nest-param-decorator-no-type": {
387
+ regex: /@Param\s*\(\s*['"`]\w+['"`]\s*\)\s*\w+\s*[,)]/,
388
+ description: "@Param('id') parameter without type annotation — `id` inferred as `any` (NestJS)",
389
+ fileIncludePattern: /\.controller\.[jt]sx?$/,
390
+ },
391
+ // --- Architecture (3 rules) ---
392
+ "nest-orm-in-controller": {
393
+ regex: /(?:@InjectRepository|this\.\w+Repository\.(?:find|save|update|delete|remove))/,
394
+ description: "Direct ORM/Repository usage in controller — violates separation of concerns (NestJS)",
395
+ fileIncludePattern: /\.controller\.[jt]sx?$/,
396
+ },
397
+ "nest-business-logic-in-controller": {
398
+ regex: /\bif\s*\(\s*\w+\s*\.\s*\w+\s*(?:===|!==|>|<|>=|<=)[\s\S]{0,200}(?:throw\s+new|await\s+this\.)/,
399
+ description: "Complex branching + async call in controller — business logic belongs in a service (NestJS)",
400
+ fileIncludePattern: /\.controller\.[jt]sx?$/,
401
+ },
402
+ "nest-moduleref-get": {
403
+ regex: /\bmoduleRef\s*\.\s*(?:get|resolve)\s*\(\s*['"`]?\w+/,
404
+ description: "Service locator via ModuleRef.get/resolve — use constructor injection instead (NestJS)",
405
+ },
406
+ // --- Performance (2 rules) ---
407
+ "nest-sync-fs-in-handler": {
408
+ regex: /\b(?:readFileSync|writeFileSync|existsSync|statSync|mkdirSync)\s*\(/,
409
+ description: "Synchronous filesystem call blocks the event loop — use fs/promises (NestJS)",
410
+ fileIncludePattern: /\.(controller|service)\.[jt]sx?$/,
411
+ },
412
+ "nest-require-primary-key": {
413
+ regex: /@Entity\s*\([\s\S]{0,200}(?:export\s+)?class\s+\w+(?:\s+extends\s+\w+)?\s*\{(?![\s\S]{0,500}@Primary(?:Generated)?Column)/,
414
+ description: "@Entity without @PrimaryColumn/@PrimaryGeneratedColumn — TypeORM will fail at runtime (NestJS)",
415
+ fileIncludePattern: /\.entity\.[jt]sx?$/,
416
+ },
417
+ // Astro anti-patterns
418
+ "astro-client-on-astro": {
419
+ regex: /client:(load|idle|visible|media|only).*\.astro/,
420
+ description: "client directive on .astro component import (Astro components cannot hydrate)",
421
+ },
422
+ "astro-glob-usage": {
423
+ regex: /Astro\.glob\s*\(/,
424
+ description: "Astro.glob() REMOVED in Astro 6 — use getCollection() or import.meta.glob() (BREAKING)",
425
+ },
426
+ "astro-set-html-xss": {
427
+ regex: /set:html=\{[^"'][^}]*\}/,
428
+ description: "set:html with dynamic content — potential XSS risk",
429
+ },
430
+ "astro-img-element": {
431
+ regex: /<img\s/,
432
+ description: "raw <img> element — use <Image> from astro:assets for optimization",
433
+ },
434
+ "astro-missing-getStaticPaths": {
435
+ regex: /\[[\w.]+\]\.astro/,
436
+ description: "dynamic route file — verify getStaticPaths is exported",
437
+ },
438
+ "astro-legacy-content-collections": {
439
+ regex: /src\/content\/config\.ts/,
440
+ description: "Legacy content collections REMOVED in Astro 6 — migrate to src/content.config.ts + Content Layer API (BREAKING)",
441
+ },
442
+ "astro-no-image-dimensions": {
443
+ regex: /<Image\s+(?![^>]*(?:width|height)\s*=)[^>]*\/?>/,
444
+ description: "<Image> without width/height — causes CLS (Cumulative Layout Shift)",
445
+ },
446
+ "astro-inline-script-no-is-inline": {
447
+ regex: /<script(?!\s+is:inline)(?:\s[^>]*)?>[\s\S]*?<\/script>/,
448
+ description: "<script> without is:inline — Astro will process/bundle it; add is:inline for raw passthrough",
449
+ },
450
+ "astro-env-secret-in-client": {
451
+ regex: /import\.meta\.env\.SECRET_/,
452
+ description: "import.meta.env.SECRET_* accessed — secret env vars are server-only, undefined in client components",
453
+ },
454
+ "astro-hardcoded-site-url": {
455
+ regex: /(?:href|src|url)\s*=\s*["']https?:\/\/(?!\/\/)[^"']*["']/,
456
+ description: "hardcoded absolute URL — use Astro.site or relative paths for portability",
457
+ },
458
+ "astro-missing-lang-attr": {
459
+ regex: /<html(?!\s[^>]*\blang\s*=)[^>]*>/,
460
+ description: "<html> without lang attribute — required for accessibility (WCAG 3.1.1)",
461
+ },
462
+ "astro-form-without-action": {
463
+ regex: /<form(?!\s[^>]*\baction\s*=)[^>]*>/,
464
+ description: "<form> without action attribute — consider Astro Actions for type-safe form handling",
465
+ },
466
+ "astro-view-transitions-deprecated": {
467
+ regex: /<ViewTransitions\s*\/?>/,
468
+ description: "<ViewTransitions /> renamed to <ClientRouter /> in Astro 6 (BREAKING) — update import from astro:transitions",
469
+ },
470
+ // Next.js anti-patterns
471
+ "nextjs-wrong-router": {
472
+ regex: /from\s+['"]next\/router['"]|require\s*\(\s*['"]next\/router['"]\s*\)/,
473
+ description: "Using next/router (Pages Router) in App Router file — use next/navigation instead",
474
+ fileExcludePattern: /(^|\/)pages\//,
475
+ },
476
+ "nextjs-fetch-waterfall": {
477
+ regex: /await\s+fetch\s*\([^)]*\)[\s\S]{0,300}await\s+fetch\s*\(/,
478
+ description: "Sequential await fetch calls — use Promise.all to avoid waterfall (Next.js performance)",
479
+ },
480
+ "nextjs-unnecessary-use-client": {
481
+ regex: /['"]use client['"](?![\s\S]*(?:useState|useEffect|useRef|useCallback|useMemo|useContext|useReducer|onClick|onChange|onSubmit|window\.|document\.|localStorage\.))/,
482
+ description: "File has 'use client' but may not need it — no hooks, events, or browser globals detected",
483
+ },
484
+ "nextjs-pages-in-app": {
485
+ regex: /./,
486
+ description: "Pages Router convention (index.tsx) inside app/ directory — use page.tsx for App Router",
487
+ fileIncludePattern: /(^|\/)app\/.*\/index\.(tsx|jsx|ts|js)$|^app\/index\.(tsx|jsx|ts|js)$/,
488
+ },
489
+ "nextjs-missing-error-boundary": {
490
+ regex: /./,
491
+ description: "Page file without sibling error.tsx — no error boundary for graceful error handling",
492
+ fileIncludePattern: /(^|\/)app\/.*\/page\.[jt]sx?$/,
493
+ },
494
+ "nextjs-use-client-in-layout": {
495
+ regex: /^[\s\S]{0,512}['"]use client['"]/,
496
+ description: "Layout file with 'use client' — layouts should be Server Components for optimal performance",
497
+ fileIncludePattern: /(^|\/)app\/.*\/layout\.[jt]sx?$|^app\/layout\.[jt]sx?$/,
498
+ },
499
+ "nextjs-missing-metadata": {
500
+ regex: /./,
501
+ description: "Page file without metadata or generateMetadata export — missing SEO metadata",
502
+ fileIncludePattern: /(^|\/)app\/.*\/page\.[jt]sx?$/,
503
+ },
504
+ "nextjs-missing-use-client": {
505
+ // Match files containing client-only API references that do NOT begin with
506
+ // a "use client" / 'use client' / `use client` directive in the first 512 bytes.
507
+ regex: /^(?![\s\S]{0,512}["'`]use client["'`])[\s\S]*(?:useState|useEffect|useRef|useCallback|useMemo|useContext|onClick=|onChange=|onSubmit=)/,
508
+ description: "Client-only API used without 'use client' directive — component will error at build (Next.js App Router)",
509
+ fileIncludePattern: /(^|\/)app\/.*\.(tsx|jsx)$/,
510
+ },
511
+ // --- Hono anti-patterns (Task 15) ---
512
+ "hono-missing-error-handler": {
513
+ regex: /new\s+(?:Hono|OpenAPIHono)\s*(?:<[^>]*>)?\s*\(\s*\)(?!(?:[\s\S](?!new\s+(?:Hono|OpenAPIHono)))*\.onError)/,
514
+ description: "Hono app created without .onError() handler — unhandled exceptions return 500 with no logging",
515
+ },
516
+ "hono-throw-raw-error": {
517
+ regex: /\(\s*c\s*:\s*Context\s*(?:,\s*next\s*:\s*Next\s*)?\)[\s\S]*?\bthrow\s+new\s+Error\s*\(/,
518
+ description: "throw new Error() inside Hono handler — use HTTPException for proper status code handling",
519
+ },
520
+ "hono-missing-validator": {
521
+ regex: /await\s+c\.req\.(?:json|parseBody)\s*\(\s*\)(?![\s\S]{0,400}?zValidator)/,
522
+ description: "c.req.json()/parseBody() without preceding zValidator — unvalidated request body",
523
+ },
524
+ "hono-unguarded-json-parse": {
525
+ regex: /(?<!try\s*\{[\s\S]{0,200})await\s+c\.req\.json\s*\(\s*\)/,
526
+ description: "await c.req.json() without try/catch — malformed JSON crashes handler",
527
+ },
528
+ "hono-env-type-any": {
529
+ regex: /new\s+Hono\s*\(\s*\)(?!\s*<)/,
530
+ description: "new Hono() without <Env> generic — loses type safety on c.env and c.var",
531
+ },
532
+ "hono-missing-status-code": {
533
+ regex: /\bc\.json\s*\(\s*\{[^}]+\}\s*\)/,
534
+ description: "c.json() without explicit status code — defaults to 200 even for errors/creations",
535
+ },
536
+ "hono-full-app-rpc-export": {
537
+ regex: /export\s+type\s+\w+\s*=\s*typeof\s+app\b/,
538
+ description: "export type X = typeof app — slow RPC pattern (Issue #3869, 8-min CI builds). Use typeof routeGroup instead",
539
+ },
540
+ // --- Database / ORM anti-patterns (db-audit feedback) ---
541
+ "unsafe-raw-sql": {
542
+ regex: /(?:\$queryRawUnsafe|\$executeRawUnsafe|knex\.raw|sequelize\.query|db\.raw)\s*\(\s*[`"'][^`"']*\$\{/,
543
+ description: "Raw SQL with template-string interpolation — SQL injection risk. Use parameterized $queryRaw`...` or query builder. Covers Prisma/Knex/Sequelize/Drizzle.",
544
+ },
545
+ "transaction-external-io": {
546
+ regex: /\$transaction\s*\(\s*async\s*\([^)]*\)\s*=>\s*\{[\s\S]{0,2000}?\b(?:fetch|axios|http|stripe|sendgrid|twilio|sendEmail|publishEvent|enqueue)\s*[.(]/,
547
+ description: "External I/O (fetch/HTTP/email/queue) inside Prisma $transaction callback — long-running transactions hold locks. Move I/O after commit.",
548
+ },
549
+ "migration-create-index-no-concurrently": {
550
+ regex: /CREATE\s+(?:UNIQUE\s+)?INDEX(?!\s+CONCURRENTLY)/i,
551
+ description: "CREATE INDEX without CONCURRENTLY — locks the table during build. Use CREATE INDEX CONCURRENTLY in PostgreSQL migrations.",
552
+ fileIncludePattern: /\/migrations?\/.*\.sql$/,
553
+ },
554
+ "migration-drop-column": {
555
+ regex: /\bALTER\s+TABLE[\s\S]{0,200}\bDROP\s+COLUMN\b/i,
556
+ description: "DROP COLUMN in migration — destructive, breaks rolling deploys. Use multi-step deprecation: stop writes → backfill → drop in next release.",
557
+ fileIncludePattern: /\/migrations?\/.*\.sql$/,
558
+ },
559
+ "migration-alter-column-type": {
560
+ regex: /\bALTER\s+TABLE[\s\S]{0,200}\bALTER\s+COLUMN[\s\S]{0,200}\bTYPE\b/i,
561
+ description: "ALTER COLUMN TYPE in migration — full table rewrite, locks table. Use ADD COLUMN + backfill + DROP COLUMN in separate releases.",
562
+ fileIncludePattern: /\/migrations?\/.*\.sql$/,
563
+ },
564
+ "migration-not-null-no-default": {
565
+ regex: /\bADD\s+COLUMN\b[\s\S]{0,200}\bNOT\s+NULL\b(?![\s\S]{0,100}\bDEFAULT\b)/i,
566
+ description: "ADD COLUMN NOT NULL without DEFAULT — fails on existing rows. Add as nullable first, backfill, then add NOT NULL constraint.",
567
+ fileIncludePattern: /\/migrations?\/.*\.sql$/,
568
+ },
569
+ // --- Python anti-patterns ---
570
+ "mutable-default": {
571
+ regex: /def\s+\w+\s*\([^)]*=\s*(?:\[\s*\]|\{\s*\}|set\s*\(\s*\))/,
572
+ description: "Mutable default argument ([], {}, set()) — shared between calls (Python)",
573
+ },
574
+ "bare-except": {
575
+ regex: /except\s*:/,
576
+ description: "Bare except: catches everything including KeyboardInterrupt (Python)",
577
+ },
578
+ "broad-except": {
579
+ regex: /except\s+(?:Exception|BaseException)\s*:/,
580
+ description: "Broad exception catch — hides real errors (Python)",
581
+ },
582
+ "global-keyword": {
583
+ regex: /\bglobal\s+\w+/,
584
+ description: "global keyword — mutable global state makes code hard to test (Python)",
585
+ },
586
+ "star-import": {
587
+ regex: /from\s+\S+\s+import\s+\*/,
588
+ description: "Star import — pollutes namespace, breaks static analysis (Python)",
589
+ },
590
+ "print-debug-py": {
591
+ regex: /^\s*print\s*\(/m,
592
+ description: "print() in production code — use logging module (Python)",
593
+ },
594
+ "eval-exec": {
595
+ regex: /\b(?:eval|exec)\s*\(/,
596
+ description: "eval()/exec() — code injection risk (Python)",
597
+ },
598
+ "shell-true": {
599
+ regex: /subprocess\.\w+\s*\([^)]*shell\s*=\s*True/,
600
+ description: "subprocess with shell=True — command injection risk (Python)",
601
+ },
602
+ "pickle-load": {
603
+ regex: /pickle\.(?:load|loads)\s*\(/,
604
+ description: "pickle.load/loads — arbitrary code execution from untrusted data (Python)",
605
+ },
606
+ "yaml-unsafe": {
607
+ regex: /yaml\.load\s*\([^)]*\)(?![\s\S]{0,30}Loader)/,
608
+ description: "yaml.load without SafeLoader — arbitrary code execution risk (Python)",
609
+ },
610
+ "open-no-with": {
611
+ regex: /(?<!with\s{1,20})\bopen\s*\([^)]+\)\s*(?:\.\w+|;|$)/m,
612
+ description: "open() without with statement — resource leak if exception occurs (Python)",
613
+ },
614
+ "string-concat-loop": {
615
+ regex: /for\s+\w+\s+in\s+[\s\S]{0,200}?\+=\s*(?:['"]|f['"]|str\()/,
616
+ description: "String concatenation in loop — O(n^2), use join() or list append (Python)",
617
+ },
618
+ "datetime-naive": {
619
+ regex: /datetime\.(?:now|utcnow)\s*\(\s*\)/,
620
+ description: "datetime.now()/utcnow() without timezone — naive datetime causes bugs (Python)",
621
+ },
622
+ "shadow-builtin": {
623
+ regex: /^(?:list|dict|set|id|type|input|map|filter|range|str|int|float|bool|tuple|bytes|object|print|open|format|len|sum|min|max|any|all|zip|enumerate|sorted|reversed|next|iter|super|hash|dir|vars|globals|locals)\s*=/m,
624
+ description: "Assignment shadows Python builtin — breaks code that uses the builtin later (Python)",
625
+ },
626
+ "n-plus-one-django": {
627
+ regex: /for\s+\w+\s+in\s+[\s\S]{0,300}?\.\w+_set\b|\.\w+\.all\(\)/,
628
+ description: "Potential N+1 query — accessing related objects in loop without select_related/prefetch_related (Django)",
629
+ },
630
+ "late-binding": {
631
+ regex: /for\s+(\w+)\s+in\s+[\s\S]{0,200}?lambda\s*[^:]*:\s*\1\b/,
632
+ description: "Late binding closure in loop — all lambdas share last loop value (Python)",
633
+ },
634
+ "assert-tuple": {
635
+ regex: /\bassert\s*\(/,
636
+ description: "assert(expr) — always True because tuple is truthy. Use assert expr without parens (Python)",
637
+ },
41
638
  };
42
639
  /**
43
640
  * Search for structural code patterns across indexed symbols.
@@ -54,10 +651,14 @@ export async function searchPatterns(repo, pattern, options) {
54
651
  // Resolve pattern: built-in name or custom regex
55
652
  let regex;
56
653
  let patternName;
654
+ let fileExcludePattern;
655
+ let fileIncludePattern;
57
656
  const builtin = BUILTIN_PATTERNS[pattern];
58
657
  if (builtin) {
59
658
  regex = builtin.regex;
60
659
  patternName = `${pattern}: ${builtin.description}`;
660
+ fileExcludePattern = builtin.fileExcludePattern;
661
+ fileIncludePattern = builtin.fileIncludePattern;
61
662
  }
62
663
  else {
63
664
  try {
@@ -80,6 +681,10 @@ export async function searchPatterns(repo, pattern, options) {
80
681
  continue;
81
682
  if (filePattern && !sym.file.includes(filePattern))
82
683
  continue;
684
+ if (fileExcludePattern && fileExcludePattern.test(sym.file))
685
+ continue;
686
+ if (fileIncludePattern && !fileIncludePattern.test(sym.file))
687
+ continue;
83
688
  scanned++;
84
689
  const match = regex.exec(sym.source);
85
690
  if (match) {
@@ -98,6 +703,47 @@ export async function searchPatterns(repo, pattern, options) {
98
703
  });
99
704
  }
100
705
  }
706
+ // File-level scanning fallback for patterns targeting non-source files
707
+ // (e.g. SQL migrations, .env files). Triggered when fileIncludePattern is set.
708
+ // This iterates index.files and reads them directly — needed because such
709
+ // files have no parsed symbols.
710
+ if (fileIncludePattern) {
711
+ for (const fileEntry of index.files) {
712
+ if (matches.length >= maxResults)
713
+ break;
714
+ if (!fileIncludePattern.test(fileEntry.path))
715
+ continue;
716
+ if (filePattern && !fileEntry.path.includes(filePattern))
717
+ continue;
718
+ if (fileExcludePattern && fileExcludePattern.test(fileEntry.path))
719
+ continue;
720
+ // Skip files that already had symbol matches above (avoid duplicates)
721
+ if (matches.some((m) => m.file === fileEntry.path))
722
+ continue;
723
+ let content;
724
+ try {
725
+ content = await readFile(join(index.root, fileEntry.path), "utf-8");
726
+ }
727
+ catch {
728
+ continue;
729
+ }
730
+ scanned++;
731
+ const match = regex.exec(content);
732
+ if (match) {
733
+ const linesBefore = content.slice(0, match.index).split("\n").length;
734
+ const matchedText = match[0].split("\n")[0];
735
+ matches.push({
736
+ name: fileEntry.path.split("/").pop() ?? fileEntry.path,
737
+ kind: "function", // file-level match has no symbol kind
738
+ file: fileEntry.path,
739
+ start_line: linesBefore,
740
+ end_line: linesBefore,
741
+ matched_pattern: patternName,
742
+ context: matchedText.trim().slice(0, 200),
743
+ });
744
+ }
745
+ }
746
+ }
101
747
  return {
102
748
  matches,
103
749
  pattern: patternName,
@@ -108,9 +754,11 @@ export async function searchPatterns(repo, pattern, options) {
108
754
  * List all available built-in patterns.
109
755
  */
110
756
  export function listPatterns() {
111
- return Object.entries(BUILTIN_PATTERNS).map(([name, { description }]) => ({
757
+ return Object.entries(BUILTIN_PATTERNS).map(([name, p]) => ({
112
758
  name,
113
- description,
759
+ description: p.description,
760
+ ...(p.fileExcludePattern ? { fileExcludePattern: p.fileExcludePattern.source } : {}),
761
+ ...(p.fileIncludePattern ? { fileIncludePattern: p.fileIncludePattern.source } : {}),
114
762
  }));
115
763
  }
116
764
  //# sourceMappingURL=pattern-tools.js.map