qa360 2.2.1 → 2.2.13
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 +28 -0
- package/README.md +79 -0
- package/cli/dist/cli-minimal.d.ts +6 -0
- package/cli/dist/cli-minimal.js +36 -0
- package/cli/dist/commands/ai.d.ts +43 -0
- package/cli/dist/commands/ai.js +616 -0
- package/cli/dist/commands/ask.d.ts +94 -0
- package/cli/dist/commands/ask.js +582 -0
- package/cli/dist/commands/coverage.d.ts +8 -0
- package/cli/dist/commands/coverage.js +252 -0
- package/cli/dist/commands/crawl.d.ts +24 -0
- package/cli/dist/commands/crawl.js +121 -0
- package/cli/dist/commands/doctor.d.ts +54 -0
- package/cli/dist/commands/doctor.js +513 -0
- package/cli/dist/commands/examples.d.ts +33 -0
- package/cli/dist/commands/examples.js +193 -0
- package/cli/dist/commands/explain.d.ts +27 -0
- package/cli/dist/commands/explain.js +630 -0
- package/cli/dist/commands/flakiness.d.ts +73 -0
- package/cli/dist/commands/flakiness.js +435 -0
- package/cli/dist/commands/generate.d.ts +66 -0
- package/cli/dist/commands/generate.js +438 -0
- package/cli/dist/commands/history.d.ts +76 -0
- package/cli/dist/commands/history.js +755 -0
- package/cli/dist/commands/init.d.ts +106 -0
- package/cli/dist/commands/init.js +616 -0
- package/cli/dist/commands/monitor.d.ts +27 -0
- package/cli/dist/commands/monitor.js +225 -0
- package/cli/dist/commands/ollama.d.ts +40 -0
- package/cli/dist/commands/ollama.js +346 -0
- package/cli/dist/commands/pack.d.ts +70 -0
- package/cli/dist/commands/pack.js +413 -0
- package/cli/dist/commands/regression.d.ts +8 -0
- package/cli/dist/commands/regression.js +340 -0
- package/cli/dist/commands/repair.d.ts +26 -0
- package/cli/dist/commands/repair.js +307 -0
- package/cli/dist/commands/report.d.ts +62 -0
- package/cli/dist/commands/report.js +378 -0
- package/cli/dist/commands/retry.d.ts +43 -0
- package/cli/dist/commands/retry.js +275 -0
- package/cli/dist/commands/run.d.ts +41 -0
- package/cli/dist/commands/run.js +171 -0
- package/cli/dist/commands/scan.d.ts +5 -0
- package/cli/dist/commands/scan.js +155 -0
- package/cli/dist/commands/secrets.d.ts +58 -0
- package/cli/dist/commands/secrets.js +289 -0
- package/cli/dist/commands/serve.d.ts +13 -0
- package/cli/dist/commands/serve.js +156 -0
- package/cli/dist/commands/slo.d.ts +8 -0
- package/cli/dist/commands/slo.js +327 -0
- package/cli/dist/commands/verify.d.ts +32 -0
- package/cli/dist/commands/verify.js +278 -0
- package/cli/dist/core/adapters/gitleaks-secrets.d.ts +114 -0
- package/cli/dist/core/adapters/gitleaks-secrets.js +410 -0
- package/cli/dist/core/adapters/jest-adapter.d.ts +44 -0
- package/cli/dist/core/adapters/jest-adapter.js +261 -0
- package/cli/dist/core/adapters/k6-perf.d.ts +85 -0
- package/cli/dist/core/adapters/k6-perf.js +398 -0
- package/cli/dist/core/adapters/osv-deps.d.ts +123 -0
- package/cli/dist/core/adapters/osv-deps.js +372 -0
- package/cli/dist/core/adapters/playwright-native-adapter.d.ts +121 -0
- package/cli/dist/core/adapters/playwright-native-adapter.js +339 -0
- package/cli/dist/core/adapters/playwright-native-api.d.ts +183 -0
- package/cli/dist/core/adapters/playwright-native-api.js +465 -0
- package/cli/dist/core/adapters/playwright-ui.d.ts +197 -0
- package/cli/dist/core/adapters/playwright-ui.js +864 -0
- package/cli/dist/core/adapters/pytest-adapter.d.ts +49 -0
- package/cli/dist/core/adapters/pytest-adapter.js +324 -0
- package/cli/dist/core/adapters/semgrep-sast.d.ts +99 -0
- package/cli/dist/core/adapters/semgrep-sast.js +322 -0
- package/cli/dist/core/adapters/unit-test-types.d.ts +119 -0
- package/cli/dist/core/adapters/unit-test-types.js +6 -0
- package/cli/dist/core/adapters/vitest-adapter.d.ts +46 -0
- package/cli/dist/core/adapters/vitest-adapter.js +272 -0
- package/cli/dist/core/adapters/zap-dast.d.ts +133 -0
- package/cli/dist/core/adapters/zap-dast.js +424 -0
- package/cli/dist/core/ai/anthropic-provider.d.ts +50 -0
- package/cli/dist/core/ai/anthropic-provider.js +223 -0
- package/cli/dist/core/ai/deepseek-provider.d.ts +81 -0
- package/cli/dist/core/ai/deepseek-provider.js +266 -0
- package/cli/dist/core/ai/index.d.ts +60 -0
- package/cli/dist/core/ai/index.js +18 -0
- package/cli/dist/core/ai/llm-client.d.ts +45 -0
- package/cli/dist/core/ai/llm-client.js +7 -0
- package/cli/dist/core/ai/mock-provider.d.ts +49 -0
- package/cli/dist/core/ai/mock-provider.js +121 -0
- package/cli/dist/core/ai/ollama-provider.d.ts +78 -0
- package/cli/dist/core/ai/ollama-provider.js +216 -0
- package/cli/dist/core/ai/openai-provider.d.ts +48 -0
- package/cli/dist/core/ai/openai-provider.js +200 -0
- package/cli/dist/core/ai/provider-factory.d.ts +160 -0
- package/cli/dist/core/ai/provider-factory.js +269 -0
- package/cli/dist/core/artifacts/index.d.ts +6 -0
- package/cli/dist/core/artifacts/index.js +6 -0
- package/cli/dist/core/artifacts/ui-artifacts.d.ts +133 -0
- package/cli/dist/core/artifacts/ui-artifacts.js +304 -0
- package/cli/dist/core/assertions/engine.d.ts +51 -0
- package/cli/dist/core/assertions/engine.js +530 -0
- package/cli/dist/core/assertions/index.d.ts +11 -0
- package/cli/dist/core/assertions/index.js +11 -0
- package/cli/dist/core/assertions/types.d.ts +121 -0
- package/cli/dist/core/assertions/types.js +37 -0
- package/cli/dist/core/auth/api-key-provider.d.ts +16 -0
- package/cli/dist/core/auth/api-key-provider.js +63 -0
- package/cli/dist/core/auth/aws-iam-provider.d.ts +35 -0
- package/cli/dist/core/auth/aws-iam-provider.js +177 -0
- package/cli/dist/core/auth/azure-ad-provider.d.ts +15 -0
- package/cli/dist/core/auth/azure-ad-provider.js +99 -0
- package/cli/dist/core/auth/basic-auth-provider.d.ts +26 -0
- package/cli/dist/core/auth/basic-auth-provider.js +111 -0
- package/cli/dist/core/auth/gcp-adc-provider.d.ts +27 -0
- package/cli/dist/core/auth/gcp-adc-provider.js +126 -0
- package/cli/dist/core/auth/index.d.ts +238 -0
- package/cli/dist/core/auth/index.js +82 -0
- package/cli/dist/core/auth/jwt-provider.d.ts +19 -0
- package/cli/dist/core/auth/jwt-provider.js +160 -0
- package/cli/dist/core/auth/manager.d.ts +84 -0
- package/cli/dist/core/auth/manager.js +230 -0
- package/cli/dist/core/auth/oauth2-provider.d.ts +17 -0
- package/cli/dist/core/auth/oauth2-provider.js +114 -0
- package/cli/dist/core/auth/totp-provider.d.ts +31 -0
- package/cli/dist/core/auth/totp-provider.js +134 -0
- package/cli/dist/core/auth/ui-login-provider.d.ts +26 -0
- package/cli/dist/core/auth/ui-login-provider.js +198 -0
- package/cli/dist/core/cache/index.d.ts +7 -0
- package/cli/dist/core/cache/index.js +6 -0
- package/cli/dist/core/cache/lru-cache.d.ts +203 -0
- package/cli/dist/core/cache/lru-cache.js +397 -0
- package/cli/dist/core/coverage/analyzer.d.ts +101 -0
- package/cli/dist/core/coverage/analyzer.js +415 -0
- package/cli/dist/core/coverage/collector.d.ts +74 -0
- package/cli/dist/core/coverage/collector.js +459 -0
- package/cli/dist/core/coverage/config.d.ts +37 -0
- package/cli/dist/core/coverage/config.js +156 -0
- package/cli/dist/core/coverage/index.d.ts +11 -0
- package/cli/dist/core/coverage/index.js +15 -0
- package/cli/dist/core/coverage/types.d.ts +267 -0
- package/cli/dist/core/coverage/types.js +6 -0
- package/cli/dist/core/coverage/vault.d.ts +95 -0
- package/cli/dist/core/coverage/vault.js +405 -0
- package/cli/dist/core/crawler/index.d.ts +57 -0
- package/cli/dist/core/crawler/index.js +281 -0
- package/cli/dist/core/crawler/journey-generator.d.ts +49 -0
- package/cli/dist/core/crawler/journey-generator.js +412 -0
- package/cli/dist/core/crawler/page-analyzer.d.ts +88 -0
- package/cli/dist/core/crawler/page-analyzer.js +709 -0
- package/cli/dist/core/crawler/selector-generator.d.ts +34 -0
- package/cli/dist/core/crawler/selector-generator.js +309 -0
- package/cli/dist/core/crawler/types.d.ts +353 -0
- package/cli/dist/core/crawler/types.js +6 -0
- package/cli/dist/core/dashboard/assets.d.ts +6 -0
- package/cli/dist/core/dashboard/assets.js +690 -0
- package/cli/dist/core/dashboard/index.d.ts +6 -0
- package/cli/dist/core/dashboard/index.js +5 -0
- package/cli/dist/core/dashboard/server.d.ts +72 -0
- package/cli/dist/core/dashboard/server.js +354 -0
- package/cli/dist/core/dashboard/types.d.ts +70 -0
- package/cli/dist/core/dashboard/types.js +5 -0
- package/cli/dist/core/discoverer/index.d.ts +115 -0
- package/cli/dist/core/discoverer/index.js +250 -0
- package/cli/dist/core/fixtures/index.d.ts +8 -0
- package/cli/dist/core/fixtures/index.js +8 -0
- package/cli/dist/core/fixtures/loader.d.ts +65 -0
- package/cli/dist/core/fixtures/loader.js +161 -0
- package/cli/dist/core/fixtures/resolver.d.ts +79 -0
- package/cli/dist/core/fixtures/resolver.js +181 -0
- package/cli/dist/core/fixtures/types.d.ts +75 -0
- package/cli/dist/core/fixtures/types.js +30 -0
- package/cli/dist/core/flakiness/index.d.ts +228 -0
- package/cli/dist/core/flakiness/index.js +384 -0
- package/cli/dist/core/generation/code-formatter.d.ts +111 -0
- package/cli/dist/core/generation/code-formatter.js +307 -0
- package/cli/dist/core/generation/code-generator.d.ts +144 -0
- package/cli/dist/core/generation/code-generator.js +293 -0
- package/cli/dist/core/generation/crawler-pack-generator.d.ts +44 -0
- package/cli/dist/core/generation/crawler-pack-generator.js +245 -0
- package/cli/dist/core/generation/generator.d.ts +40 -0
- package/cli/dist/core/generation/generator.js +76 -0
- package/cli/dist/core/generation/index.d.ts +32 -0
- package/cli/dist/core/generation/index.js +30 -0
- package/cli/dist/core/generation/pack-generator.d.ts +107 -0
- package/cli/dist/core/generation/pack-generator.js +416 -0
- package/cli/dist/core/generation/prompt-builder.d.ts +132 -0
- package/cli/dist/core/generation/prompt-builder.js +672 -0
- package/cli/dist/core/generation/source-analyzer.d.ts +213 -0
- package/cli/dist/core/generation/source-analyzer.js +657 -0
- package/cli/dist/core/generation/test-optimizer.d.ts +117 -0
- package/cli/dist/core/generation/test-optimizer.js +328 -0
- package/cli/dist/core/generation/types.d.ts +214 -0
- package/cli/dist/core/generation/types.js +4 -0
- package/cli/dist/core/hooks/compose.d.ts +61 -0
- package/cli/dist/core/hooks/compose.js +225 -0
- package/cli/dist/core/hooks/runner.d.ts +68 -0
- package/cli/dist/core/hooks/runner.js +303 -0
- package/cli/dist/core/index.d.ts +110 -0
- package/cli/dist/core/index.js +99 -0
- package/cli/dist/core/pack/migrator.d.ts +51 -0
- package/cli/dist/core/pack/migrator.js +304 -0
- package/cli/dist/core/pack/validator.d.ts +42 -0
- package/cli/dist/core/pack/validator.js +330 -0
- package/cli/dist/core/pack-v2/index.d.ts +9 -0
- package/cli/dist/core/pack-v2/index.js +8 -0
- package/cli/dist/core/pack-v2/loader.d.ts +63 -0
- package/cli/dist/core/pack-v2/loader.js +292 -0
- package/cli/dist/core/pack-v2/migrator.d.ts +62 -0
- package/cli/dist/core/pack-v2/migrator.js +505 -0
- package/cli/dist/core/pack-v2/validator.d.ts +65 -0
- package/cli/dist/core/pack-v2/validator.js +629 -0
- package/cli/dist/core/parallel/index.d.ts +6 -0
- package/cli/dist/core/parallel/index.js +6 -0
- package/cli/dist/core/parallel/parallel-runner.d.ts +107 -0
- package/cli/dist/core/parallel/parallel-runner.js +192 -0
- package/cli/dist/core/pom/base-page.d.ts +237 -0
- package/cli/dist/core/pom/base-page.js +354 -0
- package/cli/dist/core/pom/index.d.ts +22 -0
- package/cli/dist/core/pom/index.js +23 -0
- package/cli/dist/core/pom/loader.d.ts +118 -0
- package/cli/dist/core/pom/loader.js +382 -0
- package/cli/dist/core/pom/types.d.ts +112 -0
- package/cli/dist/core/pom/types.js +9 -0
- package/cli/dist/core/proof/bundle.d.ts +137 -0
- package/cli/dist/core/proof/bundle.js +160 -0
- package/cli/dist/core/proof/canonicalize.d.ts +47 -0
- package/cli/dist/core/proof/canonicalize.js +105 -0
- package/cli/dist/core/proof/index.d.ts +13 -0
- package/cli/dist/core/proof/index.js +18 -0
- package/cli/dist/core/proof/schema.d.ts +217 -0
- package/cli/dist/core/proof/schema.js +263 -0
- package/cli/dist/core/proof/signer.d.ts +111 -0
- package/cli/dist/core/proof/signer.js +226 -0
- package/cli/dist/core/proof/verifier.d.ts +97 -0
- package/cli/dist/core/proof/verifier.js +308 -0
- package/cli/dist/core/regression/detector.d.ts +107 -0
- package/cli/dist/core/regression/detector.js +497 -0
- package/cli/dist/core/regression/index.d.ts +9 -0
- package/cli/dist/core/regression/index.js +11 -0
- package/cli/dist/core/regression/trend-analyzer.d.ts +102 -0
- package/cli/dist/core/regression/trend-analyzer.js +345 -0
- package/cli/dist/core/regression/types.d.ts +222 -0
- package/cli/dist/core/regression/types.js +7 -0
- package/cli/dist/core/regression/vault.d.ts +87 -0
- package/cli/dist/core/regression/vault.js +289 -0
- package/cli/dist/core/repair/engine/fixer.d.ts +24 -0
- package/cli/dist/core/repair/engine/fixer.js +226 -0
- package/cli/dist/core/repair/engine/suggestion-engine.d.ts +18 -0
- package/cli/dist/core/repair/engine/suggestion-engine.js +187 -0
- package/cli/dist/core/repair/index.d.ts +10 -0
- package/cli/dist/core/repair/index.js +13 -0
- package/cli/dist/core/repair/repairer.d.ts +90 -0
- package/cli/dist/core/repair/repairer.js +284 -0
- package/cli/dist/core/repair/types.d.ts +91 -0
- package/cli/dist/core/repair/types.js +6 -0
- package/cli/dist/core/repair/utils/error-analyzer.d.ts +28 -0
- package/cli/dist/core/repair/utils/error-analyzer.js +264 -0
- package/cli/dist/core/reporting/html-reporter.d.ts +119 -0
- package/cli/dist/core/reporting/html-reporter.js +737 -0
- package/cli/dist/core/reporting/index.d.ts +6 -0
- package/cli/dist/core/reporting/index.js +6 -0
- package/cli/dist/core/retry/flakiness-integration.d.ts +60 -0
- package/cli/dist/core/retry/flakiness-integration.js +228 -0
- package/cli/dist/core/retry/index.d.ts +14 -0
- package/cli/dist/core/retry/index.js +16 -0
- package/cli/dist/core/retry/retry-engine.d.ts +80 -0
- package/cli/dist/core/retry/retry-engine.js +296 -0
- package/cli/dist/core/retry/types.d.ts +178 -0
- package/cli/dist/core/retry/types.js +52 -0
- package/cli/dist/core/retry/vault.d.ts +77 -0
- package/cli/dist/core/retry/vault.js +304 -0
- package/cli/dist/core/runner/e2e-helpers.d.ts +102 -0
- package/cli/dist/core/runner/e2e-helpers.js +153 -0
- package/cli/dist/core/runner/phase3-runner.d.ts +249 -0
- package/cli/dist/core/runner/phase3-runner.js +1323 -0
- package/cli/dist/core/schemas/pack.schema.json +236 -0
- package/cli/dist/core/secrets/crypto.d.ts +75 -0
- package/cli/dist/core/secrets/crypto.js +223 -0
- package/cli/dist/core/secrets/manager.d.ts +76 -0
- package/cli/dist/core/secrets/manager.js +219 -0
- package/cli/dist/core/security/redaction-patterns-extended.d.ts +27 -0
- package/cli/dist/core/security/redaction-patterns-extended.js +247 -0
- package/cli/dist/core/security/redactor.d.ts +71 -0
- package/cli/dist/core/security/redactor.js +279 -0
- package/cli/dist/core/self-healing/assertion-healer.d.ts +97 -0
- package/cli/dist/core/self-healing/assertion-healer.js +371 -0
- package/cli/dist/core/self-healing/engine.d.ts +122 -0
- package/cli/dist/core/self-healing/engine.js +538 -0
- package/cli/dist/core/self-healing/index.d.ts +10 -0
- package/cli/dist/core/self-healing/index.js +11 -0
- package/cli/dist/core/self-healing/selector-healer.d.ts +103 -0
- package/cli/dist/core/self-healing/selector-healer.js +372 -0
- package/cli/dist/core/self-healing/types.d.ts +152 -0
- package/cli/dist/core/self-healing/types.js +6 -0
- package/cli/dist/core/serve/diagnostics-collector.d.ts +32 -0
- package/cli/dist/core/serve/diagnostics-collector.js +149 -0
- package/cli/dist/core/serve/health-checker.d.ts +44 -0
- package/cli/dist/core/serve/health-checker.js +219 -0
- package/cli/dist/core/serve/index.d.ts +8 -0
- package/cli/dist/core/serve/index.js +8 -0
- package/cli/dist/core/serve/metrics-collector.d.ts +24 -0
- package/cli/dist/core/serve/metrics-collector.js +322 -0
- package/cli/dist/core/serve/process-manager.d.ts +36 -0
- package/cli/dist/core/serve/process-manager.js +213 -0
- package/cli/dist/core/serve/server.d.ts +36 -0
- package/cli/dist/core/serve/server.js +191 -0
- package/cli/dist/core/slo/config.d.ts +107 -0
- package/cli/dist/core/slo/config.js +360 -0
- package/cli/dist/core/slo/index.d.ts +11 -0
- package/cli/dist/core/slo/index.js +15 -0
- package/cli/dist/core/slo/sli-calculator.d.ts +92 -0
- package/cli/dist/core/slo/sli-calculator.js +364 -0
- package/cli/dist/core/slo/slo-tracker.d.ts +148 -0
- package/cli/dist/core/slo/slo-tracker.js +379 -0
- package/cli/dist/core/slo/types.d.ts +281 -0
- package/cli/dist/core/slo/types.js +7 -0
- package/cli/dist/core/slo/vault.d.ts +102 -0
- package/cli/dist/core/slo/vault.js +427 -0
- package/cli/dist/core/tui/index.d.ts +7 -0
- package/cli/dist/core/tui/index.js +6 -0
- package/cli/dist/core/tui/monitor.d.ts +92 -0
- package/cli/dist/core/tui/monitor.js +271 -0
- package/cli/dist/core/tui/renderer.d.ts +33 -0
- package/cli/dist/core/tui/renderer.js +218 -0
- package/cli/dist/core/tui/types.d.ts +63 -0
- package/cli/dist/core/tui/types.js +5 -0
- package/cli/dist/core/types/pack-v1.d.ts +251 -0
- package/cli/dist/core/types/pack-v1.js +5 -0
- package/cli/dist/core/types/pack-v2.d.ts +456 -0
- package/cli/dist/core/types/pack-v2.js +8 -0
- package/cli/dist/core/types/trust-score.d.ts +69 -0
- package/cli/dist/core/types/trust-score.js +191 -0
- package/cli/dist/core/vault/cas.d.ts +90 -0
- package/cli/dist/core/vault/cas.js +261 -0
- package/cli/dist/core/vault/index.d.ts +326 -0
- package/cli/dist/core/vault/index.js +1042 -0
- package/cli/dist/core/visual/index.d.ts +6 -0
- package/cli/dist/core/visual/index.js +6 -0
- package/cli/dist/core/visual/visual-regression.d.ts +113 -0
- package/cli/dist/core/visual/visual-regression.js +236 -0
- package/cli/dist/core/watch/index.d.ts +7 -0
- package/cli/dist/core/watch/index.js +6 -0
- package/cli/dist/core/watch/watch-mode.d.ts +213 -0
- package/cli/dist/core/watch/watch-mode.js +389 -0
- package/cli/dist/generators/index.d.ts +5 -0
- package/cli/dist/generators/index.js +5 -0
- package/cli/dist/generators/json-reporter.d.ts +10 -0
- package/cli/dist/generators/json-reporter.js +12 -0
- package/cli/dist/generators/test-generator.d.ts +18 -0
- package/cli/dist/generators/test-generator.js +78 -0
- package/cli/dist/index.d.ts +8 -0
- package/cli/dist/index.js +262 -0
- package/cli/dist/scanners/dom-scanner.d.ts +52 -0
- package/cli/dist/scanners/dom-scanner.js +296 -0
- package/cli/dist/scanners/index.d.ts +4 -0
- package/cli/dist/scanners/index.js +4 -0
- package/cli/dist/schemas/pack.schema.json +236 -0
- package/cli/dist/types/scan.d.ts +68 -0
- package/cli/dist/types/scan.js +4 -0
- package/cli/dist/utils/config.d.ts +35 -0
- package/cli/dist/utils/config.js +196 -0
- package/cli/package.json +1 -1
- package/package.json +26 -2
- package/.BETA_TESTING_FEEDBACK.md +0 -256
- package/.claude/settings.local.json +0 -154
- package/.editorconfig +0 -21
- package/.github/CODEOWNERS +0 -23
- package/.github/ISSUE_TEMPLATE/bug_report.yml +0 -108
- package/.github/ISSUE_TEMPLATE/feedback_dx.yml +0 -121
- package/.github/dependabot.yml +0 -35
- package/.github/workflows/mcp-dx.yml +0 -106
- package/.github/workflows/release.yml +0 -26
- package/.github/workflows/test.yml +0 -93
- package/.nvmrc +0 -1
- package/.qa360/vault.db +0 -0
- package/.qa360/vault.db-shm +0 -0
- package/.qa360/vault.db-wal +0 -0
- package/.qa360-artifacts/.gitkeep +0 -0
- package/.qa360-artifacts/baselines/.gitkeep +0 -0
- package/.qa360-artifacts/cache/.gitkeep +0 -0
- package/.qa360-artifacts/reports/.gitkeep +0 -0
- package/.qa360-artifacts/screenshots/.gitkeep +0 -0
- package/.qa360-baselines/www_xyqo_ai.baseline.json +0 -33
- package/CODEOWNERS +0 -43
- package/NOVICE_USER_GUIDE.md +0 -272
- package/adapters/README.md +0 -46
- package/check-branches.sh +0 -32
- package/cli/.qa360/keys/ed25519.key +0 -1
- package/cli/.qa360/keys/ed25519.pub +0 -1
- package/cli/examples/README.md +0 -160
- package/cli/examples/accessibility.yml +0 -48
- package/cli/examples/api-basic.yml +0 -27
- package/cli/examples/complete.yml +0 -146
- package/cli/examples/crawler.yml +0 -38
- package/cli/examples/fullstack.yml +0 -78
- package/cli/examples/security.yml +0 -58
- package/cli/examples/ui-advanced.yml +0 -49
- package/cli/examples/ui-basic.yml +0 -24
- package/cli/scripts/bundle-for-npm.sh +0 -51
- package/cli/scripts/validate-package.js +0 -116
- package/cli/src/__tests__/commands/doctor.test.ts +0 -108
- package/cli/src/__tests__/index.test.ts +0 -15
- package/cli/src/cli-minimal.ts +0 -44
- package/cli/src/commands/__tests__/crawl.test.ts +0 -412
- package/cli/src/commands/__tests__/doctor-qa360-home.test.ts +0 -156
- package/cli/src/commands/__tests__/e2e-ui-tests.test.ts +0 -494
- package/cli/src/commands/__tests__/e2e.test.ts +0 -187
- package/cli/src/commands/__tests__/flakiness.test.ts +0 -528
- package/cli/src/commands/__tests__/generate.test.ts +0 -507
- package/cli/src/commands/__tests__/history.integration.test.ts +0 -358
- package/cli/src/commands/__tests__/history.test.ts +0 -433
- package/cli/src/commands/__tests__/monitor-realworld.test.ts +0 -199
- package/cli/src/commands/__tests__/monitor.test.ts +0 -81
- package/cli/src/commands/__tests__/ollama.test.ts +0 -529
- package/cli/src/commands/__tests__/repair.test.ts +0 -225
- package/cli/src/commands/__tests__/report.integration.test.ts +0 -167
- package/cli/src/commands/__tests__/report.test.ts +0 -294
- package/cli/src/commands/__tests__/report.vitest.ts +0 -288
- package/cli/src/commands/__tests__/retry.test.ts +0 -78
- package/cli/src/commands/__tests__/run.integration.test.ts +0 -240
- package/cli/src/commands/__tests__/run.test.ts +0 -346
- package/cli/src/commands/__tests__/run.vitest.ts +0 -301
- package/cli/src/commands/__tests__/secrets.test.ts +0 -114
- package/cli/src/commands/__tests__/serve.test.ts +0 -80
- package/cli/src/commands/__tests__/verify.test.ts +0 -103
- package/cli/src/commands/ai.ts +0 -702
- package/cli/src/commands/ask.ts +0 -678
- package/cli/src/commands/coverage.ts +0 -305
- package/cli/src/commands/crawl.ts +0 -155
- package/cli/src/commands/doctor.ts +0 -610
- package/cli/src/commands/examples.ts +0 -248
- package/cli/src/commands/explain.ts +0 -710
- package/cli/src/commands/flakiness.ts +0 -560
- package/cli/src/commands/generate.ts +0 -566
- package/cli/src/commands/history.ts +0 -914
- package/cli/src/commands/init.ts +0 -777
- package/cli/src/commands/monitor.ts +0 -270
- package/cli/src/commands/ollama.ts +0 -337
- package/cli/src/commands/pack.ts +0 -497
- package/cli/src/commands/regression.ts +0 -400
- package/cli/src/commands/repair.ts +0 -356
- package/cli/src/commands/report.ts +0 -463
- package/cli/src/commands/retry.ts +0 -380
- package/cli/src/commands/run.ts +0 -220
- package/cli/src/commands/scan.ts +0 -177
- package/cli/src/commands/secrets.ts +0 -340
- package/cli/src/commands/serve.ts +0 -194
- package/cli/src/commands/slo.ts +0 -387
- package/cli/src/commands/verify-temp-note.md +0 -11
- package/cli/src/commands/verify.ts +0 -322
- package/cli/src/generators/index.ts +0 -6
- package/cli/src/generators/json-reporter.ts +0 -15
- package/cli/src/generators/test-generator.ts +0 -90
- package/cli/src/index.ts +0 -289
- package/cli/src/scanners/dom-scanner.ts +0 -360
- package/cli/src/scanners/index.ts +0 -5
- package/cli/src/types/scan.ts +0 -84
- package/cli/src/utils/config.ts +0 -145
- package/cli/tsconfig.bundle.json +0 -12
- package/cli/tsconfig.json +0 -23
- package/cli/vitest.config.ts +0 -59
- package/core/src/__tests__/adapters-contract/adapters-contract.test.md +0 -156
- package/core/src/__tests__/index.test.ts +0 -31
- package/core/src/__tests__/integration/phase3.test.ts +0 -405
- package/core/src/__tests__/pack/validator.test.ts +0 -312
- package/core/src/__tests__/secrets/crypto.test.ts +0 -190
- package/core/src/__tests__/secrets/manager.test.ts +0 -316
- package/core/src/__tests__/security/redactor-phase3.test.ts +0 -233
- package/core/src/__tests__/serve/health-checker.test.ts +0 -155
- package/core/src/__tests__/serve/process-manager.test.ts +0 -213
- package/core/src/__tests__/serve/server.test.ts +0 -103
- package/core/src/__tests__/vault/cas.test.ts +0 -178
- package/core/src/__tests__/vault/vault.test.ts +0 -296
- package/core/src/adapters/__tests__/fixtures/jest-coverage.json +0 -8
- package/core/src/adapters/__tests__/fixtures/jest-results.json +0 -41
- package/core/src/adapters/__tests__/fixtures/pytest-junit.xml +0 -16
- package/core/src/adapters/__tests__/fixtures/vitest-coverage.json +0 -8
- package/core/src/adapters/__tests__/fixtures/vitest-results.json +0 -50
- package/core/src/adapters/__tests__/gitleaks-secrets.test.ts +0 -452
- package/core/src/adapters/__tests__/jest-adapter.test.ts +0 -276
- package/core/src/adapters/__tests__/k6-perf.test.ts +0 -538
- package/core/src/adapters/__tests__/osv-deps.test.ts +0 -471
- package/core/src/adapters/__tests__/playwright-native-api.test.ts +0 -792
- package/core/src/adapters/__tests__/playwright-ui-e2e.test.ts +0 -431
- package/core/src/adapters/__tests__/playwright-ui.test.ts +0 -1073
- package/core/src/adapters/__tests__/pytest-adapter.test.ts +0 -207
- package/core/src/adapters/__tests__/semgrep-sast.test.ts +0 -436
- package/core/src/adapters/__tests__/vitest-adapter.test.ts +0 -208
- package/core/src/adapters/__tests__/zap-dast.test.ts +0 -453
- package/core/src/adapters/gitleaks-secrets.ts +0 -521
- package/core/src/adapters/jest-adapter.ts +0 -306
- package/core/src/adapters/k6-perf.ts +0 -479
- package/core/src/adapters/osv-deps.ts +0 -467
- package/core/src/adapters/playwright-native-adapter.ts +0 -472
- package/core/src/adapters/playwright-native-api.ts +0 -619
- package/core/src/adapters/playwright-ui.ts +0 -1088
- package/core/src/adapters/pytest-adapter.ts +0 -472
- package/core/src/adapters/semgrep-sast.ts +0 -410
- package/core/src/adapters/unit-test-types.ts +0 -106
- package/core/src/adapters/vitest-adapter.ts +0 -295
- package/core/src/adapters/zap-dast.ts +0 -551
- package/core/src/ai/__tests__/deepseek-provider.test.ts +0 -586
- package/core/src/ai/__tests__/ollama-provider.test.ts +0 -641
- package/core/src/ai/anthropic-provider.ts +0 -262
- package/core/src/ai/deepseek-provider.ts +0 -315
- package/core/src/ai/index.ts +0 -87
- package/core/src/ai/llm-client.ts +0 -52
- package/core/src/ai/mock-provider.ts +0 -146
- package/core/src/ai/ollama-provider.ts +0 -269
- package/core/src/ai/openai-provider.ts +0 -240
- package/core/src/ai/provider-factory.ts +0 -408
- package/core/src/artifacts/README.md +0 -78
- package/core/src/artifacts/index.ts +0 -16
- package/core/src/artifacts/ui-artifacts.ts +0 -412
- package/core/src/assertions/__tests__/engine.test.ts +0 -360
- package/core/src/assertions/engine.ts +0 -577
- package/core/src/assertions/index.ts +0 -13
- package/core/src/assertions/types.ts +0 -229
- package/core/src/auth/__tests__/api-key-provider.test.ts +0 -282
- package/core/src/auth/__tests__/auth-manager.test.ts +0 -430
- package/core/src/auth/__tests__/basic-auth-provider.test.ts +0 -364
- package/core/src/auth/__tests__/cloud-providers.test.ts +0 -751
- package/core/src/auth/__tests__/jwt-provider.test.ts +0 -400
- package/core/src/auth/__tests__/oauth2-provider.test.ts +0 -383
- package/core/src/auth/__tests__/totp-provider.test.ts +0 -294
- package/core/src/auth/__tests__/ui-login-provider.test.ts +0 -323
- package/core/src/auth/api-key-provider.ts +0 -75
- package/core/src/auth/aws-iam-provider.ts +0 -212
- package/core/src/auth/azure-ad-provider.ts +0 -126
- package/core/src/auth/basic-auth-provider.ts +0 -133
- package/core/src/auth/gcp-adc-provider.ts +0 -146
- package/core/src/auth/index.ts +0 -342
- package/core/src/auth/jwt-provider.ts +0 -193
- package/core/src/auth/manager.ts +0 -281
- package/core/src/auth/oauth2-provider.ts +0 -141
- package/core/src/auth/totp-provider.ts +0 -163
- package/core/src/auth/ui-login-provider.ts +0 -242
- package/core/src/cache/__tests__/lru-cache.test.ts +0 -564
- package/core/src/cache/index.ts +0 -13
- package/core/src/cache/lru-cache.ts +0 -536
- package/core/src/crawler/__tests__/journey-generator.test.ts +0 -344
- package/core/src/crawler/__tests__/selector-generator.test.ts +0 -211
- package/core/src/crawler/index.ts +0 -335
- package/core/src/crawler/journey-generator.ts +0 -471
- package/core/src/crawler/page-analyzer.ts +0 -857
- package/core/src/crawler/selector-generator.ts +0 -280
- package/core/src/crawler/types.ts +0 -475
- package/core/src/dashboard/__tests__/real-world.test.ts +0 -430
- package/core/src/dashboard/__tests__/server.test.ts +0 -283
- package/core/src/dashboard/__tests__/types.test.ts +0 -208
- package/core/src/dashboard/assets.ts +0 -692
- package/core/src/dashboard/index.ts +0 -17
- package/core/src/dashboard/server.ts +0 -401
- package/core/src/dashboard/types.ts +0 -78
- package/core/src/discoverer/__tests__/test-discoverer.test.ts +0 -444
- package/core/src/discoverer/index.ts +0 -374
- package/core/src/fixtures/__tests__/loader.test.ts +0 -246
- package/core/src/fixtures/__tests__/resolver.test.ts +0 -334
- package/core/src/fixtures/index.ts +0 -9
- package/core/src/fixtures/loader.ts +0 -200
- package/core/src/fixtures/resolver.ts +0 -221
- package/core/src/fixtures/types.ts +0 -86
- package/core/src/flakiness/__tests__/flakiness.test.ts +0 -554
- package/core/src/flakiness/index.ts +0 -536
- package/core/src/generation/__tests__/code-formatter.test.ts +0 -170
- package/core/src/generation/__tests__/code-generator-contract.test.ts +0 -207
- package/core/src/generation/__tests__/code-generator.test.ts +0 -586
- package/core/src/generation/__tests__/crawler-pack-generator.test.ts +0 -479
- package/core/src/generation/__tests__/generation-e2e-b2bshop.test.ts +0 -718
- package/core/src/generation/__tests__/generation-integration.test.ts +0 -655
- package/core/src/generation/__tests__/pack-generator.test.ts +0 -408
- package/core/src/generation/__tests__/prompt-builder.test.ts +0 -200
- package/core/src/generation/__tests__/real-provider-integration.test.ts +0 -414
- package/core/src/generation/__tests__/source-analyzer.test.ts +0 -774
- package/core/src/generation/__tests__/test-optimizer.test.ts +0 -255
- package/core/src/generation/code-formatter.ts +0 -408
- package/core/src/generation/code-generator.ts +0 -470
- package/core/src/generation/crawler-pack-generator.ts +0 -289
- package/core/src/generation/generator.ts +0 -113
- package/core/src/generation/index.ts +0 -59
- package/core/src/generation/pack-generator.ts +0 -527
- package/core/src/generation/prompt-builder.ts +0 -772
- package/core/src/generation/source-analyzer.ts +0 -830
- package/core/src/generation/test-optimizer.ts +0 -474
- package/core/src/generation/types.ts +0 -217
- package/core/src/hooks/__tests__/compose.test.ts +0 -636
- package/core/src/hooks/__tests__/runner.test.ts +0 -478
- package/core/src/hooks/compose.ts +0 -268
- package/core/src/hooks/runner.ts +0 -364
- package/core/src/index.ts +0 -255
- package/core/src/pack/__tests__/migrator.test.ts +0 -594
- package/core/src/pack/__tests__/validator.test.ts +0 -759
- package/core/src/pack/migrator.ts +0 -353
- package/core/src/pack/validator.ts +0 -359
- package/core/src/pack-v2/__tests__/loader.test.ts +0 -533
- package/core/src/pack-v2/__tests__/migrator.test.ts +0 -455
- package/core/src/pack-v2/__tests__/validator.test.ts +0 -609
- package/core/src/pack-v2/index.ts +0 -41
- package/core/src/pack-v2/loader.ts +0 -358
- package/core/src/pack-v2/migrator.ts +0 -540
- package/core/src/pack-v2/validator.ts +0 -731
- package/core/src/parallel/README.md +0 -143
- package/core/src/parallel/index.ts +0 -16
- package/core/src/parallel/parallel-runner.ts +0 -282
- package/core/src/pom/__tests__/loader.test.ts +0 -378
- package/core/src/pom/base-page.ts +0 -425
- package/core/src/pom/index.ts +0 -45
- package/core/src/pom/loader.ts +0 -480
- package/core/src/pom/types.ts +0 -146
- package/core/src/proof/__tests__/proof-roundtrip.test.ts +0 -149
- package/core/src/proof/__tests__/schema-validation-manual.mjs +0 -211
- package/core/src/proof/__tests__/schema-validation.test.ts +0 -336
- package/core/src/proof/__tests__/signer.test.ts +0 -486
- package/core/src/proof/__tests__/temporal-regression.test.ts +0 -537
- package/core/src/proof/__tests__/verifier-advanced.test.ts +0 -588
- package/core/src/proof/__tests__/verifier.test.ts +0 -413
- package/core/src/proof/bundle.ts +0 -290
- package/core/src/proof/canonicalize.ts +0 -116
- package/core/src/proof/index.ts +0 -74
- package/core/src/proof/schema.ts +0 -285
- package/core/src/proof/signer.ts +0 -293
- package/core/src/proof/verifier.ts +0 -380
- package/core/src/regression/__tests__/detector.test.ts +0 -396
- package/core/src/regression/__tests__/trend-analyzer.test.ts +0 -300
- package/core/src/regression/detector.ts +0 -629
- package/core/src/regression/index.ts +0 -34
- package/core/src/regression/trend-analyzer.ts +0 -468
- package/core/src/regression/types.ts +0 -295
- package/core/src/regression/vault.ts +0 -419
- package/core/src/repair/__tests__/repairer.test.ts +0 -572
- package/core/src/repair/__tests__/types.test.ts +0 -302
- package/core/src/repair/engine/__tests__/fixer.test.ts +0 -482
- package/core/src/repair/engine/__tests__/suggestion-engine.test.ts +0 -395
- package/core/src/repair/engine/fixer.ts +0 -271
- package/core/src/repair/engine/suggestion-engine.ts +0 -234
- package/core/src/repair/index.ts +0 -53
- package/core/src/repair/repairer.ts +0 -376
- package/core/src/repair/types.ts +0 -119
- package/core/src/repair/utils/__tests__/error-analyzer.test.ts +0 -454
- package/core/src/repair/utils/error-analyzer.ts +0 -308
- package/core/src/reporting/README.md +0 -144
- package/core/src/reporting/html-reporter.ts +0 -835
- package/core/src/reporting/index.ts +0 -16
- package/core/src/retry/README.md +0 -192
- package/core/src/retry/__tests__/flakiness-integration.test.ts +0 -475
- package/core/src/retry/__tests__/retry-engine.test.ts +0 -424
- package/core/src/retry/flakiness-integration.ts +0 -267
- package/core/src/retry/index.ts +0 -48
- package/core/src/retry/retry-engine.ts +0 -368
- package/core/src/retry/types.ts +0 -208
- package/core/src/retry/vault.ts +0 -413
- package/core/src/runner/__tests__/flakiness-integration.test.ts +0 -566
- package/core/src/runner/__tests__/phase3-e2e-b2bshop.test.ts +0 -218
- package/core/src/runner/__tests__/phase3-e2e-reqres.test.ts +0 -199
- package/core/src/runner/__tests__/phase3-runner.test.ts +0 -1118
- package/core/src/runner/e2e-helpers.ts +0 -216
- package/core/src/runner/phase3-runner.ts +0 -1536
- package/core/src/schemas/gherkin-report.json +0 -122
- package/core/src/secrets/__tests__/crypto.test.ts +0 -180
- package/core/src/secrets/crypto.ts +0 -289
- package/core/src/secrets/manager.ts +0 -272
- package/core/src/security/__tests__/hardening.test.ts +0 -480
- package/core/src/security/redaction-patterns-extended.ts +0 -278
- package/core/src/security/redactor.ts +0 -326
- package/core/src/self-healing/assertion-healer.ts +0 -485
- package/core/src/self-healing/engine.ts +0 -626
- package/core/src/self-healing/index.ts +0 -33
- package/core/src/self-healing/selector-healer.ts +0 -488
- package/core/src/self-healing/types.ts +0 -193
- package/core/src/serve/diagnostics-collector.ts +0 -201
- package/core/src/serve/health-checker.ts +0 -274
- package/core/src/serve/index.ts +0 -9
- package/core/src/serve/metrics-collector.ts +0 -386
- package/core/src/serve/process-manager.ts +0 -265
- package/core/src/serve/server.ts +0 -230
- package/core/src/slo/config.ts +0 -408
- package/core/src/slo/index.ts +0 -68
- package/core/src/slo/sli-calculator.ts +0 -474
- package/core/src/slo/slo-tracker.ts +0 -481
- package/core/src/slo/types.ts +0 -408
- package/core/src/slo/vault.ts +0 -600
- package/core/src/tui/__tests__/monitor.test.ts +0 -336
- package/core/src/tui/__tests__/real-world.test.ts +0 -376
- package/core/src/tui/__tests__/renderer.test.ts +0 -201
- package/core/src/tui/__tests__/types.test.ts +0 -295
- package/core/src/tui/index.ts +0 -19
- package/core/src/tui/monitor.ts +0 -331
- package/core/src/tui/renderer.ts +0 -269
- package/core/src/tui/types.ts +0 -68
- package/core/src/types/pack-v1.ts +0 -305
- package/core/src/types/pack-v2.ts +0 -525
- package/core/src/types/trust-score.ts +0 -258
- package/core/src/vault/__tests__/flakiness-vault.test.ts +0 -562
- package/core/src/vault/__tests__/vault.test.ts +0 -259
- package/core/src/vault/cas.ts +0 -323
- package/core/src/vault/index.ts +0 -1361
- package/core/src/vault/schema.sql +0 -168
- package/core/src/visual/README.md +0 -185
- package/core/src/visual/index.ts +0 -14
- package/core/src/visual/visual-regression.ts +0 -347
- package/core/src/watch/__tests__/watch-mode.test.ts +0 -192
- package/core/src/watch/index.ts +0 -14
- package/core/src/watch/watch-mode.ts +0 -565
- package/core/tsconfig.json +0 -12
- package/core/vitest.config.ts +0 -52
- package/docs/ARCHITECTURE.md +0 -901
- package/docs/AUDIT-GLOBAL-DEC2025.md +0 -271
- package/docs/BETA_TESTING.md +0 -257
- package/docs/BETA_TESTING_PLAN.md +0 -727
- package/docs/CERTIFICATION-REPORT.md +0 -142
- package/docs/COMPLETE_AUDIT_REFACTORING.md +0 -965
- package/docs/DEVELOPMENT.md +0 -545
- package/docs/DEVELOPMENT_HISTORY.md +0 -345
- package/docs/LIMITATIONS.md +0 -176
- package/docs/MIGRATION.md +0 -303
- package/docs/OPTION_3_4_EXPLORATION.md +0 -1257
- package/docs/PHASE1_PERFORMANCE.md +0 -144
- package/docs/QA360_Cloud.postman_collection.json +0 -89
- package/docs/QA360_TESTING_PHILOSOPHY.md +0 -769
- package/docs/QA_TEST_PLAN.md +0 -727
- package/docs/README.md +0 -50
- package/docs/STATUS.md +0 -198
- package/docs/STRATEGIC_STUDY_GOOSE_INTEGRATION.md +0 -615
- package/docs/USER_GUIDE.md +0 -687
- package/docs/WORK-DONE-ADAPTER-TESTS.md +0 -136
- package/docs/adapters-security.md +0 -485
- package/docs/architecture-diagram.mmd +0 -168
- package/docs/archive/ARCH-01-DAY6-BUILD-FIXES.md +0 -396
- package/docs/archive/ARCH-01-DAY6-FINAL-STATUS.md +0 -324
- package/docs/archive/ARCH-01_MCP_MERGE_ANALYSIS.md +0 -644
- package/docs/archive/ARCH-01_NEXT_STEPS.md +0 -60
- package/docs/archive/BRANCH_PROTECTION.md +0 -183
- package/docs/archive/CI_LOCKDOWN_CHECKLIST.md +0 -222
- package/docs/archive/HANDOFF_TEST-01.md +0 -669
- package/docs/archive/LEGAL_READY_PLACEHOLDERS.md +0 -372
- package/docs/archive/NODE_UPGRADE_GUIDE.md +0 -188
- package/docs/archive/PHASE1_COMPLETION.md +0 -386
- package/docs/archive/PHASE2_COMPLETION.md +0 -404
- package/docs/archive/PHASE3_AND_4_FINAL.md +0 -360
- package/docs/archive/PHASE3_COMPLETE.md +0 -301
- package/docs/archive/PHASE3_STATUS.md +0 -255
- package/docs/archive/PRE-WEEK2-AUDIT.md +0 -364
- package/docs/archive/README.md +0 -16
- package/docs/archive/SCHEMA_AJV_2020_FIX.md +0 -245
- package/docs/archive/TEST-01_AUDIT_REPORT.md +0 -240
- package/docs/archive/TEST-01_COVERAGE_PLAN.md +0 -423
- package/docs/archive/obsolete-proposals/dom-element-discovery-mode.md +0 -250
- package/docs/archive/obsolete-proposals/qa360-comprehensive-test-plan.md +0 -1249
- package/docs/archive/obsolete-proposals/qa360-quick-start-guide.md +0 -298
- package/docs/archive/obsolete-proposals/technical-plan-dom-discovery.md +0 -870
- package/docs/budgets-advanced.md +0 -308
- package/docs/examples/history-export-gc.md +0 -285
- package/docs/examples/pack-v2-complete.yaml +0 -158
- package/docs/examples/pack-v2-quickstart.yaml +0 -24
- package/docs/examples/pack-v2-ui-login.yaml +0 -81
- package/docs/examples/qa360-report.json +0 -50
- package/docs/history.md +0 -565
- package/docs/hooks.md +0 -304
- package/docs/llm-providers.md +0 -512
- package/docs/mcp-server.md +0 -651
- package/docs/mcp-tools.md +0 -1131
- package/docs/pack-v1.md +0 -383
- package/docs/pack-v2.md +0 -558
- package/docs/page-objects.md +0 -366
- package/docs/proofs.md +0 -670
- package/docs/quickstart-5min.md +0 -257
- package/docs/readiness-ci.md +0 -654
- package/docs/rfc/README.md +0 -20
- package/docs/rfc/proof-bundle-v1.md +0 -787
- package/docs/secrets.md +0 -392
- package/docs/serve.md +0 -494
- package/docs/unit-test-adapters.md +0 -168
- package/docs/vault.md +0 -491
- package/e2e/qa360-e2e.test.ts +0 -696
- package/e2e/vitest.config.ts +0 -18
- package/examples/README.md +0 -50
- package/examples/ci/docker-compose-serve.yml +0 -375
- package/examples/ci/github-actions-serve.yml +0 -345
- package/examples/ci/gitlab-ci-serve.yml +0 -407
- package/examples/datasets/README.md +0 -101
- package/examples/datasets/b2bshop.ts +0 -155
- package/examples/datasets/index.ts +0 -57
- package/examples/datasets/reqres.ts +0 -195
- package/examples/fixtures-demo/fixtures/users.yml +0 -39
- package/examples/fixtures-demo/pack.yml +0 -71
- package/examples/future-api/README.md +0 -16
- package/examples/future-api/diag.js +0 -7
- package/examples/future-api/health.js +0 -4
- package/examples/future-api/packs.js +0 -13
- package/examples/future-api/runpack.js +0 -10
- package/examples/generation/README.md +0 -148
- package/examples/generation/pack-generator-example.js +0 -115
- package/examples/generation/source-analyzer-example.js +0 -115
- package/examples/httpbin/pack.yml +0 -59
- package/examples/load-testing/mcp-load.yml +0 -115
- package/examples/load-testing/mcp-stdio.yml +0 -95
- package/examples/mcp/claude-desktop-config.json +0 -33
- package/examples/mcp/claude-desktop.json +0 -16
- package/examples/mcp/conversation-sample.md +0 -131
- package/examples/mcp/demo-60s.md +0 -330
- package/examples/mcp/sample-conversation.jsonl +0 -21
- package/examples/mcp/vscode-settings.json +0 -22
- package/examples/pack-v2-complete.yml +0 -242
- package/examples/pack-v2-examples.md +0 -244
- package/examples/pack-v2-quickstart.yml +0 -55
- package/examples/packs-business/ecommerce-api.yml +0 -121
- package/examples/packs-business/saas-dashboard-ui.yml +0 -133
- package/examples/packs-conformance/compose-multi.yml +0 -174
- package/examples/packs-conformance/full.yml +0 -152
- package/examples/packs-conformance/heavy-artifacts.yml +0 -152
- package/examples/packs-conformance/minimal.yml +0 -71
- package/examples/packs-conformance/secrets-missing.yml +0 -97
- package/examples/packs-conformance/timeouts.yml +0 -77
- package/examples/pom-demo/README.md +0 -104
- package/examples/pom-demo/pack.yml +0 -60
- package/examples/pom-demo/pages/DashboardPage.page.ts +0 -73
- package/examples/pom-demo/pages/LoginPage.page.ts +0 -76
- package/examples/proofs/e2e-playwright-proof.json +0 -75
- package/examples/proofs/httpbin-proof.json +0 -69
- package/examples/proofs/multi-adapter-proof.json +0 -117
- package/examples/proofs/test-proof.json +0 -26
- package/examples/restful-api-dev/README.md +0 -102
- package/examples/restful-api-dev/restful-api-advanced.yml +0 -29
- package/examples/restful-api-dev/restful-api-basic.yml +0 -29
- package/examples/web-lite/.github/workflows/qa360-phase3.yml +0 -73
- package/examples/web-lite/api-mock/server.js +0 -258
- package/examples/web-lite/pack.yml +0 -71
- package/examples/web-lite/services.yml +0 -43
- package/examples/web-lite/web-content/healthz +0 -1
- package/examples/web-lite/web-content/index.html +0 -259
- package/packages/mcp/CHANGELOG.md +0 -109
- package/packages/mcp/IMPLEMENTATION_SUMMARY.md +0 -350
- package/packages/mcp/LICENSE +0 -21
- package/packages/mcp/QUICK_START.md +0 -291
- package/packages/mcp/README.md +0 -294
- package/packages/mcp/TELEMETRY.md +0 -220
- package/packages/mcp/package.json +0 -91
- package/packages/mcp/scripts/generate-sbom-fallback.cjs +0 -84
- package/packages/mcp/scripts/safe-postinstall.cjs +0 -32
- package/packages/mcp/src/__tests__/contract.test.ts +0 -902
- package/packages/mcp/src/cli/cli.ts +0 -137
- package/packages/mcp/src/cli/doctor.ts +0 -286
- package/packages/mcp/src/cli/fix.ts +0 -99
- package/packages/mcp/src/cli/init.ts +0 -233
- package/packages/mcp/src/cli/postinstall.ts +0 -14
- package/packages/mcp/src/cli/reset.ts +0 -44
- package/packages/mcp/src/cli/telemetry.ts +0 -166
- package/packages/mcp/src/cli/test-dx.ts +0 -94
- package/packages/mcp/src/cli/uninstall.ts +0 -80
- package/packages/mcp/src/cli/up.ts +0 -178
- package/packages/mcp/src/index.ts +0 -12
- package/packages/mcp/src/scripts/e2e-local.ts +0 -337
- package/packages/mcp/src/scripts/verify-settings.ts +0 -242
- package/packages/mcp/src/security/audit.ts +0 -244
- package/packages/mcp/src/security/manager.ts +0 -242
- package/packages/mcp/src/server/full-server.ts +0 -212
- package/packages/mcp/src/server/minimal-server.ts +0 -134
- package/packages/mcp/src/tools/history.ts +0 -388
- package/packages/mcp/src/tools/pack.ts +0 -449
- package/packages/mcp/src/tools/registry.ts +0 -638
- package/packages/mcp/src/tools/report.ts +0 -100
- package/packages/mcp/src/tools/run.ts +0 -268
- package/packages/mcp/src/tools/secrets.ts +0 -198
- package/packages/mcp/src/tools/serve.ts +0 -221
- package/packages/mcp/src/tools/triage.ts +0 -532
- package/packages/mcp/src/tools/types.ts +0 -26
- package/packages/mcp/src/tools/vault.ts +0 -164
- package/packages/mcp/src/tools/verify.ts +0 -166
- package/packages/mcp/src/types/index.ts +0 -311
- package/packages/mcp/src/types/mcp-stubs.ts +0 -83
- package/packages/mcp/tsconfig.json +0 -16
- package/playwright.config.ts +0 -20
- package/pnpm-workspace.yaml +0 -4
- package/run-test-and-push.sh +0 -20
- package/scripts/build-proof-cli.sh +0 -110
- package/scripts/ci/check-windows-paths.js +0 -92
- package/scripts/ci/invariants.sh +0 -124
- package/scripts/ci/make-final-bundle.js +0 -106
- package/scripts/ci/mcp-run-multipack.js +0 -305
- package/scripts/ci/run-pack-suite.sh +0 -103
- package/scripts/ci/run-phase7-final.sh +0 -190
- package/scripts/ci/slo-assert.js +0 -158
- package/scripts/ci/test-fault-tolerance.sh +0 -301
- package/scripts/install-mcp.sh +0 -66
- package/scripts/mcp-smoke.mjs +0 -27
- package/scripts/smoke.sh +0 -26
- package/scripts/stress-test.js +0 -288
- package/scripts/sync-version.mjs +0 -50
- package/scripts/validate-examples.mjs +0 -404
- package/scripts/validation/simple-pack-check.sh +0 -51
- package/scripts/validation/validate-universal-pack.mjs +0 -77
- package/scripts/verify-persistence.js +0 -127
- package/test-pack.yaml +0 -43
- package/test-results/.last-run.json +0 -4
- package/test-runner.mjs +0 -87
- package/tests/artifacts.spec.js +0 -147
- package/tests/contracts.spec.js +0 -239
- package/tests/e2e/assertions.test.mjs +0 -370
- package/tests/e2e/crawler.test.mjs +0 -451
- package/tests/e2e/playwright-plus-plus.test.mjs +0 -604
- package/tests/e2e/proof-bundle.test.mjs +0 -258
- package/tests/e2e/real-world/saucedemo.test.mjs +0 -714
- package/tests/e2e/real-world/the-internet-herokuapp.test.mjs +0 -760
- package/tests/e2e/ui-actions.test.mjs +0 -546
- package/tests/gherkin.e2e.spec.ts +0 -310
- package/tests/no-console-errors.spec.js +0 -136
- package/tests/pdf.spec.ts +0 -252
- package/tests/run-pack.spec.ts +0 -58
- package/tsconfig.base.json +0 -15
- package/tsconfig.build.json +0 -8
- package/tsconfig.json +0 -37
- package/tsconfig.test.json +0 -18
- package/typedoc.json +0 -37
- package/ui/README.md +0 -50
- package/verify-proof.mjs +0 -60
package/core/src/vault/index.ts
DELETED
|
@@ -1,1361 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* QA360 Evidence Vault Engine
|
|
3
|
-
* SQLite WAL-based storage with idempotence and CAS integration
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import sqlite3pkg from 'sqlite3';
|
|
7
|
-
const { Database } = sqlite3pkg;
|
|
8
|
-
import { promisify } from 'util';
|
|
9
|
-
import { existsSync, mkdirSync } from 'fs';
|
|
10
|
-
import { join, resolve } from 'path';
|
|
11
|
-
import { randomUUID } from 'crypto';
|
|
12
|
-
import { createHash } from 'crypto';
|
|
13
|
-
import { ContentAddressableStorage, CASArtifact } from './cas.js';
|
|
14
|
-
import { SecurityRedactor } from '../security/redactor.js';
|
|
15
|
-
|
|
16
|
-
export interface VaultConfig {
|
|
17
|
-
baseDir: string;
|
|
18
|
-
enableWAL?: boolean;
|
|
19
|
-
maxConnections?: number;
|
|
20
|
-
retentionDays?: number;
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
export interface RunRecord {
|
|
24
|
-
id: string;
|
|
25
|
-
run_key?: string;
|
|
26
|
-
started_at: number;
|
|
27
|
-
ended_at?: number;
|
|
28
|
-
pack_hash: string;
|
|
29
|
-
pack_path: string;
|
|
30
|
-
status: 'running' | 'passed' | 'failed' | 'cancelled' | 'error';
|
|
31
|
-
trust_score?: number;
|
|
32
|
-
weights_json?: string;
|
|
33
|
-
sign_alg?: string;
|
|
34
|
-
signature_hex?: string;
|
|
35
|
-
proof_pdf_path?: string;
|
|
36
|
-
pinned: number;
|
|
37
|
-
created_at: number;
|
|
38
|
-
updated_at: number;
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
export interface GateRecord {
|
|
42
|
-
id?: number;
|
|
43
|
-
run_id: string;
|
|
44
|
-
name: string;
|
|
45
|
-
status: 'passed' | 'failed' | 'skipped' | 'warn';
|
|
46
|
-
duration_ms?: number;
|
|
47
|
-
metrics_json?: string;
|
|
48
|
-
budgets_json?: string;
|
|
49
|
-
started_at?: number;
|
|
50
|
-
ended_at?: number;
|
|
51
|
-
created_at: number;
|
|
52
|
-
}
|
|
53
|
-
|
|
54
|
-
export interface FindingRecord {
|
|
55
|
-
id?: number;
|
|
56
|
-
run_id: string;
|
|
57
|
-
gate: string;
|
|
58
|
-
severity: 'info' | 'low' | 'medium' | 'high' | 'critical';
|
|
59
|
-
rule: string;
|
|
60
|
-
location?: string;
|
|
61
|
-
message: string;
|
|
62
|
-
raw_output?: string;
|
|
63
|
-
fingerprint?: string;
|
|
64
|
-
created_at: number;
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
/**
|
|
68
|
-
* Flakiness history record - tracks test reliability over time
|
|
69
|
-
*/
|
|
70
|
-
export interface FlakinessRecord {
|
|
71
|
-
id?: number;
|
|
72
|
-
run_id: string;
|
|
73
|
-
test_id: string; // Hashed test identifier
|
|
74
|
-
test_name: string;
|
|
75
|
-
gate: string;
|
|
76
|
-
file_path: string;
|
|
77
|
-
score: number; // 0-100 flakiness score
|
|
78
|
-
category: 'legendary' | 'solid' | 'good' | 'shaky' | 'unstable';
|
|
79
|
-
total_runs: number;
|
|
80
|
-
successful_runs: number;
|
|
81
|
-
avg_duration_ms: number;
|
|
82
|
-
pattern_type?: string;
|
|
83
|
-
suggested_fix?: string;
|
|
84
|
-
confidence?: number;
|
|
85
|
-
first_seen: number;
|
|
86
|
-
last_seen: number;
|
|
87
|
-
current_streak?: number;
|
|
88
|
-
streak_type?: 'pass' | 'fail';
|
|
89
|
-
created_at: number;
|
|
90
|
-
}
|
|
91
|
-
|
|
92
|
-
/**
|
|
93
|
-
* Flakiness pattern detection record
|
|
94
|
-
*/
|
|
95
|
-
export interface FlakinessPatternRecord {
|
|
96
|
-
id?: number;
|
|
97
|
-
test_id: string; // Hashed test identifier
|
|
98
|
-
test_name: string;
|
|
99
|
-
pattern_type: 'timing' | 'race_condition' | 'external_dependency' | 'environment_specific' | 'selector_issue' | 'unknown';
|
|
100
|
-
description: string;
|
|
101
|
-
suggested_fix: string;
|
|
102
|
-
confidence: number; // 0-1
|
|
103
|
-
auto_fix_possible: boolean;
|
|
104
|
-
first_detected: number;
|
|
105
|
-
last_detected: number;
|
|
106
|
-
detection_count: number;
|
|
107
|
-
created_at: number;
|
|
108
|
-
updated_at: number;
|
|
109
|
-
}
|
|
110
|
-
|
|
111
|
-
/**
|
|
112
|
-
* Flakiness quarantine record
|
|
113
|
-
*/
|
|
114
|
-
export interface QuarantineRecord {
|
|
115
|
-
id?: number;
|
|
116
|
-
test_id: string; // Hashed test identifier
|
|
117
|
-
test_name: string;
|
|
118
|
-
gate: string;
|
|
119
|
-
file_path: string;
|
|
120
|
-
reason: string;
|
|
121
|
-
score: number; // Score at time of quarantine
|
|
122
|
-
category: 'shaky' | 'unstable';
|
|
123
|
-
quarantined_at: number;
|
|
124
|
-
quarantined_by: string; // 'auto' or user_id
|
|
125
|
-
resolved_at?: number;
|
|
126
|
-
resolved_by?: string;
|
|
127
|
-
notes?: string;
|
|
128
|
-
created_at: number;
|
|
129
|
-
updated_at: number;
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
export interface ArtifactRecord {
|
|
133
|
-
sha256: string;
|
|
134
|
-
mime_type: string;
|
|
135
|
-
size_bytes: number;
|
|
136
|
-
cas_path: string;
|
|
137
|
-
original_name?: string;
|
|
138
|
-
created_at: number;
|
|
139
|
-
last_accessed: number;
|
|
140
|
-
}
|
|
141
|
-
|
|
142
|
-
export interface RunArtifactLink {
|
|
143
|
-
run_id: string;
|
|
144
|
-
sha256: string;
|
|
145
|
-
label: string;
|
|
146
|
-
created_at: number;
|
|
147
|
-
}
|
|
148
|
-
|
|
149
|
-
export interface BeginRunOptions {
|
|
150
|
-
run_key?: string;
|
|
151
|
-
pack_path: string;
|
|
152
|
-
pack_hash: string;
|
|
153
|
-
weights?: Record<string, any>;
|
|
154
|
-
started_at?: number; // Optional: use pre-existing timestamp (e.g., from run start)
|
|
155
|
-
}
|
|
156
|
-
|
|
157
|
-
export interface FinishRunOptions {
|
|
158
|
-
status: 'passed' | 'failed' | 'cancelled' | 'error';
|
|
159
|
-
trust_score?: number;
|
|
160
|
-
signature?: string;
|
|
161
|
-
proof_pdf_sha?: string;
|
|
162
|
-
weights?: Record<string, any>;
|
|
163
|
-
}
|
|
164
|
-
|
|
165
|
-
export interface RunFilters {
|
|
166
|
-
status?: string[];
|
|
167
|
-
since?: Date;
|
|
168
|
-
until?: Date;
|
|
169
|
-
pack_hash?: string;
|
|
170
|
-
limit?: number;
|
|
171
|
-
offset?: number;
|
|
172
|
-
}
|
|
173
|
-
|
|
174
|
-
export interface VaultStats {
|
|
175
|
-
totalRuns: number;
|
|
176
|
-
totalGates: number;
|
|
177
|
-
totalFindings: number;
|
|
178
|
-
totalArtifacts: number;
|
|
179
|
-
vaultSizeBytes: number;
|
|
180
|
-
oldestRun?: Date;
|
|
181
|
-
newestRun?: Date;
|
|
182
|
-
}
|
|
183
|
-
|
|
184
|
-
export class EvidenceVault {
|
|
185
|
-
private db!: InstanceType<typeof Database>;
|
|
186
|
-
private cas: ContentAddressableStorage;
|
|
187
|
-
private redactor: SecurityRedactor;
|
|
188
|
-
private readonly dbPath: string;
|
|
189
|
-
private readonly baseDir: string;
|
|
190
|
-
|
|
191
|
-
constructor(config: VaultConfig) {
|
|
192
|
-
this.baseDir = resolve(config.baseDir);
|
|
193
|
-
this.dbPath = join(this.baseDir, 'vault.db');
|
|
194
|
-
this.cas = new ContentAddressableStorage(join(this.baseDir, 'runs'));
|
|
195
|
-
this.redactor = SecurityRedactor.forReports();
|
|
196
|
-
|
|
197
|
-
this.ensureDirectories();
|
|
198
|
-
this.initializeDatabase(config);
|
|
199
|
-
}
|
|
200
|
-
|
|
201
|
-
/**
|
|
202
|
-
* Open vault connection and initialize schema
|
|
203
|
-
*/
|
|
204
|
-
static async open(baseDir: string, config?: Partial<VaultConfig>): Promise<EvidenceVault> {
|
|
205
|
-
const fullConfig: VaultConfig = {
|
|
206
|
-
baseDir,
|
|
207
|
-
enableWAL: true,
|
|
208
|
-
maxConnections: 10,
|
|
209
|
-
retentionDays: 90,
|
|
210
|
-
...config
|
|
211
|
-
};
|
|
212
|
-
|
|
213
|
-
const vault = new EvidenceVault(fullConfig);
|
|
214
|
-
await vault.initializeSchema();
|
|
215
|
-
return vault;
|
|
216
|
-
}
|
|
217
|
-
|
|
218
|
-
/**
|
|
219
|
-
* Begin a new run with idempotence support
|
|
220
|
-
*/
|
|
221
|
-
async beginRun(options: BeginRunOptions): Promise<{ runId: string; isReused: boolean }> {
|
|
222
|
-
// Check for existing run with same run_key
|
|
223
|
-
if (options.run_key) {
|
|
224
|
-
const existing = await this.getRunByKey(options.run_key);
|
|
225
|
-
if (existing) {
|
|
226
|
-
console.log(`[VAULT] RUN_IDEMPOTENT_REUSED: ${existing.id} (key: ${options.run_key})`);
|
|
227
|
-
return { runId: existing.id, isReused: true };
|
|
228
|
-
}
|
|
229
|
-
}
|
|
230
|
-
|
|
231
|
-
const runId = randomUUID();
|
|
232
|
-
const now = options.started_at ?? Date.now(); // Use provided timestamp or current time
|
|
233
|
-
|
|
234
|
-
const run: Partial<RunRecord> = {
|
|
235
|
-
id: runId,
|
|
236
|
-
run_key: options.run_key,
|
|
237
|
-
started_at: now,
|
|
238
|
-
pack_hash: options.pack_hash,
|
|
239
|
-
pack_path: options.pack_path,
|
|
240
|
-
status: 'running',
|
|
241
|
-
weights_json: options.weights ? JSON.stringify(options.weights) : undefined,
|
|
242
|
-
pinned: 0,
|
|
243
|
-
created_at: now,
|
|
244
|
-
updated_at: now
|
|
245
|
-
};
|
|
246
|
-
|
|
247
|
-
await this.insertRun(run);
|
|
248
|
-
console.log(`[VAULT] RUN_STARTED: ${runId} (key: ${options.run_key || 'none'})`);
|
|
249
|
-
|
|
250
|
-
return { runId, isReused: false };
|
|
251
|
-
}
|
|
252
|
-
|
|
253
|
-
/**
|
|
254
|
-
* Finish a run with final status and signature
|
|
255
|
-
*/
|
|
256
|
-
async finishRun(runId: string, options: FinishRunOptions): Promise<void> {
|
|
257
|
-
const now = Date.now();
|
|
258
|
-
|
|
259
|
-
const updates: Partial<RunRecord> = {
|
|
260
|
-
ended_at: now,
|
|
261
|
-
status: options.status,
|
|
262
|
-
trust_score: options.trust_score,
|
|
263
|
-
signature_hex: options.signature,
|
|
264
|
-
updated_at: now
|
|
265
|
-
};
|
|
266
|
-
|
|
267
|
-
if (options.weights) {
|
|
268
|
-
updates.weights_json = JSON.stringify(options.weights);
|
|
269
|
-
}
|
|
270
|
-
|
|
271
|
-
if (options.proof_pdf_sha) {
|
|
272
|
-
const artifact = await this.getArtifact(options.proof_pdf_sha);
|
|
273
|
-
if (artifact) {
|
|
274
|
-
updates.proof_pdf_path = artifact.cas_path;
|
|
275
|
-
await this.attachArtifact(runId, { sha256: options.proof_pdf_sha, label: 'proof_pdf' });
|
|
276
|
-
}
|
|
277
|
-
}
|
|
278
|
-
|
|
279
|
-
await this.updateRun(runId, updates);
|
|
280
|
-
console.log(`[VAULT] RUN_FINISHED: ${runId} (status: ${options.status}, trust: ${options.trust_score})`);
|
|
281
|
-
}
|
|
282
|
-
|
|
283
|
-
/**
|
|
284
|
-
* Record gate execution result
|
|
285
|
-
*/
|
|
286
|
-
async recordGate(runId: string, gate: Omit<GateRecord, 'id' | 'run_id' | 'created_at'>): Promise<number> {
|
|
287
|
-
const gateRecord: Omit<GateRecord, 'id'> = {
|
|
288
|
-
run_id: runId,
|
|
289
|
-
created_at: Date.now(),
|
|
290
|
-
...gate
|
|
291
|
-
};
|
|
292
|
-
|
|
293
|
-
const gateId = await this.insertGate(gateRecord);
|
|
294
|
-
console.log(`[VAULT] GATE_RECORDED: ${runId}/${gate.name} (status: ${gate.status}, duration: ${gate.duration_ms}ms)`);
|
|
295
|
-
|
|
296
|
-
return gateId;
|
|
297
|
-
}
|
|
298
|
-
|
|
299
|
-
/**
|
|
300
|
-
* Record security/quality finding
|
|
301
|
-
*/
|
|
302
|
-
async recordFinding(runId: string, finding: Omit<FindingRecord, 'id' | 'run_id' | 'created_at'>): Promise<number> {
|
|
303
|
-
// Redact sensitive information
|
|
304
|
-
const redactedFinding: Omit<FindingRecord, 'id'> = {
|
|
305
|
-
run_id: runId,
|
|
306
|
-
created_at: Date.now(),
|
|
307
|
-
...finding,
|
|
308
|
-
message: this.redactor.redact(finding.message),
|
|
309
|
-
raw_output: finding.raw_output ? this.redactor.redact(finding.raw_output) : undefined,
|
|
310
|
-
fingerprint: finding.fingerprint || this.generateFindingFingerprint(finding)
|
|
311
|
-
};
|
|
312
|
-
|
|
313
|
-
const findingId = await this.insertFinding(redactedFinding);
|
|
314
|
-
console.log(`[VAULT] FINDING_RECORDED: ${runId}/${finding.gate} (severity: ${finding.severity}, rule: ${finding.rule})`);
|
|
315
|
-
|
|
316
|
-
return findingId;
|
|
317
|
-
}
|
|
318
|
-
|
|
319
|
-
/**
|
|
320
|
-
* Record flakiness analysis result
|
|
321
|
-
*/
|
|
322
|
-
async recordFlakiness(runId: string, flakiness: Omit<FlakinessRecord, 'id' | 'created_at' | 'run_id'>): Promise<number> {
|
|
323
|
-
const flakinessRecord: Omit<FlakinessRecord, 'id'> = {
|
|
324
|
-
run_id: runId,
|
|
325
|
-
created_at: Date.now(),
|
|
326
|
-
...flakiness
|
|
327
|
-
};
|
|
328
|
-
|
|
329
|
-
const flakinessId = await this.insertFlakiness(flakinessRecord);
|
|
330
|
-
console.log(`[VAULT] FLAKINESS_RECORDED: ${runId}/${flakiness.test_name} (score: ${flakiness.score}, category: ${flakiness.category})`);
|
|
331
|
-
|
|
332
|
-
return flakinessId;
|
|
333
|
-
}
|
|
334
|
-
|
|
335
|
-
/**
|
|
336
|
-
* Record detected flakiness pattern
|
|
337
|
-
*/
|
|
338
|
-
async recordFlakinessPattern(pattern: Omit<FlakinessPatternRecord, 'id' | 'created_at' | 'updated_at' | 'first_detected' | 'last_detected' | 'detection_count'>): Promise<number> {
|
|
339
|
-
const now = Date.now();
|
|
340
|
-
|
|
341
|
-
// Check if pattern already exists for this test
|
|
342
|
-
const existing = await this.getFlakinessPattern(pattern.test_id, pattern.pattern_type);
|
|
343
|
-
|
|
344
|
-
if (existing) {
|
|
345
|
-
// Update existing pattern
|
|
346
|
-
await this.updateFlakinessPattern(existing.id!, {
|
|
347
|
-
last_detected: now,
|
|
348
|
-
detection_count: (existing.detection_count || 0) + 1,
|
|
349
|
-
updated_at: now
|
|
350
|
-
});
|
|
351
|
-
console.log(`[VAULT] FLAKY_PATTERN_UPDATED: ${pattern.test_name} (${pattern.pattern_type})`);
|
|
352
|
-
return existing.id!;
|
|
353
|
-
}
|
|
354
|
-
|
|
355
|
-
// Insert new pattern
|
|
356
|
-
const patternRecord: Omit<FlakinessPatternRecord, 'id'> = {
|
|
357
|
-
first_detected: now,
|
|
358
|
-
last_detected: now,
|
|
359
|
-
detection_count: 1,
|
|
360
|
-
created_at: now,
|
|
361
|
-
updated_at: now,
|
|
362
|
-
...pattern
|
|
363
|
-
};
|
|
364
|
-
|
|
365
|
-
const patternId = await this.insertFlakinessPattern(patternRecord);
|
|
366
|
-
console.log(`[VAULT] FLAKY_PATTERN_DETECTED: ${pattern.test_name} (${pattern.pattern_type})`);
|
|
367
|
-
|
|
368
|
-
return patternId;
|
|
369
|
-
}
|
|
370
|
-
|
|
371
|
-
/**
|
|
372
|
-
* Add test to quarantine
|
|
373
|
-
*/
|
|
374
|
-
async addToQuarantine(quarantine: Omit<QuarantineRecord, 'id' | 'created_at' | 'updated_at'>): Promise<number> {
|
|
375
|
-
const now = Date.now();
|
|
376
|
-
|
|
377
|
-
// Check if already in quarantine
|
|
378
|
-
const existing = await this.getQuarantine(quarantine.test_id);
|
|
379
|
-
|
|
380
|
-
if (existing && !existing.resolved_at) {
|
|
381
|
-
console.log(`[VAULT] ALREADY_IN_QUARANTINE: ${quarantine.test_name}`);
|
|
382
|
-
return existing.id!;
|
|
383
|
-
}
|
|
384
|
-
|
|
385
|
-
const quarantineRecord: Omit<QuarantineRecord, 'id'> = {
|
|
386
|
-
created_at: now,
|
|
387
|
-
updated_at: now,
|
|
388
|
-
...quarantine
|
|
389
|
-
};
|
|
390
|
-
|
|
391
|
-
const quarantineId = await this.insertQuarantine(quarantineRecord);
|
|
392
|
-
console.log(`[VAULT] QUARANTINED: ${quarantine.test_name} (${quarantine.category})`);
|
|
393
|
-
|
|
394
|
-
return quarantineId;
|
|
395
|
-
}
|
|
396
|
-
|
|
397
|
-
/**
|
|
398
|
-
* Remove test from quarantine
|
|
399
|
-
*/
|
|
400
|
-
async removeFromQuarantine(testId: string, resolvedBy: string, notes?: string): Promise<void> {
|
|
401
|
-
await this.updateQuarantine(testId, {
|
|
402
|
-
resolved_at: Date.now(),
|
|
403
|
-
resolved_by: resolvedBy,
|
|
404
|
-
notes,
|
|
405
|
-
updated_at: Date.now()
|
|
406
|
-
});
|
|
407
|
-
console.log(`[VAULT] QUARANTINE_REMOVED: ${testId} (by: ${resolvedBy})`);
|
|
408
|
-
}
|
|
409
|
-
|
|
410
|
-
/**
|
|
411
|
-
* Get flakiness history for a test
|
|
412
|
-
*/
|
|
413
|
-
async getFlakinessHistory(testId: string, limit: number = 50): Promise<FlakinessRecord[]> {
|
|
414
|
-
return new Promise((resolve, reject) => {
|
|
415
|
-
this.db.all(
|
|
416
|
-
`SELECT * FROM flakiness_history WHERE test_id = ? ORDER BY created_at DESC LIMIT ?`,
|
|
417
|
-
[testId, limit],
|
|
418
|
-
(err: Error | null, rows: any) => {
|
|
419
|
-
if (err) reject(err);
|
|
420
|
-
else resolve(rows as FlakinessRecord[]);
|
|
421
|
-
}
|
|
422
|
-
);
|
|
423
|
-
});
|
|
424
|
-
}
|
|
425
|
-
|
|
426
|
-
/**
|
|
427
|
-
* Get flakiness history for a run
|
|
428
|
-
*/
|
|
429
|
-
async getRunFlakiness(runId: string): Promise<FlakinessRecord[]> {
|
|
430
|
-
return new Promise((resolve, reject) => {
|
|
431
|
-
this.db.all(
|
|
432
|
-
`SELECT * FROM flakiness_history WHERE run_id = ? ORDER BY score ASC`,
|
|
433
|
-
[runId],
|
|
434
|
-
(err: Error | null, rows: any) => {
|
|
435
|
-
if (err) reject(err);
|
|
436
|
-
else resolve(rows as FlakinessRecord[]);
|
|
437
|
-
}
|
|
438
|
-
);
|
|
439
|
-
});
|
|
440
|
-
}
|
|
441
|
-
|
|
442
|
-
/**
|
|
443
|
-
* Get all quarantined tests
|
|
444
|
-
*/
|
|
445
|
-
async getQuarantinedTests(includeResolved: boolean = false): Promise<QuarantineRecord[]> {
|
|
446
|
-
let query = 'SELECT * FROM flakiness_quarantine WHERE 1=1';
|
|
447
|
-
const params: any[] = [];
|
|
448
|
-
|
|
449
|
-
if (!includeResolved) {
|
|
450
|
-
query += ' AND resolved_at IS NULL';
|
|
451
|
-
}
|
|
452
|
-
|
|
453
|
-
query += ' ORDER BY quarantined_at DESC';
|
|
454
|
-
|
|
455
|
-
return new Promise((resolve, reject) => {
|
|
456
|
-
this.db.all(query, params, (err: Error | null, rows: any) => {
|
|
457
|
-
if (err) reject(err);
|
|
458
|
-
else resolve(rows as QuarantineRecord[]);
|
|
459
|
-
});
|
|
460
|
-
});
|
|
461
|
-
}
|
|
462
|
-
|
|
463
|
-
/**
|
|
464
|
-
* Get quarantine record for a test
|
|
465
|
-
*/
|
|
466
|
-
async getQuarantine(testId: string): Promise<QuarantineRecord | null> {
|
|
467
|
-
return new Promise((resolve, reject) => {
|
|
468
|
-
this.db.get(
|
|
469
|
-
'SELECT * FROM flakiness_quarantine WHERE test_id = ? ORDER BY created_at DESC LIMIT 1',
|
|
470
|
-
[testId],
|
|
471
|
-
(err: Error | null, row: any) => {
|
|
472
|
-
if (err) reject(err);
|
|
473
|
-
else resolve(row as QuarantineRecord || null);
|
|
474
|
-
}
|
|
475
|
-
);
|
|
476
|
-
});
|
|
477
|
-
}
|
|
478
|
-
|
|
479
|
-
/**
|
|
480
|
-
* Get flakiness pattern for a test
|
|
481
|
-
*/
|
|
482
|
-
async getFlakinessPattern(testId: string, patternType: string): Promise<FlakinessPatternRecord | null> {
|
|
483
|
-
return new Promise((resolve, reject) => {
|
|
484
|
-
this.db.get(
|
|
485
|
-
'SELECT * FROM flakiness_patterns WHERE test_id = ? AND pattern_type = ?',
|
|
486
|
-
[testId, patternType],
|
|
487
|
-
(err: Error | null, row: any) => {
|
|
488
|
-
if (err) reject(err);
|
|
489
|
-
else resolve(row as FlakinessPatternRecord || null);
|
|
490
|
-
}
|
|
491
|
-
);
|
|
492
|
-
});
|
|
493
|
-
}
|
|
494
|
-
|
|
495
|
-
/**
|
|
496
|
-
* Get flakiness trends for a test
|
|
497
|
-
*/
|
|
498
|
-
async getFlakinessTrends(testId: string, days: number = 30): Promise<{
|
|
499
|
-
testId: string;
|
|
500
|
-
currentScore: number;
|
|
501
|
-
averageScore: number;
|
|
502
|
-
trend: 'improving' | 'stable' | 'degrading';
|
|
503
|
-
dataPoints: Array<{ date: number; score: number }>;
|
|
504
|
-
}> {
|
|
505
|
-
const cutoff = Date.now() - (days * 24 * 60 * 60 * 1000);
|
|
506
|
-
|
|
507
|
-
return new Promise((resolve, reject) => {
|
|
508
|
-
this.db.all(
|
|
509
|
-
`SELECT created_at, score FROM flakiness_history WHERE test_id = ? AND created_at >= ? ORDER BY created_at ASC`,
|
|
510
|
-
[testId, cutoff],
|
|
511
|
-
(err: Error | null, rows: any[]) => {
|
|
512
|
-
if (err) {
|
|
513
|
-
reject(err);
|
|
514
|
-
} else {
|
|
515
|
-
const rawPoints = rows as Array<{ created_at: number; score: number }>;
|
|
516
|
-
// Convert to expected format with 'date' property
|
|
517
|
-
const dataPoints = rawPoints.map(p => ({ date: p.created_at, score: p.score }));
|
|
518
|
-
|
|
519
|
-
if (dataPoints.length === 0) {
|
|
520
|
-
resolve({
|
|
521
|
-
testId,
|
|
522
|
-
currentScore: 100,
|
|
523
|
-
averageScore: 100,
|
|
524
|
-
trend: 'stable',
|
|
525
|
-
dataPoints: []
|
|
526
|
-
});
|
|
527
|
-
return;
|
|
528
|
-
}
|
|
529
|
-
|
|
530
|
-
const currentScore = dataPoints[dataPoints.length - 1].score;
|
|
531
|
-
const averageScore = Math.round(dataPoints.reduce((sum, p) => sum + p.score, 0) / dataPoints.length);
|
|
532
|
-
|
|
533
|
-
// Calculate trend
|
|
534
|
-
let trend: 'improving' | 'stable' | 'degrading' = 'stable';
|
|
535
|
-
if (dataPoints.length >= 3) {
|
|
536
|
-
const recent = dataPoints.slice(-3);
|
|
537
|
-
const older = dataPoints.slice(0, -3);
|
|
538
|
-
const recentAvg = recent.reduce((sum, p) => sum + p.score, 0) / recent.length;
|
|
539
|
-
const olderAvg = older.reduce((sum, p) => sum + p.score, 0) / older.length;
|
|
540
|
-
|
|
541
|
-
if (recentAvg > olderAvg + 5) trend = 'improving';
|
|
542
|
-
else if (recentAvg < olderAvg - 5) trend = 'degrading';
|
|
543
|
-
}
|
|
544
|
-
|
|
545
|
-
resolve({
|
|
546
|
-
testId,
|
|
547
|
-
currentScore,
|
|
548
|
-
averageScore,
|
|
549
|
-
trend,
|
|
550
|
-
dataPoints
|
|
551
|
-
});
|
|
552
|
-
}
|
|
553
|
-
}
|
|
554
|
-
);
|
|
555
|
-
});
|
|
556
|
-
}
|
|
557
|
-
|
|
558
|
-
/**
|
|
559
|
-
* Store artifact in CAS and link to run
|
|
560
|
-
*/
|
|
561
|
-
async storeArtifact(runId: string, content: Buffer, mimeType: string, label: string, originalName?: string): Promise<CASArtifact> {
|
|
562
|
-
const casArtifact = await this.cas.saveArtifact(content, mimeType, originalName);
|
|
563
|
-
|
|
564
|
-
// Store artifact metadata
|
|
565
|
-
const artifact: ArtifactRecord = {
|
|
566
|
-
sha256: casArtifact.sha256,
|
|
567
|
-
mime_type: mimeType,
|
|
568
|
-
size_bytes: casArtifact.size,
|
|
569
|
-
cas_path: casArtifact.path,
|
|
570
|
-
original_name: originalName,
|
|
571
|
-
created_at: Date.now(),
|
|
572
|
-
last_accessed: Date.now()
|
|
573
|
-
};
|
|
574
|
-
|
|
575
|
-
await this.insertArtifact(artifact);
|
|
576
|
-
await this.attachArtifact(runId, { sha256: casArtifact.sha256, label });
|
|
577
|
-
|
|
578
|
-
console.log(`[VAULT] ARTIFACT_STORED: ${runId}/${label} (sha256: ${casArtifact.sha256.substring(0, 8)}..., size: ${casArtifact.size})`);
|
|
579
|
-
|
|
580
|
-
return casArtifact;
|
|
581
|
-
}
|
|
582
|
-
|
|
583
|
-
/**
|
|
584
|
-
* Attach existing artifact to run
|
|
585
|
-
*/
|
|
586
|
-
async attachArtifact(runId: string, link: { sha256: string; label: string }): Promise<void> {
|
|
587
|
-
const linkRecord: RunArtifactLink = {
|
|
588
|
-
run_id: runId,
|
|
589
|
-
sha256: link.sha256,
|
|
590
|
-
label: link.label,
|
|
591
|
-
created_at: Date.now()
|
|
592
|
-
};
|
|
593
|
-
|
|
594
|
-
await this.insertRunArtifact(linkRecord);
|
|
595
|
-
}
|
|
596
|
-
|
|
597
|
-
/**
|
|
598
|
-
* Get run by ID
|
|
599
|
-
*/
|
|
600
|
-
async getRun(runId: string): Promise<RunRecord | null> {
|
|
601
|
-
return new Promise((resolve, reject) => {
|
|
602
|
-
this.db.get(
|
|
603
|
-
'SELECT * FROM runs WHERE id = ?',
|
|
604
|
-
[runId],
|
|
605
|
-
(err: Error | null, row: any) => {
|
|
606
|
-
if (err) reject(err);
|
|
607
|
-
else resolve(row as RunRecord || null);
|
|
608
|
-
}
|
|
609
|
-
);
|
|
610
|
-
});
|
|
611
|
-
}
|
|
612
|
-
|
|
613
|
-
/**
|
|
614
|
-
* Get run by ID prefix (e.g., "6ce422f7" for shortened display)
|
|
615
|
-
* Returns null if multiple or no runs match the prefix
|
|
616
|
-
*/
|
|
617
|
-
async getRunByPrefix(prefix: string): Promise<RunRecord | null> {
|
|
618
|
-
const allRuns = await this.listRuns({});
|
|
619
|
-
const matching = allRuns.filter(r => r.id.startsWith(prefix));
|
|
620
|
-
|
|
621
|
-
if (matching.length === 0) {
|
|
622
|
-
return null;
|
|
623
|
-
}
|
|
624
|
-
if (matching.length > 1) {
|
|
625
|
-
// Ambiguous prefix - return null to indicate error
|
|
626
|
-
return null;
|
|
627
|
-
}
|
|
628
|
-
return matching[0];
|
|
629
|
-
}
|
|
630
|
-
|
|
631
|
-
/**
|
|
632
|
-
* Get run by run_key
|
|
633
|
-
*/
|
|
634
|
-
async getRunByKey(runKey: string): Promise<RunRecord | null> {
|
|
635
|
-
return new Promise((resolve, reject) => {
|
|
636
|
-
this.db.get(
|
|
637
|
-
'SELECT * FROM runs WHERE run_key = ?',
|
|
638
|
-
[runKey],
|
|
639
|
-
(err: Error | null, row: any) => {
|
|
640
|
-
if (err) reject(err);
|
|
641
|
-
else resolve(row as RunRecord || null);
|
|
642
|
-
}
|
|
643
|
-
);
|
|
644
|
-
});
|
|
645
|
-
}
|
|
646
|
-
|
|
647
|
-
/**
|
|
648
|
-
* List runs with filters
|
|
649
|
-
*/
|
|
650
|
-
async listRuns(filters?: RunFilters): Promise<RunRecord[]> {
|
|
651
|
-
let query = 'SELECT * FROM runs WHERE 1=1';
|
|
652
|
-
const params: any[] = [];
|
|
653
|
-
|
|
654
|
-
if (filters?.status?.length) {
|
|
655
|
-
query += ` AND status IN (${filters.status.map(() => '?').join(',')})`;
|
|
656
|
-
params.push(...filters.status);
|
|
657
|
-
}
|
|
658
|
-
|
|
659
|
-
if (filters?.since) {
|
|
660
|
-
query += ' AND started_at >= ?';
|
|
661
|
-
params.push(filters.since.getTime());
|
|
662
|
-
}
|
|
663
|
-
|
|
664
|
-
if (filters?.until) {
|
|
665
|
-
query += ' AND started_at <= ?';
|
|
666
|
-
params.push(filters.until.getTime());
|
|
667
|
-
}
|
|
668
|
-
|
|
669
|
-
if (filters?.pack_hash) {
|
|
670
|
-
query += ' AND pack_hash = ?';
|
|
671
|
-
params.push(filters.pack_hash);
|
|
672
|
-
}
|
|
673
|
-
|
|
674
|
-
query += ' ORDER BY started_at DESC';
|
|
675
|
-
|
|
676
|
-
if (filters?.limit) {
|
|
677
|
-
query += ' LIMIT ?';
|
|
678
|
-
params.push(filters.limit);
|
|
679
|
-
|
|
680
|
-
if (filters?.offset) {
|
|
681
|
-
query += ' OFFSET ?';
|
|
682
|
-
params.push(filters.offset);
|
|
683
|
-
}
|
|
684
|
-
}
|
|
685
|
-
|
|
686
|
-
return new Promise((resolve, reject) => {
|
|
687
|
-
this.db.all(query, params, (err: Error | null, rows: any) => {
|
|
688
|
-
if (err) reject(err);
|
|
689
|
-
else resolve(rows as RunRecord[]);
|
|
690
|
-
});
|
|
691
|
-
});
|
|
692
|
-
}
|
|
693
|
-
|
|
694
|
-
/**
|
|
695
|
-
* Get gates for a run
|
|
696
|
-
*/
|
|
697
|
-
async getGates(runId: string): Promise<GateRecord[]> {
|
|
698
|
-
return new Promise((resolve, reject) => {
|
|
699
|
-
this.db.all(
|
|
700
|
-
'SELECT * FROM gates WHERE run_id = ? ORDER BY created_at',
|
|
701
|
-
[runId],
|
|
702
|
-
(err: Error | null, rows: any) => {
|
|
703
|
-
if (err) reject(err);
|
|
704
|
-
else resolve(rows as GateRecord[]);
|
|
705
|
-
}
|
|
706
|
-
);
|
|
707
|
-
});
|
|
708
|
-
}
|
|
709
|
-
|
|
710
|
-
/**
|
|
711
|
-
* Get findings for a run
|
|
712
|
-
*/
|
|
713
|
-
async getFindings(runId: string, gate?: string): Promise<FindingRecord[]> {
|
|
714
|
-
let query = 'SELECT * FROM findings WHERE run_id = ?';
|
|
715
|
-
const params = [runId];
|
|
716
|
-
|
|
717
|
-
if (gate) {
|
|
718
|
-
query += ' AND gate = ?';
|
|
719
|
-
params.push(gate);
|
|
720
|
-
}
|
|
721
|
-
|
|
722
|
-
query += ' ORDER BY severity DESC, created_at';
|
|
723
|
-
|
|
724
|
-
return new Promise((resolve, reject) => {
|
|
725
|
-
this.db.all(query, params, (err: Error | null, rows: any) => {
|
|
726
|
-
if (err) reject(err);
|
|
727
|
-
else resolve(rows as FindingRecord[]);
|
|
728
|
-
});
|
|
729
|
-
});
|
|
730
|
-
}
|
|
731
|
-
|
|
732
|
-
/**
|
|
733
|
-
* Get artifacts for a run
|
|
734
|
-
*/
|
|
735
|
-
async getRunArtifacts(runId: string): Promise<Array<ArtifactRecord & { label: string }>> {
|
|
736
|
-
return new Promise((resolve, reject) => {
|
|
737
|
-
this.db.all(`
|
|
738
|
-
SELECT a.*, ra.label
|
|
739
|
-
FROM artifacts a
|
|
740
|
-
JOIN run_artifacts ra ON a.sha256 = ra.sha256
|
|
741
|
-
WHERE ra.run_id = ?
|
|
742
|
-
ORDER BY ra.created_at
|
|
743
|
-
`, [runId], (err: Error | null, rows: any) => {
|
|
744
|
-
if (err) reject(err);
|
|
745
|
-
else resolve(rows as Array<ArtifactRecord & { label: string }>);
|
|
746
|
-
});
|
|
747
|
-
});
|
|
748
|
-
}
|
|
749
|
-
|
|
750
|
-
/**
|
|
751
|
-
* Get artifact by SHA256
|
|
752
|
-
*/
|
|
753
|
-
async getArtifact(sha256: string): Promise<ArtifactRecord | null> {
|
|
754
|
-
return new Promise((resolve, reject) => {
|
|
755
|
-
this.db.get(
|
|
756
|
-
'SELECT * FROM artifacts WHERE sha256 = ?',
|
|
757
|
-
[sha256],
|
|
758
|
-
(err: Error | null, row: any) => {
|
|
759
|
-
if (err) reject(err);
|
|
760
|
-
else resolve(row as ArtifactRecord || null);
|
|
761
|
-
}
|
|
762
|
-
);
|
|
763
|
-
});
|
|
764
|
-
}
|
|
765
|
-
|
|
766
|
-
/**
|
|
767
|
-
* Pin/unpin run to protect from GC
|
|
768
|
-
*/
|
|
769
|
-
async pinRun(runId: string, pinned: boolean = true): Promise<void> {
|
|
770
|
-
await this.updateRun(runId, { pinned: pinned ? 1 : 0 });
|
|
771
|
-
console.log(`[VAULT] RUN_${pinned ? 'PINNED' : 'UNPINNED'}: ${runId}`);
|
|
772
|
-
}
|
|
773
|
-
|
|
774
|
-
/**
|
|
775
|
-
* Calculate pack hash for idempotence
|
|
776
|
-
*/
|
|
777
|
-
static calculatePackHash(packContent: string): string {
|
|
778
|
-
// Normalize pack content (remove comments, sort keys, etc.)
|
|
779
|
-
const normalized = packContent
|
|
780
|
-
.split('\n')
|
|
781
|
-
.filter(line => !line.trim().startsWith('#'))
|
|
782
|
-
.join('\n')
|
|
783
|
-
.trim();
|
|
784
|
-
|
|
785
|
-
return createHash('sha256').update(normalized).digest('hex');
|
|
786
|
-
}
|
|
787
|
-
|
|
788
|
-
/**
|
|
789
|
-
* Get vault statistics
|
|
790
|
-
*/
|
|
791
|
-
async getStats(): Promise<VaultStats> {
|
|
792
|
-
const queries = [
|
|
793
|
-
'SELECT COUNT(*) as count FROM runs',
|
|
794
|
-
'SELECT COUNT(*) as count FROM gates',
|
|
795
|
-
'SELECT COUNT(*) as count FROM findings',
|
|
796
|
-
'SELECT COUNT(*) as count FROM artifacts',
|
|
797
|
-
'SELECT MIN(started_at) as oldest, MAX(started_at) as newest FROM runs'
|
|
798
|
-
];
|
|
799
|
-
|
|
800
|
-
const results = await Promise.all(
|
|
801
|
-
queries.map(query => new Promise<any>((resolve, reject) => {
|
|
802
|
-
this.db.get(query, (err: Error | null, row: any) => {
|
|
803
|
-
if (err) reject(err);
|
|
804
|
-
else resolve(row);
|
|
805
|
-
});
|
|
806
|
-
}))
|
|
807
|
-
);
|
|
808
|
-
|
|
809
|
-
const casStats = await this.cas.getStats();
|
|
810
|
-
|
|
811
|
-
return {
|
|
812
|
-
totalRuns: results[0].count,
|
|
813
|
-
totalGates: results[1].count,
|
|
814
|
-
totalFindings: results[2].count,
|
|
815
|
-
totalArtifacts: results[3].count,
|
|
816
|
-
vaultSizeBytes: casStats.totalBytes,
|
|
817
|
-
oldestRun: results[4].oldest ? new Date(results[4].oldest) : undefined,
|
|
818
|
-
newestRun: results[4].newest ? new Date(results[4].newest) : undefined
|
|
819
|
-
};
|
|
820
|
-
}
|
|
821
|
-
|
|
822
|
-
// Private helper methods
|
|
823
|
-
|
|
824
|
-
private async insertRun(run: Partial<RunRecord>): Promise<void> {
|
|
825
|
-
const fields = Object.keys(run).join(', ');
|
|
826
|
-
const placeholders = Object.keys(run).map(() => '?').join(', ');
|
|
827
|
-
const values = Object.values(run);
|
|
828
|
-
|
|
829
|
-
return new Promise((resolve, reject) => {
|
|
830
|
-
this.db.run(
|
|
831
|
-
`INSERT INTO runs (${fields}) VALUES (${placeholders})`,
|
|
832
|
-
values,
|
|
833
|
-
(err: Error | null) => {
|
|
834
|
-
if (err) reject(err);
|
|
835
|
-
else resolve();
|
|
836
|
-
}
|
|
837
|
-
);
|
|
838
|
-
});
|
|
839
|
-
}
|
|
840
|
-
|
|
841
|
-
private async updateRun(runId: string, updates: Partial<RunRecord>): Promise<void> {
|
|
842
|
-
const fields = Object.keys(updates).map(key => `${key} = ?`).join(', ');
|
|
843
|
-
const values = [...Object.values(updates), runId];
|
|
844
|
-
|
|
845
|
-
return new Promise((resolve, reject) => {
|
|
846
|
-
this.db.run(
|
|
847
|
-
`UPDATE runs SET ${fields} WHERE id = ?`,
|
|
848
|
-
values,
|
|
849
|
-
(err: Error | null) => {
|
|
850
|
-
if (err) reject(err);
|
|
851
|
-
else resolve();
|
|
852
|
-
}
|
|
853
|
-
);
|
|
854
|
-
});
|
|
855
|
-
}
|
|
856
|
-
|
|
857
|
-
private async insertGate(gate: Omit<GateRecord, 'id'>): Promise<number> {
|
|
858
|
-
const fields = Object.keys(gate).join(', ');
|
|
859
|
-
const placeholders = Object.keys(gate).map(() => '?').join(', ');
|
|
860
|
-
const values = Object.values(gate);
|
|
861
|
-
|
|
862
|
-
return new Promise((resolve, reject) => {
|
|
863
|
-
this.db.run(
|
|
864
|
-
`INSERT INTO gates (${fields}) VALUES (${placeholders})`,
|
|
865
|
-
values,
|
|
866
|
-
function(err) {
|
|
867
|
-
if (err) reject(err);
|
|
868
|
-
else resolve(this.lastID);
|
|
869
|
-
}
|
|
870
|
-
);
|
|
871
|
-
});
|
|
872
|
-
}
|
|
873
|
-
|
|
874
|
-
private async insertFinding(finding: Omit<FindingRecord, 'id'>): Promise<number> {
|
|
875
|
-
const fields = Object.keys(finding).join(', ');
|
|
876
|
-
const placeholders = Object.keys(finding).map(() => '?').join(', ');
|
|
877
|
-
const values = Object.values(finding);
|
|
878
|
-
|
|
879
|
-
return new Promise((resolve, reject) => {
|
|
880
|
-
this.db.run(
|
|
881
|
-
`INSERT INTO findings (${fields}) VALUES (${placeholders})`,
|
|
882
|
-
values,
|
|
883
|
-
function(err) {
|
|
884
|
-
if (err) reject(err);
|
|
885
|
-
else resolve(this.lastID);
|
|
886
|
-
}
|
|
887
|
-
);
|
|
888
|
-
});
|
|
889
|
-
}
|
|
890
|
-
|
|
891
|
-
private async insertArtifact(artifact: ArtifactRecord): Promise<void> {
|
|
892
|
-
return new Promise((resolve, reject) => {
|
|
893
|
-
this.db.run(
|
|
894
|
-
`INSERT OR IGNORE INTO artifacts (sha256, mime_type, size_bytes, cas_path, original_name, created_at, last_accessed)
|
|
895
|
-
VALUES (?, ?, ?, ?, ?, ?, ?)`,
|
|
896
|
-
[
|
|
897
|
-
artifact.sha256,
|
|
898
|
-
artifact.mime_type,
|
|
899
|
-
artifact.size_bytes,
|
|
900
|
-
artifact.cas_path,
|
|
901
|
-
artifact.original_name,
|
|
902
|
-
artifact.created_at,
|
|
903
|
-
artifact.last_accessed
|
|
904
|
-
],
|
|
905
|
-
(err: Error | null) => {
|
|
906
|
-
if (err) reject(err);
|
|
907
|
-
else resolve();
|
|
908
|
-
}
|
|
909
|
-
);
|
|
910
|
-
});
|
|
911
|
-
}
|
|
912
|
-
|
|
913
|
-
private async insertRunArtifact(link: RunArtifactLink): Promise<void> {
|
|
914
|
-
return new Promise((resolve, reject) => {
|
|
915
|
-
this.db.run(
|
|
916
|
-
'INSERT OR IGNORE INTO run_artifacts (run_id, sha256, label, created_at) VALUES (?, ?, ?, ?)',
|
|
917
|
-
[link.run_id, link.sha256, link.label, link.created_at],
|
|
918
|
-
(err: Error | null) => {
|
|
919
|
-
if (err) reject(err);
|
|
920
|
-
else resolve();
|
|
921
|
-
}
|
|
922
|
-
);
|
|
923
|
-
});
|
|
924
|
-
}
|
|
925
|
-
|
|
926
|
-
private async insertFlakiness(flakiness: Omit<FlakinessRecord, 'id'>): Promise<number> {
|
|
927
|
-
const fields = Object.keys(flakiness).join(', ');
|
|
928
|
-
const placeholders = Object.keys(flakiness).map(() => '?').join(', ');
|
|
929
|
-
const values = Object.values(flakiness);
|
|
930
|
-
|
|
931
|
-
return new Promise((resolve, reject) => {
|
|
932
|
-
this.db.run(
|
|
933
|
-
`INSERT INTO flakiness_history (${fields}) VALUES (${placeholders})`,
|
|
934
|
-
values,
|
|
935
|
-
function(err) {
|
|
936
|
-
if (err) reject(err);
|
|
937
|
-
else resolve(this.lastID);
|
|
938
|
-
}
|
|
939
|
-
);
|
|
940
|
-
});
|
|
941
|
-
}
|
|
942
|
-
|
|
943
|
-
private async insertFlakinessPattern(pattern: Omit<FlakinessPatternRecord, 'id'>): Promise<number> {
|
|
944
|
-
const fields = Object.keys(pattern).join(', ');
|
|
945
|
-
const placeholders = Object.keys(pattern).map(() => '?').join(', ');
|
|
946
|
-
const values = Object.values(pattern);
|
|
947
|
-
|
|
948
|
-
return new Promise((resolve, reject) => {
|
|
949
|
-
this.db.run(
|
|
950
|
-
`INSERT INTO flakiness_patterns (${fields}) VALUES (${placeholders})`,
|
|
951
|
-
values,
|
|
952
|
-
function(err) {
|
|
953
|
-
if (err) reject(err);
|
|
954
|
-
else resolve(this.lastID);
|
|
955
|
-
}
|
|
956
|
-
);
|
|
957
|
-
});
|
|
958
|
-
}
|
|
959
|
-
|
|
960
|
-
private async updateFlakinessPattern(id: number, updates: Partial<FlakinessPatternRecord>): Promise<void> {
|
|
961
|
-
const fields = Object.keys(updates).map(key => `${key} = ?`).join(', ');
|
|
962
|
-
const values = [...Object.values(updates), id];
|
|
963
|
-
|
|
964
|
-
return new Promise((resolve, reject) => {
|
|
965
|
-
this.db.run(
|
|
966
|
-
`UPDATE flakiness_patterns SET ${fields} WHERE id = ?`,
|
|
967
|
-
values,
|
|
968
|
-
(err: Error | null) => {
|
|
969
|
-
if (err) reject(err);
|
|
970
|
-
else resolve();
|
|
971
|
-
}
|
|
972
|
-
);
|
|
973
|
-
});
|
|
974
|
-
}
|
|
975
|
-
|
|
976
|
-
private async insertQuarantine(quarantine: Omit<QuarantineRecord, 'id'>): Promise<number> {
|
|
977
|
-
const fields = Object.keys(quarantine).join(', ');
|
|
978
|
-
const placeholders = Object.keys(quarantine).map(() => '?').join(', ');
|
|
979
|
-
const values = Object.values(quarantine);
|
|
980
|
-
|
|
981
|
-
return new Promise((resolve, reject) => {
|
|
982
|
-
this.db.run(
|
|
983
|
-
`INSERT INTO flakiness_quarantine (${fields}) VALUES (${placeholders})`,
|
|
984
|
-
values,
|
|
985
|
-
function(err) {
|
|
986
|
-
if (err) reject(err);
|
|
987
|
-
else resolve(this.lastID);
|
|
988
|
-
}
|
|
989
|
-
);
|
|
990
|
-
});
|
|
991
|
-
}
|
|
992
|
-
|
|
993
|
-
private async updateQuarantine(testId: string, updates: Partial<QuarantineRecord>): Promise<void> {
|
|
994
|
-
const fields = Object.keys(updates).map(key => `${key} = ?`).join(', ');
|
|
995
|
-
const values = [...Object.values(updates), testId];
|
|
996
|
-
|
|
997
|
-
return new Promise((resolve, reject) => {
|
|
998
|
-
this.db.run(
|
|
999
|
-
`UPDATE flakiness_quarantine SET ${fields} WHERE test_id = ?`,
|
|
1000
|
-
values,
|
|
1001
|
-
(err: Error | null) => {
|
|
1002
|
-
if (err) reject(err);
|
|
1003
|
-
else resolve();
|
|
1004
|
-
}
|
|
1005
|
-
);
|
|
1006
|
-
});
|
|
1007
|
-
}
|
|
1008
|
-
|
|
1009
|
-
private generateFindingFingerprint(finding: Omit<FindingRecord, 'id' | 'run_id' | 'created_at'>): string {
|
|
1010
|
-
const content = `${finding.gate}:${finding.rule}:${finding.location || ''}:${finding.message}`;
|
|
1011
|
-
return createHash('sha256').update(content).digest('hex').substring(0, 16);
|
|
1012
|
-
}
|
|
1013
|
-
|
|
1014
|
-
/**
|
|
1015
|
-
* Initialize database connection
|
|
1016
|
-
*/
|
|
1017
|
-
private initializeDatabase(config: VaultConfig): void {
|
|
1018
|
-
this.db = new Database(this.dbPath);
|
|
1019
|
-
|
|
1020
|
-
if (config.enableWAL) {
|
|
1021
|
-
this.db.exec('PRAGMA journal_mode = WAL');
|
|
1022
|
-
this.db.exec('PRAGMA synchronous = NORMAL');
|
|
1023
|
-
}
|
|
1024
|
-
|
|
1025
|
-
this.db.exec('PRAGMA foreign_keys = ON');
|
|
1026
|
-
this.db.exec('PRAGMA cache_size = 10000');
|
|
1027
|
-
this.db.exec('PRAGMA temp_store = MEMORY');
|
|
1028
|
-
}
|
|
1029
|
-
|
|
1030
|
-
/**
|
|
1031
|
-
* Initialize database schema
|
|
1032
|
-
*/
|
|
1033
|
-
async initializeSchema(): Promise<void> {
|
|
1034
|
-
const schema = `
|
|
1035
|
-
-- QA360 Evidence Vault Schema
|
|
1036
|
-
-- SQLite WAL mode for concurrent access and performance
|
|
1037
|
-
|
|
1038
|
-
-- Runs table: core execution records
|
|
1039
|
-
CREATE TABLE IF NOT EXISTS runs (
|
|
1040
|
-
id TEXT PRIMARY KEY,
|
|
1041
|
-
run_key TEXT UNIQUE,
|
|
1042
|
-
started_at INTEGER NOT NULL,
|
|
1043
|
-
ended_at INTEGER,
|
|
1044
|
-
pack_hash TEXT NOT NULL,
|
|
1045
|
-
pack_path TEXT NOT NULL,
|
|
1046
|
-
status TEXT NOT NULL DEFAULT 'running',
|
|
1047
|
-
trust_score INTEGER,
|
|
1048
|
-
weights_json TEXT,
|
|
1049
|
-
sign_alg TEXT DEFAULT 'ed25519',
|
|
1050
|
-
signature_hex TEXT,
|
|
1051
|
-
proof_pdf_path TEXT,
|
|
1052
|
-
pinned INTEGER DEFAULT 0,
|
|
1053
|
-
created_at INTEGER NOT NULL DEFAULT (strftime('%s', 'now') * 1000),
|
|
1054
|
-
updated_at INTEGER NOT NULL DEFAULT (strftime('%s', 'now') * 1000)
|
|
1055
|
-
);
|
|
1056
|
-
|
|
1057
|
-
-- Gates table: individual test gate results
|
|
1058
|
-
CREATE TABLE IF NOT EXISTS gates (
|
|
1059
|
-
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
1060
|
-
run_id TEXT NOT NULL REFERENCES runs(id) ON DELETE CASCADE,
|
|
1061
|
-
name TEXT NOT NULL,
|
|
1062
|
-
status TEXT NOT NULL,
|
|
1063
|
-
duration_ms INTEGER,
|
|
1064
|
-
metrics_json TEXT,
|
|
1065
|
-
budgets_json TEXT,
|
|
1066
|
-
started_at INTEGER,
|
|
1067
|
-
ended_at INTEGER,
|
|
1068
|
-
created_at INTEGER NOT NULL DEFAULT (strftime('%s', 'now') * 1000)
|
|
1069
|
-
);
|
|
1070
|
-
|
|
1071
|
-
-- Findings table: security and quality findings
|
|
1072
|
-
CREATE TABLE IF NOT EXISTS findings (
|
|
1073
|
-
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
1074
|
-
run_id TEXT NOT NULL REFERENCES runs(id) ON DELETE CASCADE,
|
|
1075
|
-
gate TEXT NOT NULL,
|
|
1076
|
-
severity TEXT NOT NULL,
|
|
1077
|
-
rule TEXT NOT NULL,
|
|
1078
|
-
location TEXT,
|
|
1079
|
-
message TEXT,
|
|
1080
|
-
raw_output TEXT,
|
|
1081
|
-
fingerprint TEXT,
|
|
1082
|
-
created_at INTEGER NOT NULL DEFAULT (strftime('%s', 'now') * 1000)
|
|
1083
|
-
);
|
|
1084
|
-
|
|
1085
|
-
-- Artifacts table: Content-Addressable Storage metadata
|
|
1086
|
-
CREATE TABLE IF NOT EXISTS artifacts (
|
|
1087
|
-
sha256 TEXT PRIMARY KEY,
|
|
1088
|
-
mime_type TEXT NOT NULL,
|
|
1089
|
-
size_bytes INTEGER NOT NULL,
|
|
1090
|
-
cas_path TEXT NOT NULL,
|
|
1091
|
-
original_name TEXT,
|
|
1092
|
-
created_at INTEGER NOT NULL DEFAULT (strftime('%s', 'now') * 1000),
|
|
1093
|
-
last_accessed INTEGER NOT NULL DEFAULT (strftime('%s', 'now') * 1000)
|
|
1094
|
-
);
|
|
1095
|
-
|
|
1096
|
-
-- Run artifacts table: N-N relationship between runs and artifacts
|
|
1097
|
-
CREATE TABLE IF NOT EXISTS run_artifacts (
|
|
1098
|
-
run_id TEXT NOT NULL REFERENCES runs(id) ON DELETE CASCADE,
|
|
1099
|
-
sha256 TEXT NOT NULL REFERENCES artifacts(sha256) ON DELETE CASCADE,
|
|
1100
|
-
label TEXT NOT NULL,
|
|
1101
|
-
created_at INTEGER NOT NULL DEFAULT (strftime('%s', 'now') * 1000),
|
|
1102
|
-
PRIMARY KEY (run_id, sha256, label)
|
|
1103
|
-
);
|
|
1104
|
-
|
|
1105
|
-
-- Flakiness history table: tracks test reliability over time
|
|
1106
|
-
CREATE TABLE IF NOT EXISTS flakiness_history (
|
|
1107
|
-
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
1108
|
-
run_id TEXT NOT NULL REFERENCES runs(id) ON DELETE CASCADE,
|
|
1109
|
-
test_id TEXT NOT NULL,
|
|
1110
|
-
test_name TEXT NOT NULL,
|
|
1111
|
-
gate TEXT NOT NULL,
|
|
1112
|
-
file_path TEXT NOT NULL,
|
|
1113
|
-
score INTEGER NOT NULL CHECK (score >= 0 AND score <= 100),
|
|
1114
|
-
category TEXT NOT NULL CHECK (category IN ('legendary', 'solid', 'good', 'shaky', 'unstable')),
|
|
1115
|
-
total_runs INTEGER NOT NULL,
|
|
1116
|
-
successful_runs INTEGER NOT NULL,
|
|
1117
|
-
avg_duration_ms INTEGER NOT NULL,
|
|
1118
|
-
pattern_type TEXT,
|
|
1119
|
-
suggested_fix TEXT,
|
|
1120
|
-
confidence REAL,
|
|
1121
|
-
first_seen INTEGER NOT NULL,
|
|
1122
|
-
last_seen INTEGER NOT NULL,
|
|
1123
|
-
current_streak INTEGER,
|
|
1124
|
-
streak_type TEXT CHECK (streak_type IN ('pass', 'fail')),
|
|
1125
|
-
created_at INTEGER NOT NULL DEFAULT (strftime('%s', 'now') * 1000)
|
|
1126
|
-
);
|
|
1127
|
-
|
|
1128
|
-
-- Flakiness patterns table: detected patterns for flaky tests
|
|
1129
|
-
CREATE TABLE IF NOT EXISTS flakiness_patterns (
|
|
1130
|
-
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
1131
|
-
test_id TEXT NOT NULL,
|
|
1132
|
-
test_name TEXT NOT NULL,
|
|
1133
|
-
pattern_type TEXT NOT NULL CHECK (pattern_type IN ('timing', 'race_condition', 'external_dependency', 'environment_specific', 'selector_issue', 'unknown')),
|
|
1134
|
-
description TEXT NOT NULL,
|
|
1135
|
-
suggested_fix TEXT NOT NULL,
|
|
1136
|
-
confidence REAL NOT NULL CHECK (confidence >= 0 AND confidence <= 1),
|
|
1137
|
-
auto_fix_possible INTEGER NOT NULL DEFAULT 0,
|
|
1138
|
-
first_detected INTEGER NOT NULL,
|
|
1139
|
-
last_detected INTEGER NOT NULL,
|
|
1140
|
-
detection_count INTEGER NOT NULL DEFAULT 1,
|
|
1141
|
-
created_at INTEGER NOT NULL DEFAULT (strftime('%s', 'now') * 1000),
|
|
1142
|
-
updated_at INTEGER NOT NULL DEFAULT (strftime('%s', 'now') * 1000),
|
|
1143
|
-
UNIQUE(test_id, pattern_type)
|
|
1144
|
-
);
|
|
1145
|
-
|
|
1146
|
-
-- Flakiness quarantine table: tests quarantined for flakiness
|
|
1147
|
-
CREATE TABLE IF NOT EXISTS flakiness_quarantine (
|
|
1148
|
-
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
1149
|
-
test_id TEXT NOT NULL UNIQUE,
|
|
1150
|
-
test_name TEXT NOT NULL,
|
|
1151
|
-
gate TEXT NOT NULL,
|
|
1152
|
-
file_path TEXT NOT NULL,
|
|
1153
|
-
reason TEXT NOT NULL,
|
|
1154
|
-
score INTEGER NOT NULL CHECK (score >= 0 AND score <= 100),
|
|
1155
|
-
category TEXT NOT NULL CHECK (category IN ('shaky', 'unstable')),
|
|
1156
|
-
quarantined_at INTEGER NOT NULL,
|
|
1157
|
-
quarantined_by TEXT NOT NULL,
|
|
1158
|
-
resolved_at INTEGER,
|
|
1159
|
-
resolved_by TEXT,
|
|
1160
|
-
notes TEXT,
|
|
1161
|
-
created_at INTEGER NOT NULL DEFAULT (strftime('%s', 'now') * 1000),
|
|
1162
|
-
updated_at INTEGER NOT NULL DEFAULT (strftime('%s', 'now') * 1000)
|
|
1163
|
-
);
|
|
1164
|
-
|
|
1165
|
-
-- Vault metadata table: schema version and configuration
|
|
1166
|
-
CREATE TABLE IF NOT EXISTS vault_metadata (
|
|
1167
|
-
key TEXT PRIMARY KEY,
|
|
1168
|
-
value TEXT NOT NULL,
|
|
1169
|
-
updated_at INTEGER NOT NULL DEFAULT (strftime('%s', 'now') * 1000)
|
|
1170
|
-
);
|
|
1171
|
-
|
|
1172
|
-
-- Indexes for performance
|
|
1173
|
-
CREATE INDEX IF NOT EXISTS idx_runs_started_at ON runs(started_at DESC);
|
|
1174
|
-
CREATE INDEX IF NOT EXISTS idx_runs_run_key ON runs(run_key) WHERE run_key IS NOT NULL;
|
|
1175
|
-
CREATE INDEX IF NOT EXISTS idx_runs_status ON runs(status);
|
|
1176
|
-
CREATE INDEX IF NOT EXISTS idx_runs_pack_hash ON runs(pack_hash);
|
|
1177
|
-
CREATE INDEX IF NOT EXISTS idx_runs_pinned ON runs(pinned) WHERE pinned = 1;
|
|
1178
|
-
|
|
1179
|
-
CREATE INDEX IF NOT EXISTS idx_gates_run_id ON gates(run_id);
|
|
1180
|
-
CREATE INDEX IF NOT EXISTS idx_gates_run_name ON gates(run_id, name);
|
|
1181
|
-
CREATE INDEX IF NOT EXISTS idx_gates_status ON gates(status);
|
|
1182
|
-
CREATE INDEX IF NOT EXISTS idx_gates_name ON gates(name);
|
|
1183
|
-
|
|
1184
|
-
CREATE INDEX IF NOT EXISTS idx_findings_run_id ON findings(run_id);
|
|
1185
|
-
CREATE INDEX IF NOT EXISTS idx_findings_run_gate ON findings(run_id, gate);
|
|
1186
|
-
CREATE INDEX IF NOT EXISTS idx_findings_severity ON findings(severity);
|
|
1187
|
-
CREATE INDEX IF NOT EXISTS idx_findings_rule ON findings(rule);
|
|
1188
|
-
CREATE INDEX IF NOT EXISTS idx_findings_fingerprint ON findings(fingerprint);
|
|
1189
|
-
|
|
1190
|
-
CREATE INDEX IF NOT EXISTS idx_artifacts_mime_type ON artifacts(mime_type);
|
|
1191
|
-
CREATE INDEX IF NOT EXISTS idx_artifacts_size ON artifacts(size_bytes);
|
|
1192
|
-
CREATE INDEX IF NOT EXISTS idx_artifacts_created_at ON artifacts(created_at);
|
|
1193
|
-
CREATE INDEX IF NOT EXISTS idx_artifacts_last_accessed ON artifacts(last_accessed);
|
|
1194
|
-
|
|
1195
|
-
CREATE INDEX IF NOT EXISTS idx_run_artifacts_run_id ON run_artifacts(run_id);
|
|
1196
|
-
CREATE INDEX IF NOT EXISTS idx_run_artifacts_sha256 ON run_artifacts(sha256);
|
|
1197
|
-
CREATE INDEX IF NOT EXISTS idx_run_artifacts_label ON run_artifacts(label);
|
|
1198
|
-
|
|
1199
|
-
-- Flakiness indexes
|
|
1200
|
-
CREATE INDEX IF NOT EXISTS idx_flakiness_run_id ON flakiness_history(run_id);
|
|
1201
|
-
CREATE INDEX IF NOT EXISTS idx_flakiness_test_id ON flakiness_history(test_id);
|
|
1202
|
-
CREATE INDEX IF NOT EXISTS idx_flakiness_score ON flakiness_history(score);
|
|
1203
|
-
CREATE INDEX IF NOT EXISTS idx_flakiness_category ON flakiness_history(category);
|
|
1204
|
-
CREATE INDEX IF NOT EXISTS idx_flakiness_created_at ON flakiness_history(created_at DESC);
|
|
1205
|
-
|
|
1206
|
-
CREATE INDEX IF NOT EXISTS idx_flaky_patterns_test_id ON flakiness_patterns(test_id);
|
|
1207
|
-
CREATE INDEX IF NOT EXISTS idx_flaky_patterns_pattern_type ON flakiness_patterns(pattern_type);
|
|
1208
|
-
CREATE INDEX IF NOT EXISTS idx_flaky_patterns_detection_count ON flakiness_patterns(detection_count DESC);
|
|
1209
|
-
|
|
1210
|
-
CREATE INDEX IF NOT EXISTS idx_quarantine_test_id ON flakiness_quarantine(test_id);
|
|
1211
|
-
CREATE INDEX IF NOT EXISTS idx_quarantine_quarantined_at ON flakiness_quarantine(quarantined_at DESC);
|
|
1212
|
-
CREATE INDEX IF NOT EXISTS idx_quarantine_resolved_at ON flakiness_quarantine(resolved_at);
|
|
1213
|
-
CREATE INDEX IF NOT EXISTS idx_quarantine_category ON flakiness_quarantine(category);
|
|
1214
|
-
|
|
1215
|
-
-- Insert initial metadata
|
|
1216
|
-
INSERT OR IGNORE INTO vault_metadata (key, value) VALUES
|
|
1217
|
-
('schema_version', '2.0.0'),
|
|
1218
|
-
('created_at', strftime('%s', 'now') * 1000),
|
|
1219
|
-
('vault_format', 'qa360-evidence-vault-v2');
|
|
1220
|
-
|
|
1221
|
-
-- Triggers for updated_at timestamps
|
|
1222
|
-
CREATE TRIGGER IF NOT EXISTS trigger_runs_updated_at
|
|
1223
|
-
AFTER UPDATE ON runs
|
|
1224
|
-
BEGIN
|
|
1225
|
-
UPDATE runs SET updated_at = strftime('%s', 'now') * 1000 WHERE id = NEW.id;
|
|
1226
|
-
END;
|
|
1227
|
-
|
|
1228
|
-
CREATE TRIGGER IF NOT EXISTS trigger_artifacts_last_accessed
|
|
1229
|
-
AFTER UPDATE ON artifacts
|
|
1230
|
-
BEGIN
|
|
1231
|
-
UPDATE artifacts SET last_accessed = strftime('%s', 'now') * 1000 WHERE sha256 = NEW.sha256;
|
|
1232
|
-
END;
|
|
1233
|
-
|
|
1234
|
-
CREATE TRIGGER IF NOT EXISTS trigger_flakiness_patterns_updated_at
|
|
1235
|
-
AFTER UPDATE ON flakiness_patterns
|
|
1236
|
-
BEGIN
|
|
1237
|
-
UPDATE flakiness_patterns SET updated_at = strftime('%s', 'now') * 1000 WHERE id = NEW.id;
|
|
1238
|
-
END;
|
|
1239
|
-
|
|
1240
|
-
CREATE TRIGGER IF NOT EXISTS trigger_quarantine_updated_at
|
|
1241
|
-
AFTER UPDATE ON flakiness_quarantine
|
|
1242
|
-
BEGIN
|
|
1243
|
-
UPDATE flakiness_quarantine SET updated_at = strftime('%s', 'now') * 1000 WHERE id = NEW.id;
|
|
1244
|
-
END;
|
|
1245
|
-
|
|
1246
|
-
-- Views for common queries
|
|
1247
|
-
CREATE VIEW IF NOT EXISTS v_recent_runs AS
|
|
1248
|
-
SELECT
|
|
1249
|
-
r.id,
|
|
1250
|
-
r.run_key,
|
|
1251
|
-
r.started_at,
|
|
1252
|
-
r.ended_at,
|
|
1253
|
-
r.status,
|
|
1254
|
-
r.trust_score,
|
|
1255
|
-
r.pack_path,
|
|
1256
|
-
COUNT(g.id) as gate_count,
|
|
1257
|
-
COUNT(CASE WHEN g.status = 'passed' THEN 1 END) as gates_passed,
|
|
1258
|
-
COUNT(CASE WHEN g.status = 'failed' THEN 1 END) as gates_failed,
|
|
1259
|
-
COUNT(f.id) as finding_count,
|
|
1260
|
-
COUNT(CASE WHEN f.severity IN ('high', 'critical') THEN 1 END) as critical_findings
|
|
1261
|
-
FROM runs r
|
|
1262
|
-
LEFT JOIN gates g ON r.id = g.run_id
|
|
1263
|
-
LEFT JOIN findings f ON r.id = f.run_id
|
|
1264
|
-
GROUP BY r.id
|
|
1265
|
-
ORDER BY r.started_at DESC;
|
|
1266
|
-
|
|
1267
|
-
CREATE VIEW IF NOT EXISTS v_gate_trends AS
|
|
1268
|
-
SELECT
|
|
1269
|
-
g.name,
|
|
1270
|
-
g.status,
|
|
1271
|
-
r.started_at,
|
|
1272
|
-
r.trust_score,
|
|
1273
|
-
g.duration_ms,
|
|
1274
|
-
g.metrics_json
|
|
1275
|
-
FROM gates g
|
|
1276
|
-
JOIN runs r ON g.run_id = r.id
|
|
1277
|
-
WHERE r.status IN ('passed', 'failed')
|
|
1278
|
-
ORDER BY r.started_at DESC;
|
|
1279
|
-
|
|
1280
|
-
-- Flakiness summary view: aggregate flakiness stats by test
|
|
1281
|
-
CREATE VIEW IF NOT EXISTS v_flakiness_summary AS
|
|
1282
|
-
SELECT
|
|
1283
|
-
test_id,
|
|
1284
|
-
test_name,
|
|
1285
|
-
gate,
|
|
1286
|
-
file_path,
|
|
1287
|
-
MAX(score) as latest_score,
|
|
1288
|
-
MIN(score) as worst_score,
|
|
1289
|
-
AVG(score) as avg_score,
|
|
1290
|
-
COUNT(*) as total_analyses,
|
|
1291
|
-
MAX(category) as latest_category,
|
|
1292
|
-
MAX(last_seen) as last_analyzed,
|
|
1293
|
-
MAX(first_seen) as first_seen
|
|
1294
|
-
FROM flakiness_history
|
|
1295
|
-
GROUP BY test_id, test_name, gate, file_path
|
|
1296
|
-
ORDER BY latest_score ASC;
|
|
1297
|
-
|
|
1298
|
-
-- Flakiness trends view: time series analysis
|
|
1299
|
-
CREATE VIEW IF NOT EXISTS v_flakiness_trends AS
|
|
1300
|
-
SELECT
|
|
1301
|
-
DATE(created_at / 1000, 'unixepoch') as analysis_date,
|
|
1302
|
-
category,
|
|
1303
|
-
COUNT(*) as test_count,
|
|
1304
|
-
AVG(score) as avg_score
|
|
1305
|
-
FROM flakiness_history
|
|
1306
|
-
WHERE created_at >= strftime('%s', 'now', '-30 days') * 1000
|
|
1307
|
-
GROUP BY analysis_date, category
|
|
1308
|
-
ORDER BY analysis_date DESC, category DESC;
|
|
1309
|
-
|
|
1310
|
-
-- Quarantined tests view
|
|
1311
|
-
CREATE VIEW IF NOT EXISTS v_quarantined_tests AS
|
|
1312
|
-
SELECT
|
|
1313
|
-
q.*,
|
|
1314
|
-
fh.score as current_score,
|
|
1315
|
-
fh.category as current_category
|
|
1316
|
-
FROM flakiness_quarantine q
|
|
1317
|
-
LEFT JOIN (
|
|
1318
|
-
SELECT test_id, score, category
|
|
1319
|
-
FROM flakiness_history fh1
|
|
1320
|
-
WHERE fh1.created_at = (
|
|
1321
|
-
SELECT MAX(fh2.created_at)
|
|
1322
|
-
FROM flakiness_history fh2
|
|
1323
|
-
WHERE fh2.test_id = fh1.test_id
|
|
1324
|
-
)
|
|
1325
|
-
) fh ON q.test_id = fh.test_id
|
|
1326
|
-
WHERE q.resolved_at IS NULL
|
|
1327
|
-
ORDER BY q.quarantined_at DESC;
|
|
1328
|
-
`;
|
|
1329
|
-
|
|
1330
|
-
return new Promise((resolve, reject) => {
|
|
1331
|
-
this.db.exec(schema, (err: Error | null) => {
|
|
1332
|
-
if (err) {
|
|
1333
|
-
reject(new Error(`Failed to initialize schema: ${err.message}`));
|
|
1334
|
-
} else {
|
|
1335
|
-
resolve();
|
|
1336
|
-
}
|
|
1337
|
-
});
|
|
1338
|
-
});
|
|
1339
|
-
}
|
|
1340
|
-
|
|
1341
|
-
/**
|
|
1342
|
-
* Ensure required directories exist
|
|
1343
|
-
*/
|
|
1344
|
-
private ensureDirectories(): void {
|
|
1345
|
-
if (!existsSync(this.baseDir)) {
|
|
1346
|
-
mkdirSync(this.baseDir, { recursive: true });
|
|
1347
|
-
}
|
|
1348
|
-
}
|
|
1349
|
-
|
|
1350
|
-
/**
|
|
1351
|
-
* Close database connection
|
|
1352
|
-
*/
|
|
1353
|
-
async close(): Promise<void> {
|
|
1354
|
-
return new Promise((resolve, reject) => {
|
|
1355
|
-
this.db.close((err: Error | null) => {
|
|
1356
|
-
if (err) reject(err);
|
|
1357
|
-
else resolve();
|
|
1358
|
-
});
|
|
1359
|
-
});
|
|
1360
|
-
}
|
|
1361
|
-
}
|