@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.
- package/.turbo/turbo-build.log +4 -0
- package/.turbo/turbo-typecheck.log +4 -0
- package/LICENSE +21 -0
- package/dist/__tests__/gate.test.d.ts +13 -0
- package/dist/__tests__/gate.test.d.ts.map +1 -0
- package/dist/__tests__/gate.test.js +422 -0
- package/dist/__tests__/gate.test.js.map +1 -0
- package/dist/__tests__/sarif.test.d.ts +2 -0
- package/dist/__tests__/sarif.test.d.ts.map +1 -0
- package/dist/__tests__/sarif.test.js +169 -0
- package/dist/__tests__/sarif.test.js.map +1 -0
- package/dist/cli/dashboard.d.ts +6 -0
- package/dist/cli/dashboard.d.ts.map +1 -0
- package/dist/cli/dashboard.js +77 -0
- package/dist/cli/dashboard.js.map +1 -0
- package/dist/cli/fit.d.ts +37 -0
- package/dist/cli/fit.d.ts.map +1 -0
- package/dist/cli/fit.js +539 -0
- package/dist/cli/fit.js.map +1 -0
- package/dist/cli/list-checks.d.ts +6 -0
- package/dist/cli/list-checks.d.ts.map +1 -0
- package/dist/cli/list-checks.js +23 -0
- package/dist/cli/list-checks.js.map +1 -0
- package/dist/cli/list-recipes.d.ts +6 -0
- package/dist/cli/list-recipes.d.ts.map +1 -0
- package/dist/cli/list-recipes.js +31 -0
- package/dist/cli/list-recipes.js.map +1 -0
- package/dist/framework/__tests__/ast-utilities.test.d.ts +2 -0
- package/dist/framework/__tests__/ast-utilities.test.d.ts.map +1 -0
- package/dist/framework/__tests__/ast-utilities.test.js +153 -0
- package/dist/framework/__tests__/ast-utilities.test.js.map +1 -0
- package/dist/framework/__tests__/check-config.test.d.ts +2 -0
- package/dist/framework/__tests__/check-config.test.d.ts.map +1 -0
- package/dist/framework/__tests__/check-config.test.js +56 -0
- package/dist/framework/__tests__/check-config.test.js.map +1 -0
- package/dist/framework/__tests__/command-executor.test.d.ts +2 -0
- package/dist/framework/__tests__/command-executor.test.d.ts.map +1 -0
- package/dist/framework/__tests__/command-executor.test.js +71 -0
- package/dist/framework/__tests__/command-executor.test.js.map +1 -0
- package/dist/framework/__tests__/content-filter-dispatch.test.d.ts +2 -0
- package/dist/framework/__tests__/content-filter-dispatch.test.d.ts.map +1 -0
- package/dist/framework/__tests__/content-filter-dispatch.test.js +104 -0
- package/dist/framework/__tests__/content-filter-dispatch.test.js.map +1 -0
- package/dist/framework/__tests__/content-filter.test.d.ts +2 -0
- package/dist/framework/__tests__/content-filter.test.d.ts.map +1 -0
- package/dist/framework/__tests__/content-filter.test.js +126 -0
- package/dist/framework/__tests__/content-filter.test.js.map +1 -0
- package/dist/framework/__tests__/define-check.test.d.ts +2 -0
- package/dist/framework/__tests__/define-check.test.d.ts.map +1 -0
- package/dist/framework/__tests__/define-check.test.js +155 -0
- package/dist/framework/__tests__/define-check.test.js.map +1 -0
- package/dist/framework/__tests__/directive-inventory.test.d.ts +2 -0
- package/dist/framework/__tests__/directive-inventory.test.d.ts.map +1 -0
- package/dist/framework/__tests__/directive-inventory.test.js +44 -0
- package/dist/framework/__tests__/directive-inventory.test.js.map +1 -0
- package/dist/framework/__tests__/execution-context.test.d.ts +2 -0
- package/dist/framework/__tests__/execution-context.test.d.ts.map +1 -0
- package/dist/framework/__tests__/execution-context.test.js +62 -0
- package/dist/framework/__tests__/execution-context.test.js.map +1 -0
- package/dist/framework/__tests__/file-accessor.test.d.ts +2 -0
- package/dist/framework/__tests__/file-accessor.test.d.ts.map +1 -0
- package/dist/framework/__tests__/file-accessor.test.js +106 -0
- package/dist/framework/__tests__/file-accessor.test.js.map +1 -0
- package/dist/framework/__tests__/file-cache.test.d.ts +2 -0
- package/dist/framework/__tests__/file-cache.test.d.ts.map +1 -0
- package/dist/framework/__tests__/file-cache.test.js +122 -0
- package/dist/framework/__tests__/file-cache.test.js.map +1 -0
- package/dist/framework/__tests__/import-graph.test.d.ts +15 -0
- package/dist/framework/__tests__/import-graph.test.d.ts.map +1 -0
- package/dist/framework/__tests__/import-graph.test.js +164 -0
- package/dist/framework/__tests__/import-graph.test.js.map +1 -0
- package/dist/framework/__tests__/path-matcher.test.d.ts +2 -0
- package/dist/framework/__tests__/path-matcher.test.d.ts.map +1 -0
- package/dist/framework/__tests__/path-matcher.test.js +113 -0
- package/dist/framework/__tests__/path-matcher.test.js.map +1 -0
- package/dist/framework/__tests__/register-helpers.test.d.ts +2 -0
- package/dist/framework/__tests__/register-helpers.test.d.ts.map +1 -0
- package/dist/framework/__tests__/register-helpers.test.js +42 -0
- package/dist/framework/__tests__/register-helpers.test.js.map +1 -0
- package/dist/framework/__tests__/registry.test.d.ts +2 -0
- package/dist/framework/__tests__/registry.test.d.ts.map +1 -0
- package/dist/framework/__tests__/registry.test.js +208 -0
- package/dist/framework/__tests__/registry.test.js.map +1 -0
- package/dist/framework/__tests__/result-builder.test.d.ts +2 -0
- package/dist/framework/__tests__/result-builder.test.d.ts.map +1 -0
- package/dist/framework/__tests__/result-builder.test.js +153 -0
- package/dist/framework/__tests__/result-builder.test.js.map +1 -0
- package/dist/framework/__tests__/scope-resolver.test.d.ts +2 -0
- package/dist/framework/__tests__/scope-resolver.test.d.ts.map +1 -0
- package/dist/framework/__tests__/scope-resolver.test.js +140 -0
- package/dist/framework/__tests__/scope-resolver.test.js.map +1 -0
- package/dist/framework/__tests__/severity-mapping.test.d.ts +2 -0
- package/dist/framework/__tests__/severity-mapping.test.d.ts.map +1 -0
- package/dist/framework/__tests__/severity-mapping.test.js +42 -0
- package/dist/framework/__tests__/severity-mapping.test.js.map +1 -0
- package/dist/framework/__tests__/strip-literals.test.d.ts +2 -0
- package/dist/framework/__tests__/strip-literals.test.d.ts.map +1 -0
- package/dist/framework/__tests__/strip-literals.test.js +87 -0
- package/dist/framework/__tests__/strip-literals.test.js.map +1 -0
- package/dist/framework/abortable-exec.d.ts +34 -0
- package/dist/framework/abortable-exec.d.ts.map +1 -0
- package/dist/framework/abortable-exec.js +136 -0
- package/dist/framework/abortable-exec.js.map +1 -0
- package/dist/framework/ast-utilities.d.ts +41 -0
- package/dist/framework/ast-utilities.d.ts.map +1 -0
- package/dist/framework/ast-utilities.js +106 -0
- package/dist/framework/ast-utilities.js.map +1 -0
- package/dist/framework/check-config.d.ts +171 -0
- package/dist/framework/check-config.d.ts.map +1 -0
- package/dist/framework/check-config.js +114 -0
- package/dist/framework/check-config.js.map +1 -0
- package/dist/framework/check-types.d.ts +57 -0
- package/dist/framework/check-types.d.ts.map +1 -0
- package/dist/framework/check-types.js +35 -0
- package/dist/framework/check-types.js.map +1 -0
- package/dist/framework/command-executor.d.ts +25 -0
- package/dist/framework/command-executor.d.ts.map +1 -0
- package/dist/framework/command-executor.js +63 -0
- package/dist/framework/command-executor.js.map +1 -0
- package/dist/framework/constants.d.ts +9 -0
- package/dist/framework/constants.d.ts.map +1 -0
- package/dist/framework/constants.js +16 -0
- package/dist/framework/constants.js.map +1 -0
- package/dist/framework/content-filter.d.ts +33 -0
- package/dist/framework/content-filter.d.ts.map +1 -0
- package/dist/framework/content-filter.js +236 -0
- package/dist/framework/content-filter.js.map +1 -0
- package/dist/framework/define-check.d.ts +38 -0
- package/dist/framework/define-check.d.ts.map +1 -0
- package/dist/framework/define-check.js +252 -0
- package/dist/framework/define-check.js.map +1 -0
- package/dist/framework/directive-inventory.d.ts +34 -0
- package/dist/framework/directive-inventory.d.ts.map +1 -0
- package/dist/framework/directive-inventory.js +77 -0
- package/dist/framework/directive-inventory.js.map +1 -0
- package/dist/framework/directive-parsing.d.ts +20 -0
- package/dist/framework/directive-parsing.d.ts.map +1 -0
- package/dist/framework/directive-parsing.js +121 -0
- package/dist/framework/directive-parsing.js.map +1 -0
- package/dist/framework/execution-context.d.ts +95 -0
- package/dist/framework/execution-context.d.ts.map +1 -0
- package/dist/framework/execution-context.js +122 -0
- package/dist/framework/execution-context.js.map +1 -0
- package/dist/framework/file-accessor.d.ts +20 -0
- package/dist/framework/file-accessor.d.ts.map +1 -0
- package/dist/framework/file-accessor.js +122 -0
- package/dist/framework/file-accessor.js.map +1 -0
- package/dist/framework/file-cache.d.ts +70 -0
- package/dist/framework/file-cache.d.ts.map +1 -0
- package/dist/framework/file-cache.js +178 -0
- package/dist/framework/file-cache.js.map +1 -0
- package/dist/framework/file-type-filter.d.ts +11 -0
- package/dist/framework/file-type-filter.d.ts.map +1 -0
- package/dist/framework/file-type-filter.js +21 -0
- package/dist/framework/file-type-filter.js.map +1 -0
- package/dist/framework/ignore-processing.d.ts +22 -0
- package/dist/framework/ignore-processing.d.ts.map +1 -0
- package/dist/framework/ignore-processing.js +241 -0
- package/dist/framework/ignore-processing.js.map +1 -0
- package/dist/framework/import-graph.d.ts +51 -0
- package/dist/framework/import-graph.d.ts.map +1 -0
- package/dist/framework/import-graph.js +216 -0
- package/dist/framework/import-graph.js.map +1 -0
- package/dist/framework/memory-profiler.d.ts +53 -0
- package/dist/framework/memory-profiler.d.ts.map +1 -0
- package/dist/framework/memory-profiler.js +92 -0
- package/dist/framework/memory-profiler.js.map +1 -0
- package/dist/framework/parse-cache.d.ts +23 -0
- package/dist/framework/parse-cache.d.ts.map +1 -0
- package/dist/framework/parse-cache.js +37 -0
- package/dist/framework/parse-cache.js.map +1 -0
- package/dist/framework/path-matcher.d.ts +86 -0
- package/dist/framework/path-matcher.d.ts.map +1 -0
- package/dist/framework/path-matcher.js +138 -0
- package/dist/framework/path-matcher.js.map +1 -0
- package/dist/framework/register-helpers.d.ts +10 -0
- package/dist/framework/register-helpers.d.ts.map +1 -0
- package/dist/framework/register-helpers.js +17 -0
- package/dist/framework/register-helpers.js.map +1 -0
- package/dist/framework/registry.d.ts +41 -0
- package/dist/framework/registry.d.ts.map +1 -0
- package/dist/framework/registry.js +103 -0
- package/dist/framework/registry.js.map +1 -0
- package/dist/framework/result-builder.d.ts +74 -0
- package/dist/framework/result-builder.d.ts.map +1 -0
- package/dist/framework/result-builder.js +154 -0
- package/dist/framework/result-builder.js.map +1 -0
- package/dist/framework/scope-resolver.d.ts +23 -0
- package/dist/framework/scope-resolver.d.ts.map +1 -0
- package/dist/framework/scope-resolver.js +201 -0
- package/dist/framework/scope-resolver.js.map +1 -0
- package/dist/framework/severity-mapping.d.ts +13 -0
- package/dist/framework/severity-mapping.d.ts.map +1 -0
- package/dist/framework/severity-mapping.js +51 -0
- package/dist/framework/severity-mapping.js.map +1 -0
- package/dist/framework/strip-literals.d.ts +48 -0
- package/dist/framework/strip-literals.d.ts.map +1 -0
- package/dist/framework/strip-literals.js +188 -0
- package/dist/framework/strip-literals.js.map +1 -0
- package/dist/gate.d.ts +74 -0
- package/dist/gate.d.ts.map +1 -0
- package/dist/gate.js +257 -0
- package/dist/gate.js.map +1 -0
- package/dist/index.d.ts +47 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +51 -0
- package/dist/index.js.map +1 -0
- package/dist/plugins/__tests__/check-package-discovery.test.d.ts +2 -0
- package/dist/plugins/__tests__/check-package-discovery.test.d.ts.map +1 -0
- package/dist/plugins/__tests__/check-package-discovery.test.js +170 -0
- package/dist/plugins/__tests__/check-package-discovery.test.js.map +1 -0
- package/dist/plugins/__tests__/lang-domain.test.d.ts +2 -0
- package/dist/plugins/__tests__/lang-domain.test.d.ts.map +1 -0
- package/dist/plugins/__tests__/lang-domain.test.js +171 -0
- package/dist/plugins/__tests__/lang-domain.test.js.map +1 -0
- package/dist/plugins/__tests__/loader.test.d.ts +2 -0
- package/dist/plugins/__tests__/loader.test.d.ts.map +1 -0
- package/dist/plugins/__tests__/loader.test.js +194 -0
- package/dist/plugins/__tests__/loader.test.js.map +1 -0
- package/dist/plugins/check-package-discovery.d.ts +73 -0
- package/dist/plugins/check-package-discovery.d.ts.map +1 -0
- package/dist/plugins/check-package-discovery.js +212 -0
- package/dist/plugins/check-package-discovery.js.map +1 -0
- package/dist/plugins/loader.d.ts +31 -0
- package/dist/plugins/loader.d.ts.map +1 -0
- package/dist/plugins/loader.js +290 -0
- package/dist/plugins/loader.js.map +1 -0
- package/dist/plugins/types.d.ts +23 -0
- package/dist/plugins/types.d.ts.map +1 -0
- package/dist/plugins/types.js +9 -0
- package/dist/plugins/types.js.map +1 -0
- package/dist/recipes/__tests__/built-in-recipes.test.d.ts +2 -0
- package/dist/recipes/__tests__/built-in-recipes.test.d.ts.map +1 -0
- package/dist/recipes/__tests__/built-in-recipes.test.js +93 -0
- package/dist/recipes/__tests__/built-in-recipes.test.js.map +1 -0
- package/dist/recipes/__tests__/check-config.test.d.ts +5 -0
- package/dist/recipes/__tests__/check-config.test.d.ts.map +1 -0
- package/dist/recipes/__tests__/check-config.test.js +37 -0
- package/dist/recipes/__tests__/check-config.test.js.map +1 -0
- package/dist/recipes/__tests__/check-resolution.test.d.ts +2 -0
- package/dist/recipes/__tests__/check-resolution.test.d.ts.map +1 -0
- package/dist/recipes/__tests__/check-resolution.test.js +135 -0
- package/dist/recipes/__tests__/check-resolution.test.js.map +1 -0
- package/dist/recipes/__tests__/registry.test.d.ts +2 -0
- package/dist/recipes/__tests__/registry.test.d.ts.map +1 -0
- package/dist/recipes/__tests__/registry.test.js +97 -0
- package/dist/recipes/__tests__/registry.test.js.map +1 -0
- package/dist/recipes/__tests__/retry.test.d.ts +2 -0
- package/dist/recipes/__tests__/retry.test.d.ts.map +1 -0
- package/dist/recipes/__tests__/retry.test.js +75 -0
- package/dist/recipes/__tests__/retry.test.js.map +1 -0
- package/dist/recipes/__tests__/service.test.d.ts +11 -0
- package/dist/recipes/__tests__/service.test.d.ts.map +1 -0
- package/dist/recipes/__tests__/service.test.js +482 -0
- package/dist/recipes/__tests__/service.test.js.map +1 -0
- package/dist/recipes/built-in-recipes.d.ts +14 -0
- package/dist/recipes/built-in-recipes.d.ts.map +1 -0
- package/dist/recipes/built-in-recipes.js +247 -0
- package/dist/recipes/built-in-recipes.js.map +1 -0
- package/dist/recipes/check-config.d.ts +40 -0
- package/dist/recipes/check-config.d.ts.map +1 -0
- package/dist/recipes/check-config.js +61 -0
- package/dist/recipes/check-config.js.map +1 -0
- package/dist/recipes/check-resolution.d.ts +21 -0
- package/dist/recipes/check-resolution.d.ts.map +1 -0
- package/dist/recipes/check-resolution.js +121 -0
- package/dist/recipes/check-resolution.js.map +1 -0
- package/dist/recipes/check-result-processor.d.ts +51 -0
- package/dist/recipes/check-result-processor.d.ts.map +1 -0
- package/dist/recipes/check-result-processor.js +158 -0
- package/dist/recipes/check-result-processor.js.map +1 -0
- package/dist/recipes/parallel-execution.d.ts +33 -0
- package/dist/recipes/parallel-execution.d.ts.map +1 -0
- package/dist/recipes/parallel-execution.js +142 -0
- package/dist/recipes/parallel-execution.js.map +1 -0
- package/dist/recipes/registry.d.ts +81 -0
- package/dist/recipes/registry.d.ts.map +1 -0
- package/dist/recipes/registry.js +131 -0
- package/dist/recipes/registry.js.map +1 -0
- package/dist/recipes/retry.d.ts +25 -0
- package/dist/recipes/retry.d.ts.map +1 -0
- package/dist/recipes/retry.js +44 -0
- package/dist/recipes/retry.js.map +1 -0
- package/dist/recipes/sequential-execution.d.ts +10 -0
- package/dist/recipes/sequential-execution.d.ts.map +1 -0
- package/dist/recipes/sequential-execution.js +122 -0
- package/dist/recipes/sequential-execution.js.map +1 -0
- package/dist/recipes/service-types.d.ts +84 -0
- package/dist/recipes/service-types.d.ts.map +1 -0
- package/dist/recipes/service-types.js +8 -0
- package/dist/recipes/service-types.js.map +1 -0
- package/dist/recipes/service.d.ts +71 -0
- package/dist/recipes/service.d.ts.map +1 -0
- package/dist/recipes/service.js +331 -0
- package/dist/recipes/service.js.map +1 -0
- package/dist/recipes/types.d.ts +154 -0
- package/dist/recipes/types.d.ts.map +1 -0
- package/dist/recipes/types.js +54 -0
- package/dist/recipes/types.js.map +1 -0
- package/dist/sarif.d.ts +34 -0
- package/dist/sarif.d.ts.map +1 -0
- package/dist/sarif.js +192 -0
- package/dist/sarif.js.map +1 -0
- package/dist/signalers/__tests__/loader.test.d.ts +2 -0
- package/dist/signalers/__tests__/loader.test.d.ts.map +1 -0
- package/dist/signalers/__tests__/loader.test.js +74 -0
- package/dist/signalers/__tests__/loader.test.js.map +1 -0
- package/dist/signalers/index.d.ts +8 -0
- package/dist/signalers/index.d.ts.map +1 -0
- package/dist/signalers/index.js +9 -0
- package/dist/signalers/index.js.map +1 -0
- package/dist/signalers/loader.d.ts +24 -0
- package/dist/signalers/loader.d.ts.map +1 -0
- package/dist/signalers/loader.js +108 -0
- package/dist/signalers/loader.js.map +1 -0
- package/dist/signalers/schema.d.ts +288 -0
- package/dist/signalers/schema.d.ts.map +1 -0
- package/dist/signalers/schema.js +99 -0
- package/dist/signalers/schema.js.map +1 -0
- package/dist/signalers/types.d.ts +13 -0
- package/dist/signalers/types.d.ts.map +1 -0
- package/dist/signalers/types.js +5 -0
- package/dist/signalers/types.js.map +1 -0
- package/dist/targets/__tests__/loader.test.d.ts +2 -0
- package/dist/targets/__tests__/loader.test.d.ts.map +1 -0
- package/dist/targets/__tests__/loader.test.js +127 -0
- package/dist/targets/__tests__/loader.test.js.map +1 -0
- package/dist/targets/__tests__/resolver.test.d.ts +2 -0
- package/dist/targets/__tests__/resolver.test.d.ts.map +1 -0
- package/dist/targets/__tests__/resolver.test.js +54 -0
- package/dist/targets/__tests__/resolver.test.js.map +1 -0
- package/dist/targets/__tests__/target-registry.test.d.ts +2 -0
- package/dist/targets/__tests__/target-registry.test.d.ts.map +1 -0
- package/dist/targets/__tests__/target-registry.test.js +89 -0
- package/dist/targets/__tests__/target-registry.test.js.map +1 -0
- package/dist/targets/index.d.ts +10 -0
- package/dist/targets/index.d.ts.map +1 -0
- package/dist/targets/index.js +12 -0
- package/dist/targets/index.js.map +1 -0
- package/dist/targets/loader.d.ts +19 -0
- package/dist/targets/loader.d.ts.map +1 -0
- package/dist/targets/loader.js +159 -0
- package/dist/targets/loader.js.map +1 -0
- package/dist/targets/resolver.d.ts +19 -0
- package/dist/targets/resolver.d.ts.map +1 -0
- package/dist/targets/resolver.js +37 -0
- package/dist/targets/resolver.js.map +1 -0
- package/dist/targets/target-registry.d.ts +61 -0
- package/dist/targets/target-registry.d.ts.map +1 -0
- package/dist/targets/target-registry.js +93 -0
- package/dist/targets/target-registry.js.map +1 -0
- package/dist/targets/types.d.ts +85 -0
- package/dist/targets/types.d.ts.map +1 -0
- package/dist/targets/types.js +5 -0
- package/dist/targets/types.js.map +1 -0
- package/dist/tool.d.ts +17 -0
- package/dist/tool.d.ts.map +1 -0
- package/dist/tool.js +282 -0
- package/dist/tool.js.map +1 -0
- package/dist/types/findings.d.ts +117 -0
- package/dist/types/findings.d.ts.map +1 -0
- package/dist/types/findings.js +93 -0
- package/dist/types/findings.js.map +1 -0
- package/dist/types/severity.d.ts +15 -0
- package/dist/types/severity.d.ts.map +1 -0
- package/dist/types/severity.js +36 -0
- package/dist/types/severity.js.map +1 -0
- package/package.json +45 -0
- package/src/__tests__/gate.test.ts +537 -0
- package/src/__tests__/sarif.test.ts +201 -0
- package/src/cli/dashboard.ts +93 -0
- package/src/cli/fit.ts +612 -0
- package/src/cli/list-checks.ts +32 -0
- package/src/cli/list-recipes.ts +38 -0
- package/src/framework/__tests__/ast-utilities.test.ts +157 -0
- package/src/framework/__tests__/check-config.test.ts +65 -0
- package/src/framework/__tests__/command-executor.test.ts +79 -0
- package/src/framework/__tests__/content-filter-dispatch.test.ts +132 -0
- package/src/framework/__tests__/content-filter.test.ts +136 -0
- package/src/framework/__tests__/define-check.test.ts +180 -0
- package/src/framework/__tests__/directive-inventory.test.ts +53 -0
- package/src/framework/__tests__/execution-context.test.ts +80 -0
- package/src/framework/__tests__/file-accessor.test.ts +121 -0
- package/src/framework/__tests__/file-cache.test.ts +142 -0
- package/src/framework/__tests__/import-graph.test.ts +282 -0
- package/src/framework/__tests__/path-matcher.test.ts +130 -0
- package/src/framework/__tests__/register-helpers.test.ts +51 -0
- package/src/framework/__tests__/registry.test.ts +243 -0
- package/src/framework/__tests__/result-builder.test.ts +178 -0
- package/src/framework/__tests__/scope-resolver.test.ts +208 -0
- package/src/framework/__tests__/severity-mapping.test.ts +50 -0
- package/src/framework/__tests__/strip-literals.test.ts +109 -0
- package/src/framework/abortable-exec.ts +177 -0
- package/src/framework/ast-utilities.ts +112 -0
- package/src/framework/check-config.ts +339 -0
- package/src/framework/check-types.ts +77 -0
- package/src/framework/command-executor.ts +100 -0
- package/src/framework/constants.ts +16 -0
- package/src/framework/content-filter.ts +288 -0
- package/src/framework/define-check.ts +336 -0
- package/src/framework/directive-inventory.ts +110 -0
- package/src/framework/directive-parsing.ts +152 -0
- package/src/framework/execution-context.ts +247 -0
- package/src/framework/file-accessor.ts +171 -0
- package/src/framework/file-cache.ts +220 -0
- package/src/framework/file-type-filter.ts +25 -0
- package/src/framework/ignore-processing.ts +350 -0
- package/src/framework/import-graph.ts +280 -0
- package/src/framework/memory-profiler.ts +145 -0
- package/src/framework/parse-cache.ts +38 -0
- package/src/framework/path-matcher.ts +191 -0
- package/src/framework/register-helpers.ts +20 -0
- package/src/framework/registry.ts +125 -0
- package/src/framework/result-builder.ts +225 -0
- package/src/framework/scope-resolver.ts +262 -0
- package/src/framework/severity-mapping.ts +56 -0
- package/src/framework/strip-literals.ts +200 -0
- package/src/gate.ts +337 -0
- package/src/index.ts +110 -0
- package/src/plugins/__tests__/check-package-discovery.test.ts +204 -0
- package/src/plugins/__tests__/lang-domain.test.ts +198 -0
- package/src/plugins/__tests__/loader.test.ts +226 -0
- package/src/plugins/check-package-discovery.ts +242 -0
- package/src/plugins/loader.ts +327 -0
- package/src/plugins/types.ts +25 -0
- package/src/recipes/__tests__/built-in-recipes.test.ts +107 -0
- package/src/recipes/__tests__/check-config.test.ts +51 -0
- package/src/recipes/__tests__/check-resolution.test.ts +185 -0
- package/src/recipes/__tests__/registry.test.ts +115 -0
- package/src/recipes/__tests__/retry.test.ts +83 -0
- package/src/recipes/__tests__/service.test.ts +572 -0
- package/src/recipes/built-in-recipes.ts +273 -0
- package/src/recipes/check-config.ts +64 -0
- package/src/recipes/check-resolution.ts +169 -0
- package/src/recipes/check-result-processor.ts +258 -0
- package/src/recipes/parallel-execution.ts +220 -0
- package/src/recipes/registry.ts +192 -0
- package/src/recipes/retry.ts +69 -0
- package/src/recipes/sequential-execution.ts +139 -0
- package/src/recipes/service-types.ts +105 -0
- package/src/recipes/service.ts +400 -0
- package/src/recipes/types.ts +247 -0
- package/src/sarif.ts +232 -0
- package/src/signalers/__tests__/loader.test.ts +99 -0
- package/src/signalers/index.ts +9 -0
- package/src/signalers/loader.ts +141 -0
- package/src/signalers/schema.ts +117 -0
- package/src/signalers/types.ts +15 -0
- package/src/targets/__tests__/loader.test.ts +170 -0
- package/src/targets/__tests__/resolver.test.ts +74 -0
- package/src/targets/__tests__/target-registry.test.ts +103 -0
- package/src/targets/index.ts +13 -0
- package/src/targets/loader.ts +214 -0
- package/src/targets/resolver.ts +44 -0
- package/src/targets/target-registry.ts +111 -0
- package/src/targets/types.ts +89 -0
- package/src/tool.ts +302 -0
- package/src/types/findings.ts +239 -0
- package/src/types/severity.ts +39 -0
- package/tsconfig.json +8 -0
- 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
|
+
});
|