@opensip-tools/fitness 1.0.4

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 (461) hide show
  1. package/.turbo/turbo-build.log +4 -0
  2. package/.turbo/turbo-typecheck.log +4 -0
  3. package/LICENSE +21 -0
  4. package/dist/__tests__/gate.test.d.ts +13 -0
  5. package/dist/__tests__/gate.test.d.ts.map +1 -0
  6. package/dist/__tests__/gate.test.js +422 -0
  7. package/dist/__tests__/gate.test.js.map +1 -0
  8. package/dist/__tests__/sarif.test.d.ts +2 -0
  9. package/dist/__tests__/sarif.test.d.ts.map +1 -0
  10. package/dist/__tests__/sarif.test.js +169 -0
  11. package/dist/__tests__/sarif.test.js.map +1 -0
  12. package/dist/cli/dashboard.d.ts +6 -0
  13. package/dist/cli/dashboard.d.ts.map +1 -0
  14. package/dist/cli/dashboard.js +77 -0
  15. package/dist/cli/dashboard.js.map +1 -0
  16. package/dist/cli/fit.d.ts +37 -0
  17. package/dist/cli/fit.d.ts.map +1 -0
  18. package/dist/cli/fit.js +539 -0
  19. package/dist/cli/fit.js.map +1 -0
  20. package/dist/cli/list-checks.d.ts +6 -0
  21. package/dist/cli/list-checks.d.ts.map +1 -0
  22. package/dist/cli/list-checks.js +23 -0
  23. package/dist/cli/list-checks.js.map +1 -0
  24. package/dist/cli/list-recipes.d.ts +6 -0
  25. package/dist/cli/list-recipes.d.ts.map +1 -0
  26. package/dist/cli/list-recipes.js +31 -0
  27. package/dist/cli/list-recipes.js.map +1 -0
  28. package/dist/framework/__tests__/ast-utilities.test.d.ts +2 -0
  29. package/dist/framework/__tests__/ast-utilities.test.d.ts.map +1 -0
  30. package/dist/framework/__tests__/ast-utilities.test.js +153 -0
  31. package/dist/framework/__tests__/ast-utilities.test.js.map +1 -0
  32. package/dist/framework/__tests__/check-config.test.d.ts +2 -0
  33. package/dist/framework/__tests__/check-config.test.d.ts.map +1 -0
  34. package/dist/framework/__tests__/check-config.test.js +56 -0
  35. package/dist/framework/__tests__/check-config.test.js.map +1 -0
  36. package/dist/framework/__tests__/command-executor.test.d.ts +2 -0
  37. package/dist/framework/__tests__/command-executor.test.d.ts.map +1 -0
  38. package/dist/framework/__tests__/command-executor.test.js +71 -0
  39. package/dist/framework/__tests__/command-executor.test.js.map +1 -0
  40. package/dist/framework/__tests__/content-filter-dispatch.test.d.ts +2 -0
  41. package/dist/framework/__tests__/content-filter-dispatch.test.d.ts.map +1 -0
  42. package/dist/framework/__tests__/content-filter-dispatch.test.js +104 -0
  43. package/dist/framework/__tests__/content-filter-dispatch.test.js.map +1 -0
  44. package/dist/framework/__tests__/content-filter.test.d.ts +2 -0
  45. package/dist/framework/__tests__/content-filter.test.d.ts.map +1 -0
  46. package/dist/framework/__tests__/content-filter.test.js +126 -0
  47. package/dist/framework/__tests__/content-filter.test.js.map +1 -0
  48. package/dist/framework/__tests__/define-check.test.d.ts +2 -0
  49. package/dist/framework/__tests__/define-check.test.d.ts.map +1 -0
  50. package/dist/framework/__tests__/define-check.test.js +155 -0
  51. package/dist/framework/__tests__/define-check.test.js.map +1 -0
  52. package/dist/framework/__tests__/directive-inventory.test.d.ts +2 -0
  53. package/dist/framework/__tests__/directive-inventory.test.d.ts.map +1 -0
  54. package/dist/framework/__tests__/directive-inventory.test.js +44 -0
  55. package/dist/framework/__tests__/directive-inventory.test.js.map +1 -0
  56. package/dist/framework/__tests__/execution-context.test.d.ts +2 -0
  57. package/dist/framework/__tests__/execution-context.test.d.ts.map +1 -0
  58. package/dist/framework/__tests__/execution-context.test.js +62 -0
  59. package/dist/framework/__tests__/execution-context.test.js.map +1 -0
  60. package/dist/framework/__tests__/file-accessor.test.d.ts +2 -0
  61. package/dist/framework/__tests__/file-accessor.test.d.ts.map +1 -0
  62. package/dist/framework/__tests__/file-accessor.test.js +106 -0
  63. package/dist/framework/__tests__/file-accessor.test.js.map +1 -0
  64. package/dist/framework/__tests__/file-cache.test.d.ts +2 -0
  65. package/dist/framework/__tests__/file-cache.test.d.ts.map +1 -0
  66. package/dist/framework/__tests__/file-cache.test.js +122 -0
  67. package/dist/framework/__tests__/file-cache.test.js.map +1 -0
  68. package/dist/framework/__tests__/import-graph.test.d.ts +15 -0
  69. package/dist/framework/__tests__/import-graph.test.d.ts.map +1 -0
  70. package/dist/framework/__tests__/import-graph.test.js +164 -0
  71. package/dist/framework/__tests__/import-graph.test.js.map +1 -0
  72. package/dist/framework/__tests__/path-matcher.test.d.ts +2 -0
  73. package/dist/framework/__tests__/path-matcher.test.d.ts.map +1 -0
  74. package/dist/framework/__tests__/path-matcher.test.js +113 -0
  75. package/dist/framework/__tests__/path-matcher.test.js.map +1 -0
  76. package/dist/framework/__tests__/register-helpers.test.d.ts +2 -0
  77. package/dist/framework/__tests__/register-helpers.test.d.ts.map +1 -0
  78. package/dist/framework/__tests__/register-helpers.test.js +42 -0
  79. package/dist/framework/__tests__/register-helpers.test.js.map +1 -0
  80. package/dist/framework/__tests__/registry.test.d.ts +2 -0
  81. package/dist/framework/__tests__/registry.test.d.ts.map +1 -0
  82. package/dist/framework/__tests__/registry.test.js +208 -0
  83. package/dist/framework/__tests__/registry.test.js.map +1 -0
  84. package/dist/framework/__tests__/result-builder.test.d.ts +2 -0
  85. package/dist/framework/__tests__/result-builder.test.d.ts.map +1 -0
  86. package/dist/framework/__tests__/result-builder.test.js +153 -0
  87. package/dist/framework/__tests__/result-builder.test.js.map +1 -0
  88. package/dist/framework/__tests__/scope-resolver.test.d.ts +2 -0
  89. package/dist/framework/__tests__/scope-resolver.test.d.ts.map +1 -0
  90. package/dist/framework/__tests__/scope-resolver.test.js +140 -0
  91. package/dist/framework/__tests__/scope-resolver.test.js.map +1 -0
  92. package/dist/framework/__tests__/severity-mapping.test.d.ts +2 -0
  93. package/dist/framework/__tests__/severity-mapping.test.d.ts.map +1 -0
  94. package/dist/framework/__tests__/severity-mapping.test.js +42 -0
  95. package/dist/framework/__tests__/severity-mapping.test.js.map +1 -0
  96. package/dist/framework/__tests__/strip-literals.test.d.ts +2 -0
  97. package/dist/framework/__tests__/strip-literals.test.d.ts.map +1 -0
  98. package/dist/framework/__tests__/strip-literals.test.js +87 -0
  99. package/dist/framework/__tests__/strip-literals.test.js.map +1 -0
  100. package/dist/framework/abortable-exec.d.ts +34 -0
  101. package/dist/framework/abortable-exec.d.ts.map +1 -0
  102. package/dist/framework/abortable-exec.js +136 -0
  103. package/dist/framework/abortable-exec.js.map +1 -0
  104. package/dist/framework/ast-utilities.d.ts +41 -0
  105. package/dist/framework/ast-utilities.d.ts.map +1 -0
  106. package/dist/framework/ast-utilities.js +106 -0
  107. package/dist/framework/ast-utilities.js.map +1 -0
  108. package/dist/framework/check-config.d.ts +171 -0
  109. package/dist/framework/check-config.d.ts.map +1 -0
  110. package/dist/framework/check-config.js +114 -0
  111. package/dist/framework/check-config.js.map +1 -0
  112. package/dist/framework/check-types.d.ts +57 -0
  113. package/dist/framework/check-types.d.ts.map +1 -0
  114. package/dist/framework/check-types.js +35 -0
  115. package/dist/framework/check-types.js.map +1 -0
  116. package/dist/framework/command-executor.d.ts +25 -0
  117. package/dist/framework/command-executor.d.ts.map +1 -0
  118. package/dist/framework/command-executor.js +63 -0
  119. package/dist/framework/command-executor.js.map +1 -0
  120. package/dist/framework/constants.d.ts +9 -0
  121. package/dist/framework/constants.d.ts.map +1 -0
  122. package/dist/framework/constants.js +16 -0
  123. package/dist/framework/constants.js.map +1 -0
  124. package/dist/framework/content-filter.d.ts +33 -0
  125. package/dist/framework/content-filter.d.ts.map +1 -0
  126. package/dist/framework/content-filter.js +236 -0
  127. package/dist/framework/content-filter.js.map +1 -0
  128. package/dist/framework/define-check.d.ts +38 -0
  129. package/dist/framework/define-check.d.ts.map +1 -0
  130. package/dist/framework/define-check.js +252 -0
  131. package/dist/framework/define-check.js.map +1 -0
  132. package/dist/framework/directive-inventory.d.ts +34 -0
  133. package/dist/framework/directive-inventory.d.ts.map +1 -0
  134. package/dist/framework/directive-inventory.js +77 -0
  135. package/dist/framework/directive-inventory.js.map +1 -0
  136. package/dist/framework/directive-parsing.d.ts +20 -0
  137. package/dist/framework/directive-parsing.d.ts.map +1 -0
  138. package/dist/framework/directive-parsing.js +121 -0
  139. package/dist/framework/directive-parsing.js.map +1 -0
  140. package/dist/framework/execution-context.d.ts +95 -0
  141. package/dist/framework/execution-context.d.ts.map +1 -0
  142. package/dist/framework/execution-context.js +122 -0
  143. package/dist/framework/execution-context.js.map +1 -0
  144. package/dist/framework/file-accessor.d.ts +20 -0
  145. package/dist/framework/file-accessor.d.ts.map +1 -0
  146. package/dist/framework/file-accessor.js +122 -0
  147. package/dist/framework/file-accessor.js.map +1 -0
  148. package/dist/framework/file-cache.d.ts +70 -0
  149. package/dist/framework/file-cache.d.ts.map +1 -0
  150. package/dist/framework/file-cache.js +178 -0
  151. package/dist/framework/file-cache.js.map +1 -0
  152. package/dist/framework/file-type-filter.d.ts +11 -0
  153. package/dist/framework/file-type-filter.d.ts.map +1 -0
  154. package/dist/framework/file-type-filter.js +21 -0
  155. package/dist/framework/file-type-filter.js.map +1 -0
  156. package/dist/framework/ignore-processing.d.ts +22 -0
  157. package/dist/framework/ignore-processing.d.ts.map +1 -0
  158. package/dist/framework/ignore-processing.js +241 -0
  159. package/dist/framework/ignore-processing.js.map +1 -0
  160. package/dist/framework/import-graph.d.ts +51 -0
  161. package/dist/framework/import-graph.d.ts.map +1 -0
  162. package/dist/framework/import-graph.js +216 -0
  163. package/dist/framework/import-graph.js.map +1 -0
  164. package/dist/framework/memory-profiler.d.ts +53 -0
  165. package/dist/framework/memory-profiler.d.ts.map +1 -0
  166. package/dist/framework/memory-profiler.js +92 -0
  167. package/dist/framework/memory-profiler.js.map +1 -0
  168. package/dist/framework/parse-cache.d.ts +23 -0
  169. package/dist/framework/parse-cache.d.ts.map +1 -0
  170. package/dist/framework/parse-cache.js +37 -0
  171. package/dist/framework/parse-cache.js.map +1 -0
  172. package/dist/framework/path-matcher.d.ts +86 -0
  173. package/dist/framework/path-matcher.d.ts.map +1 -0
  174. package/dist/framework/path-matcher.js +138 -0
  175. package/dist/framework/path-matcher.js.map +1 -0
  176. package/dist/framework/register-helpers.d.ts +10 -0
  177. package/dist/framework/register-helpers.d.ts.map +1 -0
  178. package/dist/framework/register-helpers.js +17 -0
  179. package/dist/framework/register-helpers.js.map +1 -0
  180. package/dist/framework/registry.d.ts +41 -0
  181. package/dist/framework/registry.d.ts.map +1 -0
  182. package/dist/framework/registry.js +103 -0
  183. package/dist/framework/registry.js.map +1 -0
  184. package/dist/framework/result-builder.d.ts +74 -0
  185. package/dist/framework/result-builder.d.ts.map +1 -0
  186. package/dist/framework/result-builder.js +154 -0
  187. package/dist/framework/result-builder.js.map +1 -0
  188. package/dist/framework/scope-resolver.d.ts +23 -0
  189. package/dist/framework/scope-resolver.d.ts.map +1 -0
  190. package/dist/framework/scope-resolver.js +201 -0
  191. package/dist/framework/scope-resolver.js.map +1 -0
  192. package/dist/framework/severity-mapping.d.ts +13 -0
  193. package/dist/framework/severity-mapping.d.ts.map +1 -0
  194. package/dist/framework/severity-mapping.js +51 -0
  195. package/dist/framework/severity-mapping.js.map +1 -0
  196. package/dist/framework/strip-literals.d.ts +48 -0
  197. package/dist/framework/strip-literals.d.ts.map +1 -0
  198. package/dist/framework/strip-literals.js +188 -0
  199. package/dist/framework/strip-literals.js.map +1 -0
  200. package/dist/gate.d.ts +74 -0
  201. package/dist/gate.d.ts.map +1 -0
  202. package/dist/gate.js +257 -0
  203. package/dist/gate.js.map +1 -0
  204. package/dist/index.d.ts +47 -0
  205. package/dist/index.d.ts.map +1 -0
  206. package/dist/index.js +51 -0
  207. package/dist/index.js.map +1 -0
  208. package/dist/plugins/__tests__/check-package-discovery.test.d.ts +2 -0
  209. package/dist/plugins/__tests__/check-package-discovery.test.d.ts.map +1 -0
  210. package/dist/plugins/__tests__/check-package-discovery.test.js +170 -0
  211. package/dist/plugins/__tests__/check-package-discovery.test.js.map +1 -0
  212. package/dist/plugins/__tests__/lang-domain.test.d.ts +2 -0
  213. package/dist/plugins/__tests__/lang-domain.test.d.ts.map +1 -0
  214. package/dist/plugins/__tests__/lang-domain.test.js +171 -0
  215. package/dist/plugins/__tests__/lang-domain.test.js.map +1 -0
  216. package/dist/plugins/__tests__/loader.test.d.ts +2 -0
  217. package/dist/plugins/__tests__/loader.test.d.ts.map +1 -0
  218. package/dist/plugins/__tests__/loader.test.js +194 -0
  219. package/dist/plugins/__tests__/loader.test.js.map +1 -0
  220. package/dist/plugins/check-package-discovery.d.ts +73 -0
  221. package/dist/plugins/check-package-discovery.d.ts.map +1 -0
  222. package/dist/plugins/check-package-discovery.js +212 -0
  223. package/dist/plugins/check-package-discovery.js.map +1 -0
  224. package/dist/plugins/loader.d.ts +31 -0
  225. package/dist/plugins/loader.d.ts.map +1 -0
  226. package/dist/plugins/loader.js +290 -0
  227. package/dist/plugins/loader.js.map +1 -0
  228. package/dist/plugins/types.d.ts +23 -0
  229. package/dist/plugins/types.d.ts.map +1 -0
  230. package/dist/plugins/types.js +9 -0
  231. package/dist/plugins/types.js.map +1 -0
  232. package/dist/recipes/__tests__/built-in-recipes.test.d.ts +2 -0
  233. package/dist/recipes/__tests__/built-in-recipes.test.d.ts.map +1 -0
  234. package/dist/recipes/__tests__/built-in-recipes.test.js +93 -0
  235. package/dist/recipes/__tests__/built-in-recipes.test.js.map +1 -0
  236. package/dist/recipes/__tests__/check-config.test.d.ts +5 -0
  237. package/dist/recipes/__tests__/check-config.test.d.ts.map +1 -0
  238. package/dist/recipes/__tests__/check-config.test.js +37 -0
  239. package/dist/recipes/__tests__/check-config.test.js.map +1 -0
  240. package/dist/recipes/__tests__/check-resolution.test.d.ts +2 -0
  241. package/dist/recipes/__tests__/check-resolution.test.d.ts.map +1 -0
  242. package/dist/recipes/__tests__/check-resolution.test.js +135 -0
  243. package/dist/recipes/__tests__/check-resolution.test.js.map +1 -0
  244. package/dist/recipes/__tests__/registry.test.d.ts +2 -0
  245. package/dist/recipes/__tests__/registry.test.d.ts.map +1 -0
  246. package/dist/recipes/__tests__/registry.test.js +97 -0
  247. package/dist/recipes/__tests__/registry.test.js.map +1 -0
  248. package/dist/recipes/__tests__/retry.test.d.ts +2 -0
  249. package/dist/recipes/__tests__/retry.test.d.ts.map +1 -0
  250. package/dist/recipes/__tests__/retry.test.js +75 -0
  251. package/dist/recipes/__tests__/retry.test.js.map +1 -0
  252. package/dist/recipes/__tests__/service.test.d.ts +11 -0
  253. package/dist/recipes/__tests__/service.test.d.ts.map +1 -0
  254. package/dist/recipes/__tests__/service.test.js +482 -0
  255. package/dist/recipes/__tests__/service.test.js.map +1 -0
  256. package/dist/recipes/built-in-recipes.d.ts +14 -0
  257. package/dist/recipes/built-in-recipes.d.ts.map +1 -0
  258. package/dist/recipes/built-in-recipes.js +247 -0
  259. package/dist/recipes/built-in-recipes.js.map +1 -0
  260. package/dist/recipes/check-config.d.ts +40 -0
  261. package/dist/recipes/check-config.d.ts.map +1 -0
  262. package/dist/recipes/check-config.js +61 -0
  263. package/dist/recipes/check-config.js.map +1 -0
  264. package/dist/recipes/check-resolution.d.ts +21 -0
  265. package/dist/recipes/check-resolution.d.ts.map +1 -0
  266. package/dist/recipes/check-resolution.js +121 -0
  267. package/dist/recipes/check-resolution.js.map +1 -0
  268. package/dist/recipes/check-result-processor.d.ts +51 -0
  269. package/dist/recipes/check-result-processor.d.ts.map +1 -0
  270. package/dist/recipes/check-result-processor.js +158 -0
  271. package/dist/recipes/check-result-processor.js.map +1 -0
  272. package/dist/recipes/parallel-execution.d.ts +33 -0
  273. package/dist/recipes/parallel-execution.d.ts.map +1 -0
  274. package/dist/recipes/parallel-execution.js +142 -0
  275. package/dist/recipes/parallel-execution.js.map +1 -0
  276. package/dist/recipes/registry.d.ts +81 -0
  277. package/dist/recipes/registry.d.ts.map +1 -0
  278. package/dist/recipes/registry.js +131 -0
  279. package/dist/recipes/registry.js.map +1 -0
  280. package/dist/recipes/retry.d.ts +25 -0
  281. package/dist/recipes/retry.d.ts.map +1 -0
  282. package/dist/recipes/retry.js +44 -0
  283. package/dist/recipes/retry.js.map +1 -0
  284. package/dist/recipes/sequential-execution.d.ts +10 -0
  285. package/dist/recipes/sequential-execution.d.ts.map +1 -0
  286. package/dist/recipes/sequential-execution.js +122 -0
  287. package/dist/recipes/sequential-execution.js.map +1 -0
  288. package/dist/recipes/service-types.d.ts +84 -0
  289. package/dist/recipes/service-types.d.ts.map +1 -0
  290. package/dist/recipes/service-types.js +8 -0
  291. package/dist/recipes/service-types.js.map +1 -0
  292. package/dist/recipes/service.d.ts +71 -0
  293. package/dist/recipes/service.d.ts.map +1 -0
  294. package/dist/recipes/service.js +331 -0
  295. package/dist/recipes/service.js.map +1 -0
  296. package/dist/recipes/types.d.ts +154 -0
  297. package/dist/recipes/types.d.ts.map +1 -0
  298. package/dist/recipes/types.js +54 -0
  299. package/dist/recipes/types.js.map +1 -0
  300. package/dist/sarif.d.ts +34 -0
  301. package/dist/sarif.d.ts.map +1 -0
  302. package/dist/sarif.js +192 -0
  303. package/dist/sarif.js.map +1 -0
  304. package/dist/signalers/__tests__/loader.test.d.ts +2 -0
  305. package/dist/signalers/__tests__/loader.test.d.ts.map +1 -0
  306. package/dist/signalers/__tests__/loader.test.js +74 -0
  307. package/dist/signalers/__tests__/loader.test.js.map +1 -0
  308. package/dist/signalers/index.d.ts +8 -0
  309. package/dist/signalers/index.d.ts.map +1 -0
  310. package/dist/signalers/index.js +9 -0
  311. package/dist/signalers/index.js.map +1 -0
  312. package/dist/signalers/loader.d.ts +24 -0
  313. package/dist/signalers/loader.d.ts.map +1 -0
  314. package/dist/signalers/loader.js +108 -0
  315. package/dist/signalers/loader.js.map +1 -0
  316. package/dist/signalers/schema.d.ts +288 -0
  317. package/dist/signalers/schema.d.ts.map +1 -0
  318. package/dist/signalers/schema.js +99 -0
  319. package/dist/signalers/schema.js.map +1 -0
  320. package/dist/signalers/types.d.ts +13 -0
  321. package/dist/signalers/types.d.ts.map +1 -0
  322. package/dist/signalers/types.js +5 -0
  323. package/dist/signalers/types.js.map +1 -0
  324. package/dist/targets/__tests__/loader.test.d.ts +2 -0
  325. package/dist/targets/__tests__/loader.test.d.ts.map +1 -0
  326. package/dist/targets/__tests__/loader.test.js +127 -0
  327. package/dist/targets/__tests__/loader.test.js.map +1 -0
  328. package/dist/targets/__tests__/resolver.test.d.ts +2 -0
  329. package/dist/targets/__tests__/resolver.test.d.ts.map +1 -0
  330. package/dist/targets/__tests__/resolver.test.js +54 -0
  331. package/dist/targets/__tests__/resolver.test.js.map +1 -0
  332. package/dist/targets/__tests__/target-registry.test.d.ts +2 -0
  333. package/dist/targets/__tests__/target-registry.test.d.ts.map +1 -0
  334. package/dist/targets/__tests__/target-registry.test.js +89 -0
  335. package/dist/targets/__tests__/target-registry.test.js.map +1 -0
  336. package/dist/targets/index.d.ts +10 -0
  337. package/dist/targets/index.d.ts.map +1 -0
  338. package/dist/targets/index.js +12 -0
  339. package/dist/targets/index.js.map +1 -0
  340. package/dist/targets/loader.d.ts +19 -0
  341. package/dist/targets/loader.d.ts.map +1 -0
  342. package/dist/targets/loader.js +159 -0
  343. package/dist/targets/loader.js.map +1 -0
  344. package/dist/targets/resolver.d.ts +19 -0
  345. package/dist/targets/resolver.d.ts.map +1 -0
  346. package/dist/targets/resolver.js +37 -0
  347. package/dist/targets/resolver.js.map +1 -0
  348. package/dist/targets/target-registry.d.ts +61 -0
  349. package/dist/targets/target-registry.d.ts.map +1 -0
  350. package/dist/targets/target-registry.js +93 -0
  351. package/dist/targets/target-registry.js.map +1 -0
  352. package/dist/targets/types.d.ts +85 -0
  353. package/dist/targets/types.d.ts.map +1 -0
  354. package/dist/targets/types.js +5 -0
  355. package/dist/targets/types.js.map +1 -0
  356. package/dist/tool.d.ts +17 -0
  357. package/dist/tool.d.ts.map +1 -0
  358. package/dist/tool.js +282 -0
  359. package/dist/tool.js.map +1 -0
  360. package/dist/types/findings.d.ts +117 -0
  361. package/dist/types/findings.d.ts.map +1 -0
  362. package/dist/types/findings.js +93 -0
  363. package/dist/types/findings.js.map +1 -0
  364. package/dist/types/severity.d.ts +15 -0
  365. package/dist/types/severity.d.ts.map +1 -0
  366. package/dist/types/severity.js +36 -0
  367. package/dist/types/severity.js.map +1 -0
  368. package/package.json +45 -0
  369. package/src/__tests__/gate.test.ts +537 -0
  370. package/src/__tests__/sarif.test.ts +201 -0
  371. package/src/cli/dashboard.ts +93 -0
  372. package/src/cli/fit.ts +612 -0
  373. package/src/cli/list-checks.ts +32 -0
  374. package/src/cli/list-recipes.ts +38 -0
  375. package/src/framework/__tests__/ast-utilities.test.ts +157 -0
  376. package/src/framework/__tests__/check-config.test.ts +65 -0
  377. package/src/framework/__tests__/command-executor.test.ts +79 -0
  378. package/src/framework/__tests__/content-filter-dispatch.test.ts +132 -0
  379. package/src/framework/__tests__/content-filter.test.ts +136 -0
  380. package/src/framework/__tests__/define-check.test.ts +180 -0
  381. package/src/framework/__tests__/directive-inventory.test.ts +53 -0
  382. package/src/framework/__tests__/execution-context.test.ts +80 -0
  383. package/src/framework/__tests__/file-accessor.test.ts +121 -0
  384. package/src/framework/__tests__/file-cache.test.ts +142 -0
  385. package/src/framework/__tests__/import-graph.test.ts +282 -0
  386. package/src/framework/__tests__/path-matcher.test.ts +130 -0
  387. package/src/framework/__tests__/register-helpers.test.ts +51 -0
  388. package/src/framework/__tests__/registry.test.ts +243 -0
  389. package/src/framework/__tests__/result-builder.test.ts +178 -0
  390. package/src/framework/__tests__/scope-resolver.test.ts +208 -0
  391. package/src/framework/__tests__/severity-mapping.test.ts +50 -0
  392. package/src/framework/__tests__/strip-literals.test.ts +109 -0
  393. package/src/framework/abortable-exec.ts +177 -0
  394. package/src/framework/ast-utilities.ts +112 -0
  395. package/src/framework/check-config.ts +339 -0
  396. package/src/framework/check-types.ts +77 -0
  397. package/src/framework/command-executor.ts +100 -0
  398. package/src/framework/constants.ts +16 -0
  399. package/src/framework/content-filter.ts +288 -0
  400. package/src/framework/define-check.ts +336 -0
  401. package/src/framework/directive-inventory.ts +110 -0
  402. package/src/framework/directive-parsing.ts +152 -0
  403. package/src/framework/execution-context.ts +247 -0
  404. package/src/framework/file-accessor.ts +171 -0
  405. package/src/framework/file-cache.ts +220 -0
  406. package/src/framework/file-type-filter.ts +25 -0
  407. package/src/framework/ignore-processing.ts +350 -0
  408. package/src/framework/import-graph.ts +280 -0
  409. package/src/framework/memory-profiler.ts +145 -0
  410. package/src/framework/parse-cache.ts +38 -0
  411. package/src/framework/path-matcher.ts +191 -0
  412. package/src/framework/register-helpers.ts +20 -0
  413. package/src/framework/registry.ts +125 -0
  414. package/src/framework/result-builder.ts +225 -0
  415. package/src/framework/scope-resolver.ts +262 -0
  416. package/src/framework/severity-mapping.ts +56 -0
  417. package/src/framework/strip-literals.ts +200 -0
  418. package/src/gate.ts +337 -0
  419. package/src/index.ts +110 -0
  420. package/src/plugins/__tests__/check-package-discovery.test.ts +204 -0
  421. package/src/plugins/__tests__/lang-domain.test.ts +198 -0
  422. package/src/plugins/__tests__/loader.test.ts +226 -0
  423. package/src/plugins/check-package-discovery.ts +242 -0
  424. package/src/plugins/loader.ts +327 -0
  425. package/src/plugins/types.ts +25 -0
  426. package/src/recipes/__tests__/built-in-recipes.test.ts +107 -0
  427. package/src/recipes/__tests__/check-config.test.ts +51 -0
  428. package/src/recipes/__tests__/check-resolution.test.ts +185 -0
  429. package/src/recipes/__tests__/registry.test.ts +115 -0
  430. package/src/recipes/__tests__/retry.test.ts +83 -0
  431. package/src/recipes/__tests__/service.test.ts +572 -0
  432. package/src/recipes/built-in-recipes.ts +273 -0
  433. package/src/recipes/check-config.ts +64 -0
  434. package/src/recipes/check-resolution.ts +169 -0
  435. package/src/recipes/check-result-processor.ts +258 -0
  436. package/src/recipes/parallel-execution.ts +220 -0
  437. package/src/recipes/registry.ts +192 -0
  438. package/src/recipes/retry.ts +69 -0
  439. package/src/recipes/sequential-execution.ts +139 -0
  440. package/src/recipes/service-types.ts +105 -0
  441. package/src/recipes/service.ts +400 -0
  442. package/src/recipes/types.ts +247 -0
  443. package/src/sarif.ts +232 -0
  444. package/src/signalers/__tests__/loader.test.ts +99 -0
  445. package/src/signalers/index.ts +9 -0
  446. package/src/signalers/loader.ts +141 -0
  447. package/src/signalers/schema.ts +117 -0
  448. package/src/signalers/types.ts +15 -0
  449. package/src/targets/__tests__/loader.test.ts +170 -0
  450. package/src/targets/__tests__/resolver.test.ts +74 -0
  451. package/src/targets/__tests__/target-registry.test.ts +103 -0
  452. package/src/targets/index.ts +13 -0
  453. package/src/targets/loader.ts +214 -0
  454. package/src/targets/resolver.ts +44 -0
  455. package/src/targets/target-registry.ts +111 -0
  456. package/src/targets/types.ts +89 -0
  457. package/src/tool.ts +302 -0
  458. package/src/types/findings.ts +239 -0
  459. package/src/types/severity.ts +39 -0
  460. package/tsconfig.json +8 -0
  461. package/vitest.config.ts +33 -0
