vaspera 2.9.2 → 2.10.1
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/CHANGELOG.md +68 -0
- package/README.md +58 -1
- package/dist/__tests__/autofix/branch-manager.test.d.ts +2 -0
- package/dist/__tests__/autofix/branch-manager.test.d.ts.map +1 -0
- package/dist/__tests__/autofix/branch-manager.test.js +60 -0
- package/dist/__tests__/autofix/branch-manager.test.js.map +1 -0
- package/dist/__tests__/autofix/commit-generator.test.d.ts +2 -0
- package/dist/__tests__/autofix/commit-generator.test.d.ts.map +1 -0
- package/dist/__tests__/autofix/commit-generator.test.js +147 -0
- package/dist/__tests__/autofix/commit-generator.test.js.map +1 -0
- package/dist/__tests__/autofix/constitution.test.d.ts +9 -0
- package/dist/__tests__/autofix/constitution.test.d.ts.map +1 -0
- package/dist/__tests__/autofix/constitution.test.js +421 -0
- package/dist/__tests__/autofix/constitution.test.js.map +1 -0
- package/dist/__tests__/autofix/pr-generator.test.d.ts +2 -0
- package/dist/__tests__/autofix/pr-generator.test.d.ts.map +1 -0
- package/dist/__tests__/autofix/pr-generator.test.js +152 -0
- package/dist/__tests__/autofix/pr-generator.test.js.map +1 -0
- package/dist/__tests__/property-test-helpers.d.ts +87 -0
- package/dist/__tests__/property-test-helpers.d.ts.map +1 -0
- package/dist/__tests__/property-test-helpers.js +136 -0
- package/dist/__tests__/property-test-helpers.js.map +1 -0
- package/dist/__tests__/scanners/ai-code/ai-detector.test.d.ts +2 -0
- package/dist/__tests__/scanners/ai-code/ai-detector.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/ai-code/ai-detector.test.js +188 -0
- package/dist/__tests__/scanners/ai-code/ai-detector.test.js.map +1 -0
- package/dist/__tests__/scanners/ai-code/confidence-scorer.test.d.ts +2 -0
- package/dist/__tests__/scanners/ai-code/confidence-scorer.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/ai-code/confidence-scorer.test.js +363 -0
- package/dist/__tests__/scanners/ai-code/confidence-scorer.test.js.map +1 -0
- package/dist/__tests__/scanners/ai-code/hallucination-checker.test.d.ts +2 -0
- package/dist/__tests__/scanners/ai-code/hallucination-checker.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/ai-code/hallucination-checker.test.js +226 -0
- package/dist/__tests__/scanners/ai-code/hallucination-checker.test.js.map +1 -0
- package/dist/__tests__/scanners/ai-code/index.test.d.ts +2 -0
- package/dist/__tests__/scanners/ai-code/index.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/ai-code/index.test.js +214 -0
- package/dist/__tests__/scanners/ai-code/index.test.js.map +1 -0
- package/dist/__tests__/scanners/dast/index.test.d.ts +2 -0
- package/dist/__tests__/scanners/dast/index.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/dast/index.test.js +183 -0
- package/dist/__tests__/scanners/dast/index.test.js.map +1 -0
- package/dist/__tests__/scanners/dast/nuclei.test.d.ts +2 -0
- package/dist/__tests__/scanners/dast/nuclei.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/dast/nuclei.test.js +166 -0
- package/dist/__tests__/scanners/dast/nuclei.test.js.map +1 -0
- package/dist/__tests__/scanners/dast/zap.test.d.ts +2 -0
- package/dist/__tests__/scanners/dast/zap.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/dast/zap.test.js +158 -0
- package/dist/__tests__/scanners/dast/zap.test.js.map +1 -0
- package/dist/__tests__/scanners/deploy/health-checker.test.d.ts +2 -0
- package/dist/__tests__/scanners/deploy/health-checker.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/deploy/health-checker.test.js +67 -0
- package/dist/__tests__/scanners/deploy/health-checker.test.js.map +1 -0
- package/dist/__tests__/scanners/deploy/index.test.d.ts +2 -0
- package/dist/__tests__/scanners/deploy/index.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/deploy/index.test.js +84 -0
- package/dist/__tests__/scanners/deploy/index.test.js.map +1 -0
- package/dist/__tests__/scanners/deploy/provider-detector.test.d.ts +2 -0
- package/dist/__tests__/scanners/deploy/provider-detector.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/deploy/provider-detector.test.js +88 -0
- package/dist/__tests__/scanners/deploy/provider-detector.test.js.map +1 -0
- package/dist/__tests__/scanners/deploy/types.test.d.ts +2 -0
- package/dist/__tests__/scanners/deploy/types.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/deploy/types.test.js +126 -0
- package/dist/__tests__/scanners/deploy/types.test.js.map +1 -0
- package/dist/__tests__/scanners/fp-feedback.test.d.ts +2 -0
- package/dist/__tests__/scanners/fp-feedback.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/fp-feedback.test.js +202 -0
- package/dist/__tests__/scanners/fp-feedback.test.js.map +1 -0
- package/dist/__tests__/scanners/fp-filter.property.test.d.ts +9 -0
- package/dist/__tests__/scanners/fp-filter.property.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/fp-filter.property.test.js +253 -0
- package/dist/__tests__/scanners/fp-filter.property.test.js.map +1 -0
- package/dist/__tests__/scanners/fp-filter.test.d.ts +2 -0
- package/dist/__tests__/scanners/fp-filter.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/fp-filter.test.js +234 -0
- package/dist/__tests__/scanners/fp-filter.test.js.map +1 -0
- package/dist/__tests__/scanners/fp-tracker.test.d.ts +2 -0
- package/dist/__tests__/scanners/fp-tracker.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/fp-tracker.test.js +262 -0
- package/dist/__tests__/scanners/fp-tracker.test.js.map +1 -0
- package/dist/__tests__/scanners/logic/endpoint-analyzer.property.test.d.ts +10 -0
- package/dist/__tests__/scanners/logic/endpoint-analyzer.property.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/logic/endpoint-analyzer.property.test.js +238 -0
- package/dist/__tests__/scanners/logic/endpoint-analyzer.property.test.js.map +1 -0
- package/dist/__tests__/scanners/logic/endpoint-analyzer.test.d.ts +2 -0
- package/dist/__tests__/scanners/logic/endpoint-analyzer.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/logic/endpoint-analyzer.test.js +55 -0
- package/dist/__tests__/scanners/logic/endpoint-analyzer.test.js.map +1 -0
- package/dist/__tests__/scanners/logic/index.test.d.ts +2 -0
- package/dist/__tests__/scanners/logic/index.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/logic/index.test.js +165 -0
- package/dist/__tests__/scanners/logic/index.test.js.map +1 -0
- package/dist/__tests__/scanners/logic/types.test.d.ts +2 -0
- package/dist/__tests__/scanners/logic/types.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/logic/types.test.js +85 -0
- package/dist/__tests__/scanners/logic/types.test.js.map +1 -0
- package/dist/__tests__/scanners/runtime/app-launcher.test.d.ts +2 -0
- package/dist/__tests__/scanners/runtime/app-launcher.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/runtime/app-launcher.test.js +94 -0
- package/dist/__tests__/scanners/runtime/app-launcher.test.js.map +1 -0
- package/dist/__tests__/scanners/runtime/golden-path-runner.test.d.ts +2 -0
- package/dist/__tests__/scanners/runtime/golden-path-runner.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/runtime/golden-path-runner.test.js +195 -0
- package/dist/__tests__/scanners/runtime/golden-path-runner.test.js.map +1 -0
- package/dist/__tests__/scanners/runtime/index.test.d.ts +2 -0
- package/dist/__tests__/scanners/runtime/index.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/runtime/index.test.js +120 -0
- package/dist/__tests__/scanners/runtime/index.test.js.map +1 -0
- package/dist/__tests__/scanners/runtime/types.test.d.ts +2 -0
- package/dist/__tests__/scanners/runtime/types.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/runtime/types.test.js +126 -0
- package/dist/__tests__/scanners/runtime/types.test.js.map +1 -0
- package/dist/__tests__/scanners/scale/bottleneck-detector.test.d.ts +2 -0
- package/dist/__tests__/scanners/scale/bottleneck-detector.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/scale/bottleneck-detector.test.js +187 -0
- package/dist/__tests__/scanners/scale/bottleneck-detector.test.js.map +1 -0
- package/dist/__tests__/scanners/scale/index.test.d.ts +2 -0
- package/dist/__tests__/scanners/scale/index.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/scale/index.test.js +87 -0
- package/dist/__tests__/scanners/scale/index.test.js.map +1 -0
- package/dist/__tests__/scanners/scale/load-profiler.test.d.ts +2 -0
- package/dist/__tests__/scanners/scale/load-profiler.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/scale/load-profiler.test.js +122 -0
- package/dist/__tests__/scanners/scale/load-profiler.test.js.map +1 -0
- package/dist/__tests__/scanners/scale/types.test.d.ts +2 -0
- package/dist/__tests__/scanners/scale/types.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/scale/types.test.js +129 -0
- package/dist/__tests__/scanners/scale/types.test.js.map +1 -0
- package/dist/action/pr-comment.test.js +4 -0
- package/dist/action/pr-comment.test.js.map +1 -1
- package/dist/action/sarif-upload.test.js +4 -0
- package/dist/action/sarif-upload.test.js.map +1 -1
- package/dist/autofix/branch-manager.d.ts +115 -0
- package/dist/autofix/branch-manager.d.ts.map +1 -0
- package/dist/autofix/branch-manager.js +308 -0
- package/dist/autofix/branch-manager.js.map +1 -0
- package/dist/autofix/commit-generator.d.ts +55 -0
- package/dist/autofix/commit-generator.d.ts.map +1 -0
- package/dist/autofix/commit-generator.js +277 -0
- package/dist/autofix/commit-generator.js.map +1 -0
- package/dist/autofix/constitution.d.ts +77 -0
- package/dist/autofix/constitution.d.ts.map +1 -0
- package/dist/autofix/constitution.js +261 -0
- package/dist/autofix/constitution.js.map +1 -0
- package/dist/autofix/constitution.schema.d.ts +441 -0
- package/dist/autofix/constitution.schema.d.ts.map +1 -0
- package/dist/autofix/constitution.schema.js +144 -0
- package/dist/autofix/constitution.schema.js.map +1 -0
- package/dist/autofix/index.d.ts +13 -0
- package/dist/autofix/index.d.ts.map +1 -0
- package/dist/autofix/index.js +15 -0
- package/dist/autofix/index.js.map +1 -0
- package/dist/autofix/pr-generator.d.ts +57 -0
- package/dist/autofix/pr-generator.d.ts.map +1 -0
- package/dist/autofix/pr-generator.js +597 -0
- package/dist/autofix/pr-generator.js.map +1 -0
- package/dist/autofix/types.d.ts +151 -0
- package/dist/autofix/types.d.ts.map +1 -0
- package/dist/autofix/types.js +22 -0
- package/dist/autofix/types.js.map +1 -0
- package/dist/eval/fixtures.d.ts +20 -0
- package/dist/eval/fixtures.d.ts.map +1 -1
- package/dist/eval/fixtures.js +430 -0
- package/dist/eval/fixtures.js.map +1 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +874 -0
- package/dist/index.js.map +1 -1
- package/dist/install-skills.d.ts +11 -0
- package/dist/install-skills.d.ts.map +1 -0
- package/dist/install-skills.js +81 -0
- package/dist/install-skills.js.map +1 -0
- package/dist/scanners/ai-code/ai-detector.d.ts +25 -0
- package/dist/scanners/ai-code/ai-detector.d.ts.map +1 -0
- package/dist/scanners/ai-code/ai-detector.js +192 -0
- package/dist/scanners/ai-code/ai-detector.js.map +1 -0
- package/dist/scanners/ai-code/confidence-scorer.d.ts +40 -0
- package/dist/scanners/ai-code/confidence-scorer.d.ts.map +1 -0
- package/dist/scanners/ai-code/confidence-scorer.js +148 -0
- package/dist/scanners/ai-code/confidence-scorer.js.map +1 -0
- package/dist/scanners/ai-code/hallucination-checker.d.ts +36 -0
- package/dist/scanners/ai-code/hallucination-checker.d.ts.map +1 -0
- package/dist/scanners/ai-code/hallucination-checker.js +298 -0
- package/dist/scanners/ai-code/hallucination-checker.js.map +1 -0
- package/dist/scanners/ai-code/index.d.ts +30 -0
- package/dist/scanners/ai-code/index.d.ts.map +1 -0
- package/dist/scanners/ai-code/index.js +224 -0
- package/dist/scanners/ai-code/index.js.map +1 -0
- package/dist/scanners/ai-code/types.d.ts +192 -0
- package/dist/scanners/ai-code/types.d.ts.map +1 -0
- package/dist/scanners/ai-code/types.js +37 -0
- package/dist/scanners/ai-code/types.js.map +1 -0
- package/dist/scanners/cache.d.ts.map +1 -1
- package/dist/scanners/cache.js +4 -0
- package/dist/scanners/cache.js.map +1 -1
- package/dist/scanners/dast/index.d.ts +39 -0
- package/dist/scanners/dast/index.d.ts.map +1 -0
- package/dist/scanners/dast/index.js +259 -0
- package/dist/scanners/dast/index.js.map +1 -0
- package/dist/scanners/dast/nuclei.d.ts +26 -0
- package/dist/scanners/dast/nuclei.d.ts.map +1 -0
- package/dist/scanners/dast/nuclei.js +354 -0
- package/dist/scanners/dast/nuclei.js.map +1 -0
- package/dist/scanners/dast/types.d.ts +306 -0
- package/dist/scanners/dast/types.d.ts.map +1 -0
- package/dist/scanners/dast/types.js +52 -0
- package/dist/scanners/dast/types.js.map +1 -0
- package/dist/scanners/dast/zap.d.ts +26 -0
- package/dist/scanners/dast/zap.d.ts.map +1 -0
- package/dist/scanners/dast/zap.js +453 -0
- package/dist/scanners/dast/zap.js.map +1 -0
- package/dist/scanners/deploy/health-checker.d.ts +38 -0
- package/dist/scanners/deploy/health-checker.d.ts.map +1 -0
- package/dist/scanners/deploy/health-checker.js +272 -0
- package/dist/scanners/deploy/health-checker.js.map +1 -0
- package/dist/scanners/deploy/index.d.ts +44 -0
- package/dist/scanners/deploy/index.d.ts.map +1 -0
- package/dist/scanners/deploy/index.js +208 -0
- package/dist/scanners/deploy/index.js.map +1 -0
- package/dist/scanners/deploy/provider-detector.d.ts +25 -0
- package/dist/scanners/deploy/provider-detector.d.ts.map +1 -0
- package/dist/scanners/deploy/provider-detector.js +177 -0
- package/dist/scanners/deploy/provider-detector.js.map +1 -0
- package/dist/scanners/deploy/types.d.ts +406 -0
- package/dist/scanners/deploy/types.d.ts.map +1 -0
- package/dist/scanners/deploy/types.js +58 -0
- package/dist/scanners/deploy/types.js.map +1 -0
- package/dist/scanners/deploy/vercel-integration.d.ts +52 -0
- package/dist/scanners/deploy/vercel-integration.d.ts.map +1 -0
- package/dist/scanners/deploy/vercel-integration.js +280 -0
- package/dist/scanners/deploy/vercel-integration.js.map +1 -0
- package/dist/scanners/fp-feedback.d.ts +140 -0
- package/dist/scanners/fp-feedback.d.ts.map +1 -0
- package/dist/scanners/fp-feedback.js +292 -0
- package/dist/scanners/fp-feedback.js.map +1 -0
- package/dist/scanners/fp-filter.d.ts +94 -0
- package/dist/scanners/fp-filter.d.ts.map +1 -0
- package/dist/scanners/fp-filter.js +397 -0
- package/dist/scanners/fp-filter.js.map +1 -0
- package/dist/scanners/fp-tracker.d.ts +125 -0
- package/dist/scanners/fp-tracker.d.ts.map +1 -0
- package/dist/scanners/fp-tracker.js +330 -0
- package/dist/scanners/fp-tracker.js.map +1 -0
- package/dist/scanners/index.d.ts.map +1 -1
- package/dist/scanners/index.js +56 -0
- package/dist/scanners/index.js.map +1 -1
- package/dist/scanners/index.test.js +6 -6
- package/dist/scanners/index.test.js.map +1 -1
- package/dist/scanners/logic/auth-flow-analyzer.d.ts +18 -0
- package/dist/scanners/logic/auth-flow-analyzer.d.ts.map +1 -0
- package/dist/scanners/logic/auth-flow-analyzer.js +384 -0
- package/dist/scanners/logic/auth-flow-analyzer.js.map +1 -0
- package/dist/scanners/logic/endpoint-analyzer.d.ts +29 -0
- package/dist/scanners/logic/endpoint-analyzer.d.ts.map +1 -0
- package/dist/scanners/logic/endpoint-analyzer.js +528 -0
- package/dist/scanners/logic/endpoint-analyzer.js.map +1 -0
- package/dist/scanners/logic/index.d.ts +41 -0
- package/dist/scanners/logic/index.d.ts.map +1 -0
- package/dist/scanners/logic/index.js +268 -0
- package/dist/scanners/logic/index.js.map +1 -0
- package/dist/scanners/logic/types.d.ts +254 -0
- package/dist/scanners/logic/types.d.ts.map +1 -0
- package/dist/scanners/logic/types.js +142 -0
- package/dist/scanners/logic/types.js.map +1 -0
- package/dist/scanners/runtime/app-launcher.d.ts +33 -0
- package/dist/scanners/runtime/app-launcher.d.ts.map +1 -0
- package/dist/scanners/runtime/app-launcher.js +419 -0
- package/dist/scanners/runtime/app-launcher.js.map +1 -0
- package/dist/scanners/runtime/golden-path-runner.d.ts +48 -0
- package/dist/scanners/runtime/golden-path-runner.d.ts.map +1 -0
- package/dist/scanners/runtime/golden-path-runner.js +373 -0
- package/dist/scanners/runtime/golden-path-runner.js.map +1 -0
- package/dist/scanners/runtime/index.d.ts +41 -0
- package/dist/scanners/runtime/index.d.ts.map +1 -0
- package/dist/scanners/runtime/index.js +164 -0
- package/dist/scanners/runtime/index.js.map +1 -0
- package/dist/scanners/runtime/playwright-executor.d.ts +50 -0
- package/dist/scanners/runtime/playwright-executor.d.ts.map +1 -0
- package/dist/scanners/runtime/playwright-executor.js +387 -0
- package/dist/scanners/runtime/playwright-executor.js.map +1 -0
- package/dist/scanners/runtime/types.d.ts +215 -0
- package/dist/scanners/runtime/types.d.ts.map +1 -0
- package/dist/scanners/runtime/types.js +40 -0
- package/dist/scanners/runtime/types.js.map +1 -0
- package/dist/scanners/scale/bottleneck-detector.d.ts +17 -0
- package/dist/scanners/scale/bottleneck-detector.d.ts.map +1 -0
- package/dist/scanners/scale/bottleneck-detector.js +250 -0
- package/dist/scanners/scale/bottleneck-detector.js.map +1 -0
- package/dist/scanners/scale/capacity-estimator.d.ts +17 -0
- package/dist/scanners/scale/capacity-estimator.d.ts.map +1 -0
- package/dist/scanners/scale/capacity-estimator.js +197 -0
- package/dist/scanners/scale/capacity-estimator.js.map +1 -0
- package/dist/scanners/scale/index.d.ts +37 -0
- package/dist/scanners/scale/index.d.ts.map +1 -0
- package/dist/scanners/scale/index.js +101 -0
- package/dist/scanners/scale/index.js.map +1 -0
- package/dist/scanners/scale/load-profiler.d.ts +48 -0
- package/dist/scanners/scale/load-profiler.d.ts.map +1 -0
- package/dist/scanners/scale/load-profiler.js +377 -0
- package/dist/scanners/scale/load-profiler.js.map +1 -0
- package/dist/scanners/scale/types.d.ts +529 -0
- package/dist/scanners/scale/types.d.ts.map +1 -0
- package/dist/scanners/scale/types.js +57 -0
- package/dist/scanners/scale/types.js.map +1 -0
- package/dist/scanners/secrets.d.ts.map +1 -1
- package/dist/scanners/secrets.js +13 -2
- package/dist/scanners/secrets.js.map +1 -1
- package/dist/scanners/types.d.ts +1 -1
- package/dist/scanners/types.d.ts.map +1 -1
- package/dist/scanners/types.js +4 -0
- package/dist/scanners/types.js.map +1 -1
- package/package.json +8 -4
- package/skills/vaspera-add-tests/SKILL.md +102 -0
- package/skills/vaspera-ai-verify/SKILL.md +166 -0
- package/skills/vaspera-audit/SKILL.md +67 -0
- package/skills/vaspera-certify/SKILL.md +130 -0
- package/skills/vaspera-deploy/SKILL.md +152 -0
- package/skills/vaspera-fix-critical/SKILL.md +52 -0
- package/skills/vaspera-fix-high/SKILL.md +81 -0
- package/skills/vaspera-fix-medium/SKILL.md +56 -0
- package/skills/vaspera-fix-rls/SKILL.md +85 -0
- package/skills/vaspera-harden/SKILL.md +102 -0
- package/skills/vaspera-help/SKILL.md +61 -0
- package/skills/vaspera-load-test/SKILL.md +167 -0
- package/skills/vaspera-verify/SKILL.md +70 -0
- package/skills/vaspera-verify-e2e/SKILL.md +117 -0
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Vercel Integration
|
|
3
|
+
*
|
|
4
|
+
* Integration with Vercel for deployment verification.
|
|
5
|
+
* Uses the Vercel API for deployment status, promotion, and rollback.
|
|
6
|
+
*
|
|
7
|
+
* @module scanners/deploy/vercel-integration
|
|
8
|
+
*/
|
|
9
|
+
import type { DeploymentInfo, CanaryAnalysis } from "./types.js";
|
|
10
|
+
/**
|
|
11
|
+
* Check if Vercel integration is available
|
|
12
|
+
*/
|
|
13
|
+
export declare function isVercelAvailable(): boolean;
|
|
14
|
+
/**
|
|
15
|
+
* Get latest deployment for a project
|
|
16
|
+
*/
|
|
17
|
+
export declare function getLatestDeployment(projectId: string, options?: {
|
|
18
|
+
target?: string;
|
|
19
|
+
}): Promise<DeploymentInfo | null>;
|
|
20
|
+
/**
|
|
21
|
+
* Get deployment by ID
|
|
22
|
+
*/
|
|
23
|
+
export declare function getDeployment(deploymentId: string): Promise<DeploymentInfo | null>;
|
|
24
|
+
/**
|
|
25
|
+
* List recent deployments
|
|
26
|
+
*/
|
|
27
|
+
export declare function listDeployments(projectId: string, options?: {
|
|
28
|
+
limit?: number;
|
|
29
|
+
target?: string;
|
|
30
|
+
}): Promise<DeploymentInfo[]>;
|
|
31
|
+
/**
|
|
32
|
+
* Promote a preview deployment to production
|
|
33
|
+
*/
|
|
34
|
+
export declare function promoteToProduction(deploymentId: string, projectId: string): Promise<{
|
|
35
|
+
success: boolean;
|
|
36
|
+
error?: string;
|
|
37
|
+
}>;
|
|
38
|
+
/**
|
|
39
|
+
* Rollback to a previous deployment
|
|
40
|
+
*/
|
|
41
|
+
export declare function rollback(projectId: string, deploymentId: string): Promise<{
|
|
42
|
+
success: boolean;
|
|
43
|
+
error?: string;
|
|
44
|
+
}>;
|
|
45
|
+
/**
|
|
46
|
+
* Analyze canary deployment (simulated - would use Vercel Edge Config in production)
|
|
47
|
+
*/
|
|
48
|
+
export declare function analyzeCanary(deploymentUrl: string, options?: {
|
|
49
|
+
duration?: number;
|
|
50
|
+
checkInterval?: number;
|
|
51
|
+
}): Promise<CanaryAnalysis>;
|
|
52
|
+
//# sourceMappingURL=vercel-integration.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"vercel-integration.d.ts","sourceRoot":"","sources":["../../../src/scanners/deploy/vercel-integration.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAGH,OAAO,KAAK,EACV,cAAc,EAGd,cAAc,EACf,MAAM,YAAY,CAAC;AAWpB;;GAEG;AACH,wBAAgB,iBAAiB,IAAI,OAAO,CAE3C;AAkED;;GAEG;AACH,wBAAsB,mBAAmB,CACvC,SAAS,EAAE,MAAM,EACjB,OAAO,GAAE;IAAE,MAAM,CAAC,EAAE,MAAM,CAAA;CAAO,GAChC,OAAO,CAAC,cAAc,GAAG,IAAI,CAAC,CA8ChC;AAED;;GAEG;AACH,wBAAsB,aAAa,CAAC,YAAY,EAAE,MAAM,GAAG,OAAO,CAAC,cAAc,GAAG,IAAI,CAAC,CA8BxF;AAED;;GAEG;AACH,wBAAsB,eAAe,CACnC,SAAS,EAAE,MAAM,EACjB,OAAO,GAAE;IAAE,KAAK,CAAC,EAAE,MAAM,CAAC;IAAC,MAAM,CAAC,EAAE,MAAM,CAAA;CAAO,GAChD,OAAO,CAAC,cAAc,EAAE,CAAC,CAkC3B;AAED;;GAEG;AACH,wBAAsB,mBAAmB,CACvC,YAAY,EAAE,MAAM,EACpB,SAAS,EAAE,MAAM,GAChB,OAAO,CAAC;IAAE,OAAO,EAAE,OAAO,CAAC;IAAC,KAAK,CAAC,EAAE,MAAM,CAAA;CAAE,CAAC,CAuB/C;AAED;;GAEG;AACH,wBAAsB,QAAQ,CAC5B,SAAS,EAAE,MAAM,EACjB,YAAY,EAAE,MAAM,GACnB,OAAO,CAAC;IAAE,OAAO,EAAE,OAAO,CAAC;IAAC,KAAK,CAAC,EAAE,MAAM,CAAA;CAAE,CAAC,CAG/C;AAED;;GAEG;AACH,wBAAsB,aAAa,CACjC,aAAa,EAAE,MAAM,EACrB,OAAO,GAAE;IACP,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,aAAa,CAAC,EAAE,MAAM,CAAC;CACnB,GACL,OAAO,CAAC,cAAc,CAAC,CA2DzB"}
|
|
@@ -0,0 +1,280 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Vercel Integration
|
|
3
|
+
*
|
|
4
|
+
* Integration with Vercel for deployment verification.
|
|
5
|
+
* Uses the Vercel API for deployment status, promotion, and rollback.
|
|
6
|
+
*
|
|
7
|
+
* @module scanners/deploy/vercel-integration
|
|
8
|
+
*/
|
|
9
|
+
import { logger } from "../../logger.js";
|
|
10
|
+
const VERCEL_API_BASE = "https://api.vercel.com";
|
|
11
|
+
/**
|
|
12
|
+
* Get Vercel API token from environment
|
|
13
|
+
*/
|
|
14
|
+
function getVercelToken() {
|
|
15
|
+
return process.env.VERCEL_TOKEN || process.env.VERCEL_ACCESS_TOKEN;
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* Check if Vercel integration is available
|
|
19
|
+
*/
|
|
20
|
+
export function isVercelAvailable() {
|
|
21
|
+
return !!getVercelToken();
|
|
22
|
+
}
|
|
23
|
+
/**
|
|
24
|
+
* Make authenticated request to Vercel API
|
|
25
|
+
*/
|
|
26
|
+
async function vercelFetch(endpoint, options = {}) {
|
|
27
|
+
const token = getVercelToken();
|
|
28
|
+
if (!token) {
|
|
29
|
+
throw new Error("VERCEL_TOKEN not configured");
|
|
30
|
+
}
|
|
31
|
+
const url = `${VERCEL_API_BASE}${endpoint}`;
|
|
32
|
+
const response = await fetch(url, {
|
|
33
|
+
...options,
|
|
34
|
+
headers: {
|
|
35
|
+
Authorization: `Bearer ${token}`,
|
|
36
|
+
"Content-Type": "application/json",
|
|
37
|
+
...options.headers,
|
|
38
|
+
},
|
|
39
|
+
});
|
|
40
|
+
return response;
|
|
41
|
+
}
|
|
42
|
+
/**
|
|
43
|
+
* Map Vercel status to our status
|
|
44
|
+
*/
|
|
45
|
+
function mapVercelStatus(state) {
|
|
46
|
+
switch (state) {
|
|
47
|
+
case "QUEUED":
|
|
48
|
+
case "INITIALIZING":
|
|
49
|
+
return "pending";
|
|
50
|
+
case "BUILDING":
|
|
51
|
+
return "building";
|
|
52
|
+
case "DEPLOYING":
|
|
53
|
+
return "deploying";
|
|
54
|
+
case "READY":
|
|
55
|
+
return "ready";
|
|
56
|
+
case "ERROR":
|
|
57
|
+
return "failed";
|
|
58
|
+
case "CANCELED":
|
|
59
|
+
return "canceled";
|
|
60
|
+
default:
|
|
61
|
+
return "pending";
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* Map Vercel target to environment
|
|
66
|
+
*/
|
|
67
|
+
function mapVercelTarget(target) {
|
|
68
|
+
switch (target) {
|
|
69
|
+
case "production":
|
|
70
|
+
return "production";
|
|
71
|
+
case "preview":
|
|
72
|
+
return "preview";
|
|
73
|
+
case "development":
|
|
74
|
+
return "development";
|
|
75
|
+
default:
|
|
76
|
+
return "preview";
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* Get latest deployment for a project
|
|
81
|
+
*/
|
|
82
|
+
export async function getLatestDeployment(projectId, options = {}) {
|
|
83
|
+
try {
|
|
84
|
+
const params = new URLSearchParams({
|
|
85
|
+
projectId,
|
|
86
|
+
limit: "1",
|
|
87
|
+
...(options.target && { target: options.target }),
|
|
88
|
+
});
|
|
89
|
+
const response = await vercelFetch(`/v6/deployments?${params}`);
|
|
90
|
+
if (!response.ok) {
|
|
91
|
+
const error = await response.json();
|
|
92
|
+
logger.error("deploy.vercel_api_error", {
|
|
93
|
+
status: response.status,
|
|
94
|
+
error: error.error?.message,
|
|
95
|
+
});
|
|
96
|
+
return null;
|
|
97
|
+
}
|
|
98
|
+
const data = await response.json();
|
|
99
|
+
const deployment = data.deployments?.[0];
|
|
100
|
+
if (!deployment)
|
|
101
|
+
return null;
|
|
102
|
+
return {
|
|
103
|
+
id: deployment.uid,
|
|
104
|
+
provider: "vercel",
|
|
105
|
+
environment: mapVercelTarget(deployment.target),
|
|
106
|
+
status: mapVercelStatus(deployment.state),
|
|
107
|
+
url: `https://${deployment.url}`,
|
|
108
|
+
createdAt: new Date(deployment.created).toISOString(),
|
|
109
|
+
readyAt: deployment.ready ? new Date(deployment.ready).toISOString() : undefined,
|
|
110
|
+
meta: {
|
|
111
|
+
commit: deployment.meta?.githubCommitSha,
|
|
112
|
+
branch: deployment.meta?.githubCommitRef,
|
|
113
|
+
buildTime: deployment.buildingAt && deployment.ready
|
|
114
|
+
? deployment.ready - deployment.buildingAt
|
|
115
|
+
: undefined,
|
|
116
|
+
},
|
|
117
|
+
};
|
|
118
|
+
}
|
|
119
|
+
catch (error) {
|
|
120
|
+
logger.error("deploy.vercel_get_deployment_error", {
|
|
121
|
+
error: error.message,
|
|
122
|
+
});
|
|
123
|
+
return null;
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
/**
|
|
127
|
+
* Get deployment by ID
|
|
128
|
+
*/
|
|
129
|
+
export async function getDeployment(deploymentId) {
|
|
130
|
+
try {
|
|
131
|
+
const response = await vercelFetch(`/v13/deployments/${deploymentId}`);
|
|
132
|
+
if (!response.ok) {
|
|
133
|
+
return null;
|
|
134
|
+
}
|
|
135
|
+
const deployment = await response.json();
|
|
136
|
+
return {
|
|
137
|
+
id: deployment.id,
|
|
138
|
+
provider: "vercel",
|
|
139
|
+
environment: mapVercelTarget(deployment.target),
|
|
140
|
+
status: mapVercelStatus(deployment.readyState),
|
|
141
|
+
url: `https://${deployment.url}`,
|
|
142
|
+
createdAt: new Date(deployment.createdAt).toISOString(),
|
|
143
|
+
readyAt: deployment.ready ? new Date(deployment.ready).toISOString() : undefined,
|
|
144
|
+
meta: {
|
|
145
|
+
commit: deployment.meta?.githubCommitSha,
|
|
146
|
+
branch: deployment.meta?.githubCommitRef,
|
|
147
|
+
},
|
|
148
|
+
};
|
|
149
|
+
}
|
|
150
|
+
catch (error) {
|
|
151
|
+
logger.error("deploy.vercel_get_deployment_error", {
|
|
152
|
+
deploymentId,
|
|
153
|
+
error: error.message,
|
|
154
|
+
});
|
|
155
|
+
return null;
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
/**
|
|
159
|
+
* List recent deployments
|
|
160
|
+
*/
|
|
161
|
+
export async function listDeployments(projectId, options = {}) {
|
|
162
|
+
try {
|
|
163
|
+
const params = new URLSearchParams({
|
|
164
|
+
projectId,
|
|
165
|
+
limit: String(options.limit || 10),
|
|
166
|
+
...(options.target && { target: options.target }),
|
|
167
|
+
});
|
|
168
|
+
const response = await vercelFetch(`/v6/deployments?${params}`);
|
|
169
|
+
if (!response.ok) {
|
|
170
|
+
return [];
|
|
171
|
+
}
|
|
172
|
+
const data = await response.json();
|
|
173
|
+
return (data.deployments || []).map((deployment) => ({
|
|
174
|
+
id: deployment.uid,
|
|
175
|
+
provider: "vercel",
|
|
176
|
+
environment: mapVercelTarget(deployment.target),
|
|
177
|
+
status: mapVercelStatus(deployment.state),
|
|
178
|
+
url: `https://${deployment.url}`,
|
|
179
|
+
createdAt: new Date(deployment.created).toISOString(),
|
|
180
|
+
meta: {
|
|
181
|
+
commit: deployment.meta?.githubCommitSha,
|
|
182
|
+
branch: deployment.meta?.githubCommitRef,
|
|
183
|
+
},
|
|
184
|
+
}));
|
|
185
|
+
}
|
|
186
|
+
catch (error) {
|
|
187
|
+
logger.error("deploy.vercel_list_deployments_error", {
|
|
188
|
+
error: error.message,
|
|
189
|
+
});
|
|
190
|
+
return [];
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
194
|
+
* Promote a preview deployment to production
|
|
195
|
+
*/
|
|
196
|
+
export async function promoteToProduction(deploymentId, projectId) {
|
|
197
|
+
try {
|
|
198
|
+
const response = await vercelFetch(`/v10/projects/${projectId}/promote/${deploymentId}`, {
|
|
199
|
+
method: "POST",
|
|
200
|
+
});
|
|
201
|
+
if (!response.ok) {
|
|
202
|
+
const error = await response.json();
|
|
203
|
+
return {
|
|
204
|
+
success: false,
|
|
205
|
+
error: error.error?.message || "Promotion failed",
|
|
206
|
+
};
|
|
207
|
+
}
|
|
208
|
+
logger.info("deploy.vercel_promoted", { deploymentId, projectId });
|
|
209
|
+
return { success: true };
|
|
210
|
+
}
|
|
211
|
+
catch (error) {
|
|
212
|
+
return {
|
|
213
|
+
success: false,
|
|
214
|
+
error: error.message,
|
|
215
|
+
};
|
|
216
|
+
}
|
|
217
|
+
}
|
|
218
|
+
/**
|
|
219
|
+
* Rollback to a previous deployment
|
|
220
|
+
*/
|
|
221
|
+
export async function rollback(projectId, deploymentId) {
|
|
222
|
+
// Vercel rollback is essentially promoting an older deployment
|
|
223
|
+
return promoteToProduction(deploymentId, projectId);
|
|
224
|
+
}
|
|
225
|
+
/**
|
|
226
|
+
* Analyze canary deployment (simulated - would use Vercel Edge Config in production)
|
|
227
|
+
*/
|
|
228
|
+
export async function analyzeCanary(deploymentUrl, options = {}) {
|
|
229
|
+
const { duration = 60000, checkInterval = 5000 } = options;
|
|
230
|
+
const startTime = Date.now();
|
|
231
|
+
const checks = [];
|
|
232
|
+
logger.info("deploy.canary_analysis_started", { deploymentUrl, duration });
|
|
233
|
+
// Run health checks for the duration
|
|
234
|
+
while (Date.now() - startTime < duration) {
|
|
235
|
+
try {
|
|
236
|
+
const checkStart = Date.now();
|
|
237
|
+
const response = await fetch(deploymentUrl, {
|
|
238
|
+
signal: AbortSignal.timeout(5000),
|
|
239
|
+
});
|
|
240
|
+
const latency = Date.now() - checkStart;
|
|
241
|
+
checks.push({
|
|
242
|
+
success: response.ok,
|
|
243
|
+
latency,
|
|
244
|
+
});
|
|
245
|
+
}
|
|
246
|
+
catch {
|
|
247
|
+
checks.push({ success: false, latency: 5000 });
|
|
248
|
+
}
|
|
249
|
+
await new Promise((resolve) => setTimeout(resolve, checkInterval));
|
|
250
|
+
}
|
|
251
|
+
// Calculate metrics
|
|
252
|
+
const successfulChecks = checks.filter((c) => c.success);
|
|
253
|
+
const errorRate = 1 - (successfulChecks.length / checks.length);
|
|
254
|
+
const latencies = successfulChecks.map((c) => c.latency).sort((a, b) => a - b);
|
|
255
|
+
const p95Index = Math.floor(latencies.length * 0.95);
|
|
256
|
+
const p95Latency = latencies[p95Index] || 0;
|
|
257
|
+
const thresholdsPassed = errorRate < 0.01 && p95Latency < 500;
|
|
258
|
+
const analysis = {
|
|
259
|
+
phase: thresholdsPassed ? "passed" : "failed",
|
|
260
|
+
trafficPercent: 10, // Simulated
|
|
261
|
+
duration: Date.now() - startTime,
|
|
262
|
+
metrics: {
|
|
263
|
+
errorRate,
|
|
264
|
+
p95Latency,
|
|
265
|
+
requestCount: checks.length,
|
|
266
|
+
healthChecksPassed: successfulChecks.length,
|
|
267
|
+
healthChecksFailed: checks.length - successfulChecks.length,
|
|
268
|
+
},
|
|
269
|
+
thresholdsPassed,
|
|
270
|
+
recommendation: thresholdsPassed ? "promote" : "rollback",
|
|
271
|
+
};
|
|
272
|
+
logger.info("deploy.canary_analysis_completed", {
|
|
273
|
+
phase: analysis.phase,
|
|
274
|
+
errorRate,
|
|
275
|
+
p95Latency,
|
|
276
|
+
recommendation: analysis.recommendation,
|
|
277
|
+
});
|
|
278
|
+
return analysis;
|
|
279
|
+
}
|
|
280
|
+
//# sourceMappingURL=vercel-integration.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"vercel-integration.js","sourceRoot":"","sources":["../../../src/scanners/deploy/vercel-integration.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,140 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* False Positive Feedback
|
|
3
|
+
*
|
|
4
|
+
* Handles user feedback on findings and maintains a feedback database
|
|
5
|
+
* for improving FP detection over time.
|
|
6
|
+
*
|
|
7
|
+
* @module scanners/fp-feedback
|
|
8
|
+
*/
|
|
9
|
+
import type { ScannerType, DeterministicFinding } from "./types.js";
|
|
10
|
+
import type { Severity } from "../certification/types.js";
|
|
11
|
+
/**
|
|
12
|
+
* Reason for marking a finding as FP
|
|
13
|
+
*/
|
|
14
|
+
export type FPReason = "test-code" | "false-pattern-match" | "sanitized-elsewhere" | "intentional" | "vendor-code" | "generated-code" | "example-code" | "configuration" | "other";
|
|
15
|
+
/**
|
|
16
|
+
* FP reason descriptions for UI
|
|
17
|
+
*/
|
|
18
|
+
export declare const FP_REASON_DESCRIPTIONS: Record<FPReason, string>;
|
|
19
|
+
/**
|
|
20
|
+
* A single feedback entry
|
|
21
|
+
*/
|
|
22
|
+
export interface FeedbackEntry {
|
|
23
|
+
/** Unique feedback ID */
|
|
24
|
+
id: string;
|
|
25
|
+
/** Finding ID this feedback relates to */
|
|
26
|
+
findingId: string;
|
|
27
|
+
/** Scanner that generated the finding */
|
|
28
|
+
scanner: ScannerType;
|
|
29
|
+
/** Rule ID */
|
|
30
|
+
ruleId: string;
|
|
31
|
+
/** File path */
|
|
32
|
+
file: string;
|
|
33
|
+
/** Line number */
|
|
34
|
+
line?: number;
|
|
35
|
+
/** Original severity */
|
|
36
|
+
severity: Severity;
|
|
37
|
+
/** Verdict: true positive or false positive */
|
|
38
|
+
verdict: "tp" | "fp";
|
|
39
|
+
/** Reason for FP (if verdict is FP) */
|
|
40
|
+
reason?: FPReason;
|
|
41
|
+
/** Additional details/notes */
|
|
42
|
+
details?: string;
|
|
43
|
+
/** Who submitted the feedback */
|
|
44
|
+
submittedBy?: string;
|
|
45
|
+
/** When the feedback was submitted */
|
|
46
|
+
submittedAt: string;
|
|
47
|
+
/** Hash of the code at feedback time (for invalidation) */
|
|
48
|
+
codeHash?: string;
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* Feedback database
|
|
52
|
+
*/
|
|
53
|
+
export interface FeedbackDatabase {
|
|
54
|
+
version: string;
|
|
55
|
+
projectPath: string;
|
|
56
|
+
entries: FeedbackEntry[];
|
|
57
|
+
stats: {
|
|
58
|
+
totalFeedback: number;
|
|
59
|
+
tpCount: number;
|
|
60
|
+
fpCount: number;
|
|
61
|
+
byScanner: Record<string, {
|
|
62
|
+
tp: number;
|
|
63
|
+
fp: number;
|
|
64
|
+
}>;
|
|
65
|
+
byReason: Record<string, number>;
|
|
66
|
+
};
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Load feedback database
|
|
70
|
+
*/
|
|
71
|
+
export declare function loadFeedbackDatabase(projectPath: string): Promise<FeedbackDatabase>;
|
|
72
|
+
/**
|
|
73
|
+
* Save feedback database
|
|
74
|
+
*/
|
|
75
|
+
export declare function saveFeedbackDatabase(projectPath: string, db: FeedbackDatabase): Promise<void>;
|
|
76
|
+
/**
|
|
77
|
+
* Submit feedback for a finding
|
|
78
|
+
*/
|
|
79
|
+
export declare function submitFeedback(projectPath: string, finding: DeterministicFinding, verdict: "tp" | "fp", options?: {
|
|
80
|
+
reason?: FPReason;
|
|
81
|
+
details?: string;
|
|
82
|
+
submittedBy?: string;
|
|
83
|
+
codeHash?: string;
|
|
84
|
+
}): Promise<FeedbackEntry>;
|
|
85
|
+
/**
|
|
86
|
+
* Get feedback for a specific finding
|
|
87
|
+
*/
|
|
88
|
+
export declare function getFeedbackForFinding(projectPath: string, scanner: ScannerType, ruleId: string, file: string, line?: number): Promise<FeedbackEntry[]>;
|
|
89
|
+
/**
|
|
90
|
+
* Get all feedback for a rule
|
|
91
|
+
*/
|
|
92
|
+
export declare function getFeedbackForRule(projectPath: string, scanner: ScannerType, ruleId: string): Promise<FeedbackEntry[]>;
|
|
93
|
+
/**
|
|
94
|
+
* Check if a finding has existing feedback
|
|
95
|
+
*/
|
|
96
|
+
export declare function hasFeedback(projectPath: string, scanner: ScannerType, ruleId: string, file: string, line?: number): Promise<boolean>;
|
|
97
|
+
/**
|
|
98
|
+
* Get suppression suggestions based on feedback
|
|
99
|
+
*/
|
|
100
|
+
export declare function getSuppressionSuggestions(projectPath: string, options?: {
|
|
101
|
+
minFPRate?: number;
|
|
102
|
+
minSampleSize?: number;
|
|
103
|
+
}): Promise<Array<{
|
|
104
|
+
scanner: ScannerType;
|
|
105
|
+
ruleId: string;
|
|
106
|
+
fpRate: number;
|
|
107
|
+
sampleSize: number;
|
|
108
|
+
suggestion: "disable" | "review" | "keep";
|
|
109
|
+
commonReasons: FPReason[];
|
|
110
|
+
}>>;
|
|
111
|
+
/**
|
|
112
|
+
* Generate feedback summary report
|
|
113
|
+
*/
|
|
114
|
+
export declare function generateFeedbackReport(projectPath: string): Promise<{
|
|
115
|
+
overview: {
|
|
116
|
+
totalFeedback: number;
|
|
117
|
+
tpCount: number;
|
|
118
|
+
fpCount: number;
|
|
119
|
+
overallFPRate: number;
|
|
120
|
+
};
|
|
121
|
+
byScanner: Array<{
|
|
122
|
+
scanner: ScannerType;
|
|
123
|
+
total: number;
|
|
124
|
+
tp: number;
|
|
125
|
+
fp: number;
|
|
126
|
+
fpRate: number;
|
|
127
|
+
}>;
|
|
128
|
+
topFPReasons: Array<{
|
|
129
|
+
reason: FPReason;
|
|
130
|
+
count: number;
|
|
131
|
+
percentage: number;
|
|
132
|
+
}>;
|
|
133
|
+
recentFeedback: FeedbackEntry[];
|
|
134
|
+
suppressionSuggestions: Awaited<ReturnType<typeof getSuppressionSuggestions>>;
|
|
135
|
+
}>;
|
|
136
|
+
/**
|
|
137
|
+
* Clear old feedback entries
|
|
138
|
+
*/
|
|
139
|
+
export declare function pruneOldFeedback(projectPath: string, maxAgeDays?: number): Promise<number>;
|
|
140
|
+
//# sourceMappingURL=fp-feedback.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fp-feedback.d.ts","sourceRoot":"","sources":["../../src/scanners/fp-feedback.ts"],"names":[],"mappings":"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"}
|