@@ -0,0 +1,243 @@
1
+ import { NotFoundError } from '@opensip-tools/core';
2
+ import { describe, it, expect } from 'vitest';
3
+
4
+ import { CheckRegistry } from '../../framework/registry.js';
5
+
6
+ import type { Check } from '../../framework/check-types.js';
7
+
8
+ /** Create a minimal Check stub for testing the registry. */
9
+ function makeCheck(overrides: {
10
+ slug: string;
11
+ tags?: string[];
12
+ disabled?: boolean;
13
+ id?: string;
14
+ }): Check {
15
+ return {
16
+ config: {
17
+ id: overrides.id ?? `id-${overrides.slug}`,
18
+ slug: overrides.slug,
19
+ tags: overrides.tags ?? ['quality'],
20
+ description: `Check: ${overrides.slug}`,
21
+ analysisMode: 'analyze',
22
+ scope: { include: [], exclude: [] },
23
+ itemType: 'file' as any,
24
+ disabled: overrides.disabled,
25
+ // eslint-disable-next-line @typescript-eslint/require-await -- mock matches expected `() => Promise<CheckResult>` shape
26
+ execute: async () => ({ findings: [], passed: true }),
27
+ },
28
+ // eslint-disable-next-line @typescript-eslint/require-await -- mock matches expected `() => Promise<CheckResult>` shape
29
+ run: async () => ({ findings: [], passed: true }),
30
+ getScope: () => ({ include: [], exclude: [] }),
31
+ getMatcher: () => ({ matches: () => true }),
32
+ } as unknown as Check;
33
+ }
34
+
35
+ describe('CheckRegistry', () => {
36
+ describe('register and get', () => {
37
+ it('registers and retrieves a check by slug', () => {
38
+ const registry = new CheckRegistry();
39
+ const check = makeCheck({ slug: 'my-check' });
40
+ registry.register(check);
41
+ expect(registry.get('my-check')).toBe(check);
42
+ });
43
+
44
+ it('throws NotFoundError for missing slug', () => {
45
+ const registry = new CheckRegistry();
46
+ expect(() => registry.get('nonexistent')).toThrowError(NotFoundError);
47
+ expect(() => registry.get('nonexistent')).toThrowError('Check not found: nonexistent');
48
+ });
49
+ });
50
+
51
+ describe('has', () => {
52
+ it('returns true for registered check', () => {
53
+ const registry = new CheckRegistry();
54
+ registry.register(makeCheck({ slug: 'exists' }));
55
+ expect(registry.has('exists')).toBe(true);
56
+ });
57
+
58
+ it('returns false for unregistered check', () => {
59
+ const registry = new CheckRegistry();
60
+ expect(registry.has('nope')).toBe(false);
61
+ });
62
+ });
63
+
64
+ describe('duplicate handling', () => {
65
+ it('silently skips duplicate registration', () => {
66
+ const registry = new CheckRegistry();
67
+ const check1 = makeCheck({ slug: 'dup', id: 'first' });
68
+ const check2 = makeCheck({ slug: 'dup', id: 'second' });
69
+ registry.register(check1);
70
+ registry.register(check2);
71
+ // First registration wins
72
+ expect(registry.get('dup').config.id).toBe('first');
73
+ expect(registry.size).toBe(1);
74
+ });
75
+ });
76
+
77
+ describe('list', () => {
78
+ it('returns all registered checks', () => {
79
+ const registry = new CheckRegistry();
80
+ registry.register(makeCheck({ slug: 'a' }));
81
+ registry.register(makeCheck({ slug: 'b' }));
82
+ registry.register(makeCheck({ slug: 'c' }));
83
+ expect(registry.list()).toHaveLength(3);
84
+ });
85
+
86
+ it('returns empty array when no checks registered', () => {
87
+ const registry = new CheckRegistry();
88
+ expect(registry.list()).toEqual([]);
89
+ });
90
+ });
91
+
92
+ describe('listEnabled', () => {
93
+ it('excludes disabled checks', () => {
94
+ const registry = new CheckRegistry();
95
+ registry.register(makeCheck({ slug: 'active' }));
96
+ registry.register(makeCheck({ slug: 'off', disabled: true }));
97
+ registry.register(makeCheck({ slug: 'also-active' }));
98
+
99
+ const enabled = registry.listEnabled();
100
+ expect(enabled).toHaveLength(2);
101
+ expect(enabled.map(c => c.config.slug)).toEqual(['active', 'also-active']);
102
+ });
103
+
104
+ it('returns all checks when none are disabled', () => {
105
+ const registry = new CheckRegistry();
106
+ registry.register(makeCheck({ slug: 'x' }));
107
+ registry.register(makeCheck({ slug: 'y' }));
108
+ expect(registry.listEnabled()).toHaveLength(2);
109
+ });
110
+ });
111
+
112
+ describe('byTag', () => {
113
+ it('filters enabled checks by tag', () => {
114
+ const registry = new CheckRegistry();
115
+ registry.register(makeCheck({ slug: 'sec1', tags: ['security'] }));
116
+ registry.register(makeCheck({ slug: 'qual1', tags: ['quality'] }));
117
+ registry.register(makeCheck({ slug: 'sec2', tags: ['security'] }));
118
+ registry.register(makeCheck({ slug: 'sec-off', tags: ['security'], disabled: true }));
119
+
120
+ const securityChecks = registry.byTag('security');
121
+ expect(securityChecks).toHaveLength(2);
122
+ expect(securityChecks.map(c => c.config.slug)).toEqual(['sec1', 'sec2']);
123
+ });
124
+
125
+ it('returns empty array for unknown tag', () => {
126
+ const registry = new CheckRegistry();
127
+ registry.register(makeCheck({ slug: 'a', tags: ['quality'] }));
128
+ expect(registry.byTag('nonexistent')).toEqual([]);
129
+ });
130
+ });
131
+
132
+ describe('getBySlug', () => {
133
+ it('returns the check when found', () => {
134
+ const registry = new CheckRegistry();
135
+ const check = makeCheck({ slug: 'findme' });
136
+ registry.register(check);
137
+ expect(registry.getBySlug('findme')).toBe(check);
138
+ });
139
+
140
+ it('returns undefined when not found (no throw)', () => {
141
+ const registry = new CheckRegistry();
142
+ expect(registry.getBySlug('missing')).toBeUndefined();
143
+ });
144
+ });
145
+
146
+ describe('listSlugs', () => {
147
+ it('returns all registered slugs', () => {
148
+ const registry = new CheckRegistry();
149
+ registry.register(makeCheck({ slug: 'alpha' }));
150
+ registry.register(makeCheck({ slug: 'beta' }));
151
+ registry.register(makeCheck({ slug: 'gamma' }));
152
+
153
+ const slugs = registry.listSlugs();
154
+ expect(slugs).toEqual(['alpha', 'beta', 'gamma']);
155
+ });
156
+
157
+ it('returns empty array when empty', () => {
158
+ const registry = new CheckRegistry();
159
+ expect(registry.listSlugs()).toEqual([]);
160
+ });
161
+ });
162
+
163
+ describe('size', () => {
164
+ it('reflects current count', () => {
165
+ const registry = new CheckRegistry();
166
+ expect(registry.size).toBe(0);
167
+ registry.register(makeCheck({ slug: 'one' }));
168
+ expect(registry.size).toBe(1);
169
+ registry.register(makeCheck({ slug: 'two' }));
170
+ expect(registry.size).toBe(2);
171
+ });
172
+ });
173
+
174
+ describe('namespace support', () => {
175
+ it('registers with namespace as namespace:slug key', () => {
176
+ const registry = new CheckRegistry();
177
+ const check = makeCheck({ slug: 'my-check' });
178
+ registry.register(check, 'my-plugin');
179
+ expect(registry.get('my-plugin:my-check')).toBe(check);
180
+ });
181
+
182
+ it('resolves bare slug to namespaced check when unambiguous', () => {
183
+ const registry = new CheckRegistry();
184
+ const check = makeCheck({ slug: 'my-check' });
185
+ registry.register(check, 'my-plugin');
186
+ expect(registry.get('my-check')).toBe(check);
187
+ expect(registry.getBySlug('my-check')).toBe(check);
188
+ expect(registry.has('my-check')).toBe(true);
189
+ });
190
+
191
+ it('allows same slug in different namespaces', () => {
192
+ const registry = new CheckRegistry();
193
+ const check1 = makeCheck({ slug: 'check', id: 'first' });
194
+ const check2 = makeCheck({ slug: 'check', id: 'second' });
195
+ registry.register(check1, 'plugin-a');
196
+ registry.register(check2, 'plugin-b');
197
+ expect(registry.size).toBe(2);
198
+ expect(registry.get('plugin-a:check').config.id).toBe('first');
199
+ expect(registry.get('plugin-b:check').config.id).toBe('second');
200
+ });
201
+
202
+ it('returns first registered for ambiguous bare slug', () => {
203
+ const registry = new CheckRegistry();
204
+ registry.register(makeCheck({ slug: 'dup', id: 'first' }), 'ns-a');
205
+ registry.register(makeCheck({ slug: 'dup', id: 'second' }), 'ns-b');
206
+ // Resolves to first registered
207
+ expect(registry.get('dup').config.id).toBe('first');
208
+ });
209
+
210
+ it('silently skips duplicate namespace:slug registration', () => {
211
+ const registry = new CheckRegistry();
212
+ const check1 = makeCheck({ slug: 'x', id: 'first' });
213
+ const check2 = makeCheck({ slug: 'x', id: 'second' });
214
+ registry.register(check1, 'ns');
215
+ registry.register(check2, 'ns');
216
+ expect(registry.size).toBe(1);
217
+ expect(registry.get('ns:x').config.id).toBe('first');
218
+ });
219
+
220
+ it('listSlugs returns namespaced keys', () => {
221
+ const registry = new CheckRegistry();
222
+ registry.register(makeCheck({ slug: 'a' }), 'pkg');
223
+ registry.register(makeCheck({ slug: 'b' }));
224
+ expect(registry.listSlugs()).toEqual(['pkg:a', 'b']);
225
+ });
226
+
227
+ it('listByBareSlug returns all checks with that slug', () => {
228
+ const registry = new CheckRegistry();
229
+ registry.register(makeCheck({ slug: 'check', id: '1' }), 'ns-a');
230
+ registry.register(makeCheck({ slug: 'check', id: '2' }), 'ns-b');
231
+ registry.register(makeCheck({ slug: 'other' }), 'ns-a');
232
+ const results = registry.listByBareSlug('check');
233
+ expect(results).toHaveLength(2);
234
+ expect(results.map(c => c.config.id)).toEqual(['1', '2']);
235
+ });
236
+
237
+ it('returns undefined for unknown namespaced slug', () => {
238
+ const registry = new CheckRegistry();
239
+ registry.register(makeCheck({ slug: 'a' }), 'pkg');
240
+ expect(registry.getBySlug('other:a')).toBeUndefined();
241
+ });
242
+ });
243
+ });
@@ -0,0 +1,178 @@
1
+ import { createSignal } from '@opensip-tools/core';
2
+ import { describe, expect, it } from 'vitest';
3
+
4
+ import { ResultBuilder, extractSnippet, getLineNumber, isAPIFile } from '../result-builder.js';
5
+
6
+ import type { Signal } from '@opensip-tools/core';
7
+
8
+ const sig = (overrides: Partial<Parameters<typeof createSignal>[0]> = {}): Signal =>
9
+ createSignal({
10
+ source: 'fitness',
11
+ provider: 'opensip',
12
+ severity: 'high',
13
+ category: 'warning',
14
+ ruleId: 'fit:test',
15
+ message: 'something',
16
+ code: { file: 'src/x.ts', line: 1 },
17
+ ...overrides,
18
+ });
19
+
20
+ describe('ResultBuilder', () => {
21
+ it('builds an empty result with totalItems', () => {
22
+ const result = ResultBuilder.create({ checkId: 'c1', itemType: 'files' })
23
+ .totalItems(10)
24
+ .build();
25
+ expect(result.signals.length).toBe(0);
26
+ expect(result.metadata.totalItems).toBe(10);
27
+ });
28
+
29
+ it('chains addSignal and addSignals', () => {
30
+ const builder = ResultBuilder.create({ checkId: 'c1', itemType: 'files' });
31
+ builder.addSignal(sig({ severity: 'high' }));
32
+ builder.addSignals([sig({ severity: 'medium' }), sig({ severity: 'medium' })]);
33
+ expect(builder.signalCount).toBe(3);
34
+ });
35
+
36
+ it('addSignals with empty array is a no-op', () => {
37
+ const builder = ResultBuilder.create({ checkId: 'c1', itemType: 'files' });
38
+ builder.addSignals([]);
39
+ expect(builder.signalCount).toBe(0);
40
+ });
41
+
42
+ it('separates errors from warnings via signal severity', () => {
43
+ const builder = ResultBuilder.create({ checkId: 'c1', itemType: 'files' });
44
+ builder.addSignal(sig({ severity: 'high' }));
45
+ builder.addSignal(sig({ severity: 'medium' }));
46
+ expect(builder.errorCount).toBe(1);
47
+ expect(builder.warningCount).toBe(1);
48
+ });
49
+
50
+ it('hasSignals reflects the signal collection', () => {
51
+ const builder = ResultBuilder.create({ checkId: 'c1', itemType: 'files' });
52
+ expect(builder.hasSignals).toBe(false);
53
+ builder.addSignal(sig());
54
+ expect(builder.hasSignals).toBe(true);
55
+ });
56
+
57
+ it('willPass is true when there are no errors', () => {
58
+ const builder = ResultBuilder.create({ checkId: 'c1', itemType: 'files' });
59
+ expect(builder.willPass).toBe(true);
60
+ builder.addSignal(sig({ severity: 'medium' }));
61
+ expect(builder.willPass).toBe(true); // warnings don't fail
62
+ builder.addSignal(sig({ severity: 'high' }));
63
+ expect(builder.willPass).toBe(false);
64
+ });
65
+
66
+ it('ignoredCount and incrementIgnored both update internal state', () => {
67
+ const builder = ResultBuilder.create({ checkId: 'c1', itemType: 'files' });
68
+ builder.ignoredCount(5);
69
+ builder.incrementIgnored();
70
+ builder.incrementIgnored(2);
71
+ builder.totalItems(1);
72
+ const built = builder.build();
73
+ expect(built.ignoredCount).toBe(8);
74
+ });
75
+
76
+ it('duration is included on the built result', () => {
77
+ const result = ResultBuilder.create({ checkId: 'c1', itemType: 'files' })
78
+ .totalItems(1)
79
+ .duration(123)
80
+ .build();
81
+ expect(result.metadata.durationMs).toBe(123);
82
+ });
83
+
84
+ it('filesScanned is included on the built result', () => {
85
+ const result = ResultBuilder.create({ checkId: 'c1', itemType: 'files' })
86
+ .totalItems(5)
87
+ .filesScanned(3)
88
+ .build();
89
+ expect(result.metadata.filesScanned).toBe(3);
90
+ });
91
+
92
+ it('extra payload merges into the built result', () => {
93
+ const result = ResultBuilder.create({ checkId: 'c1', itemType: 'files' })
94
+ .totalItems(1)
95
+ .extra({ foo: 'bar' })
96
+ .extra({ baz: 1 })
97
+ .build();
98
+ expect(result.metadata.extra).toEqual({ foo: 'bar', baz: 1 });
99
+ });
100
+
101
+ it('buildError returns an error result with the given message', () => {
102
+ const builder = ResultBuilder.create({ checkId: 'c1', itemType: 'files' });
103
+ const out = builder.buildError('something exploded', new Error('boom'));
104
+ expect(out.signals.length).toBe(0);
105
+ expect(out.info.label).toMatch(/error|exploded/i);
106
+ });
107
+
108
+ it('clamps compliance to 0 when violations span more files than scanned', () => {
109
+ const builder = ResultBuilder.create({ checkId: 'c1', itemType: 'files' });
110
+ builder.totalItems(2);
111
+ builder.addSignal(sig({ code: { file: 'a' } }));
112
+ builder.addSignal(sig({ code: { file: 'b' } }));
113
+ builder.addSignal(sig({ code: { file: 'c' } }));
114
+ const result = builder.build();
115
+ expect(result.metadata.totalItems).toBe(2);
116
+ // No negative value should appear in info label
117
+ expect(result.info.label).not.toContain('-');
118
+ });
119
+
120
+ it('builds violations-mode info when totalItems is 0', () => {
121
+ const result = ResultBuilder.create({ checkId: 'c1', itemType: 'files' })
122
+ .addSignal(sig({ severity: 'high' }))
123
+ .build();
124
+ expect(result.metadata.totalItems).toBe(0);
125
+ expect(result.signals.length).toBe(1);
126
+ });
127
+ });
128
+
129
+ describe('extractSnippet', () => {
130
+ const content = ['line1', 'line2', 'line3', 'line4', 'line5'].join('\n');
131
+
132
+ it('extracts default 2 context lines around the target', () => {
133
+ const out = extractSnippet(content, 3);
134
+ expect(out.snippet).toContain('1 | line1');
135
+ expect(out.snippet).toContain('5 | line5');
136
+ });
137
+
138
+ it('clamps to start of file', () => {
139
+ const out = extractSnippet(content, 1);
140
+ expect(out.snippet.split('\n')[0]).toBe('1 | line1');
141
+ });
142
+
143
+ it('clamps to end of file', () => {
144
+ const out = extractSnippet(content, 5);
145
+ expect(out.snippet).toContain('5 | line5');
146
+ });
147
+
148
+ it('honors a custom contextLines value', () => {
149
+ const out = extractSnippet(content, 3, 0);
150
+ expect(out.snippet).toBe('3 | line3');
151
+ });
152
+ });
153
+
154
+ describe('getLineNumber', () => {
155
+ it('returns the line number for a character index', () => {
156
+ const text = 'line1\nline2\nline3';
157
+ expect(getLineNumber(text, 0)).toBe(1);
158
+ expect(getLineNumber(text, 6)).toBe(2);
159
+ expect(getLineNumber(text, 12)).toBe(3);
160
+ });
161
+
162
+ it('returns 1 for negative indices', () => {
163
+ expect(getLineNumber('abc', -5)).toBe(1);
164
+ });
165
+ });
166
+
167
+ describe('isAPIFile', () => {
168
+ it.each([
169
+ ['src/api/users.ts', true],
170
+ ['src/routes/foo.ts', true],
171
+ ['src/users-handler.ts', true],
172
+ ['src/users.handler.ts', true],
173
+ ['src/lib/util.ts', false],
174
+ ['src/components/Button.tsx', false],
175
+ ])('isAPIFile(%s) === %s', (path, expected) => {
176
+ expect(isAPIFile(path)).toBe(expected);
177
+ });
178
+ });
@@ -0,0 +1,208 @@
1
+ import { mkdirSync, mkdtempSync, rmSync, writeFileSync } from 'node:fs';
2
+ import { tmpdir } from 'node:os';
3
+ import { dirname, join } from 'node:path';
4
+
5
+ import { afterEach, beforeEach, describe, expect, it } from 'vitest';
6
+
7
+ import { TargetRegistry } from '../../targets/target-registry.js';
8
+ import { buildScopeBasedFileMap } from '../scope-resolver.js';
9
+
10
+ import type { Target, TargetsConfig } from '../../targets/types.js';
11
+
12
+ let testDir: string;
13
+
14
+ function fixture(rel: string, content = ''): string {
15
+ const abs = join(testDir, rel);
16
+ mkdirSync(dirname(abs), { recursive: true });
17
+ writeFileSync(abs, content);
18
+ return abs;
19
+ }
20
+
21
+ function makeRegistry(targets: Target[]): TargetRegistry {
22
+ const reg = new TargetRegistry();
23
+ for (const t of targets) reg.register(t);
24
+ return reg;
25
+ }
26
+
27
+ function makeTarget(name: string, opts: Partial<Target['config']>): Target {
28
+ return {
29
+ config: {
30
+ name,
31
+ description: name,
32
+ include: opts.include ?? [],
33
+ exclude: opts.exclude ?? [],
34
+ ...(opts.languages && { languages: opts.languages }),
35
+ ...(opts.concerns && { concerns: opts.concerns }),
36
+ },
37
+ };
38
+ }
39
+
40
+ function makeConfig(overrides: Partial<TargetsConfig> = {}): TargetsConfig {
41
+ return {
42
+ globalExcludes: overrides.globalExcludes ?? [],
43
+ checkOverrides: overrides.checkOverrides ?? {},
44
+ ...overrides,
45
+ };
46
+ }
47
+
48
+ beforeEach(() => {
49
+
50
+ testDir = mkdtempSync(join(tmpdir(), 'opensip-scope-resolver-'));
51
+ });
52
+
53
+ afterEach(() => {
54
+ rmSync(testDir, { recursive: true, force: true });
55
+ });
56
+
57
+ describe('buildScopeBasedFileMap', () => {
58
+ it('returns an empty map when no checks have scopes or overrides', () => {
59
+ const registry = makeRegistry([
60
+ makeTarget('src', { include: ['src/**/*.ts'], languages: ['typescript'], concerns: ['backend'] }),
61
+ ]);
62
+ const out = buildScopeBasedFileMap([], registry, makeConfig(), testDir);
63
+ expect(out.size).toBe(0);
64
+ });
65
+
66
+ it('matches checks to targets by scope.languages + concerns', () => {
67
+ fixture('src/a.ts');
68
+ fixture('src/b.ts');
69
+ fixture('lib/c.rs');
70
+
71
+ const registry = makeRegistry([
72
+ makeTarget('ts-src', {
73
+ include: ['src/**/*.ts'],
74
+ languages: ['typescript'],
75
+ concerns: ['backend'],
76
+ }),
77
+ makeTarget('rs-lib', {
78
+ include: ['lib/**/*.rs'],
79
+ languages: ['rust'],
80
+ concerns: ['backend'],
81
+ }),
82
+ ]);
83
+
84
+ const out = buildScopeBasedFileMap(
85
+ [{ slug: 'ts-check', scope: { languages: ['typescript'], concerns: ['backend'] } }],
86
+ registry,
87
+ makeConfig(),
88
+ testDir,
89
+ );
90
+
91
+ const tsFiles = out.get('ts-check');
92
+ expect(tsFiles).toBeDefined();
93
+ expect(tsFiles?.length).toBe(2);
94
+ expect(tsFiles?.every((f) => f.endsWith('.ts'))).toBe(true);
95
+ });
96
+
97
+ it('returns empty when scope matches no targets', () => {
98
+ fixture('src/a.ts');
99
+ const registry = makeRegistry([
100
+ makeTarget('ts-src', { include: ['src/**/*.ts'], languages: ['typescript'], concerns: ['backend'] }),
101
+ ]);
102
+
103
+ const out = buildScopeBasedFileMap(
104
+ [{ slug: 'cobol-check', scope: { languages: ['cobol'], concerns: ['mainframe'] } }],
105
+ registry,
106
+ makeConfig(),
107
+ testDir,
108
+ );
109
+
110
+ expect(out.get('cobol-check')).toEqual([]);
111
+ });
112
+
113
+ it('uses checkOverrides when present', () => {
114
+ fixture('src/a.ts');
115
+ fixture('test/b.ts');
116
+ const registry = makeRegistry([
117
+ makeTarget('src', { include: ['src/**/*.ts'] }),
118
+ makeTarget('test', { include: ['test/**/*.ts'] }),
119
+ ]);
120
+
121
+ const out = buildScopeBasedFileMap(
122
+ [{ slug: 'narrow-check', scope: { languages: [], concerns: [] } }],
123
+ registry,
124
+ makeConfig({ checkOverrides: { 'narrow-check': 'test' } }),
125
+ testDir,
126
+ );
127
+
128
+ const files = out.get('narrow-check');
129
+ expect(files?.length).toBe(1);
130
+ expect(files?.[0]?.endsWith('test/b.ts')).toBe(true);
131
+ });
132
+
133
+ it('honors checkOverrides that point to multiple targets', () => {
134
+ fixture('src/a.ts');
135
+ fixture('test/b.ts');
136
+ const registry = makeRegistry([
137
+ makeTarget('src', { include: ['src/**/*.ts'] }),
138
+ makeTarget('test', { include: ['test/**/*.ts'] }),
139
+ ]);
140
+
141
+ const out = buildScopeBasedFileMap(
142
+ [{ slug: 'broad-check', scope: { languages: [], concerns: [] } }],
143
+ registry,
144
+ makeConfig({ checkOverrides: { 'broad-check': ['src', 'test'] } }),
145
+ testDir,
146
+ );
147
+
148
+ expect(out.get('broad-check')?.length).toBe(2);
149
+ });
150
+
151
+ it('applies globalExcludes during pre-resolution', () => {
152
+ fixture('src/a.ts');
153
+ fixture('src/ignore-me/b.ts');
154
+ const registry = makeRegistry([
155
+ makeTarget('src', { include: ['src/**/*.ts'], languages: ['typescript'], concerns: ['backend'] }),
156
+ ]);
157
+
158
+ const out = buildScopeBasedFileMap(
159
+ [{ slug: 'check', scope: { languages: ['typescript'], concerns: ['backend'] } }],
160
+ registry,
161
+ makeConfig({ globalExcludes: ['**/ignore-me/**'] }),
162
+ testDir,
163
+ );
164
+
165
+ const files = out.get('check') ?? [];
166
+ expect(files.some((f) => f.includes('ignore-me'))).toBe(false);
167
+ expect(files.length).toBe(1);
168
+ });
169
+
170
+ it('applies per-target excludes', () => {
171
+ fixture('src/a.ts');
172
+ fixture('src/skip.ts');
173
+ const registry = makeRegistry([
174
+ makeTarget('src', {
175
+ include: ['src/**/*.ts'],
176
+ exclude: ['**/skip.ts'],
177
+ languages: ['typescript'],
178
+ concerns: ['backend'],
179
+ }),
180
+ ]);
181
+
182
+ const out = buildScopeBasedFileMap(
183
+ [{ slug: 'check', scope: { languages: ['typescript'], concerns: ['backend'] } }],
184
+ registry,
185
+ makeConfig(),
186
+ testDir,
187
+ );
188
+
189
+ const files = out.get('check') ?? [];
190
+ expect(files.some((f) => f.endsWith('skip.ts'))).toBe(false);
191
+ });
192
+
193
+ it('skips checks without a scope and without an override', () => {
194
+ fixture('src/a.ts');
195
+ const registry = makeRegistry([
196
+ makeTarget('src', { include: ['src/**/*.ts'] }),
197
+ ]);
198
+
199
+ const out = buildScopeBasedFileMap(
200
+ [{ slug: 'unscoped' }],
201
+ registry,
202
+ makeConfig(),
203
+ testDir,
204
+ );
205
+
206
+ expect(out.has('unscoped')).toBe(false);
207
+ });
208
+ });
@@ -0,0 +1,50 @@
1
+ import { describe, expect, it } from 'vitest';
2
+
3
+ import { mapFindingSeverity, mapTagsToSignalCategory } from '../severity-mapping.js';
4
+
5
+ describe('mapFindingSeverity', () => {
6
+ it('maps "error" to "high"', () => {
7
+ expect(mapFindingSeverity('error')).toBe('high');
8
+ });
9
+
10
+ it('maps "warning" to "medium"', () => {
11
+ expect(mapFindingSeverity('warning')).toBe('medium');
12
+ });
13
+
14
+ it('falls back to "medium" for unknown severities', () => {
15
+ // @ts-expect-error — exercising the runtime default
16
+ expect(mapFindingSeverity('whatever')).toBe('medium');
17
+ });
18
+ });
19
+
20
+ describe('mapTagsToSignalCategory', () => {
21
+ it.each([
22
+ ['security', 'security'],
23
+ ['performance', 'performance'],
24
+ ['architecture', 'architecture'],
25
+ ['resilience', 'resilience'],
26
+ ['testing', 'testing'],
27
+ ['documentation', 'documentation'],
28
+ ])('maps %s tag to %s category', (tag, category) => {
29
+ expect(mapTagsToSignalCategory([tag])).toBe(category);
30
+ });
31
+
32
+ it('maps "quality" tag to "warning" category', () => {
33
+ expect(mapTagsToSignalCategory(['quality'])).toBe('warning');
34
+ });
35
+
36
+ it('falls back to "warning" when no recognized tag matches', () => {
37
+ expect(mapTagsToSignalCategory(['custom-tag'])).toBe('warning');
38
+ });
39
+
40
+ it('returns "warning" for an empty tag list', () => {
41
+ expect(mapTagsToSignalCategory([])).toBe('warning');
42
+ });
43
+
44
+ it('uses the first matching tag (priority order)', () => {
45
+ // 'quality' comes after 'security' lexically but the function iterates the
46
+ // input array in order, so 'security' wins here.
47
+ expect(mapTagsToSignalCategory(['quality', 'security'])).toBe('warning');
48
+ expect(mapTagsToSignalCategory(['security', 'quality'])).toBe('security');
49
+ });
50
+ });