@tyroneross/navgator 0.2.2 → 0.9.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.agents/plugins/marketplace.json +20 -0
- package/.claude-plugin/marketplace.json +21 -7
- package/.claude-plugin/plugin.json +16 -11
- package/.codex-plugin/plugin.json +31 -0
- package/.mcp.json +8 -0
- package/CLAUDE.md +197 -23
- package/LICENSE +202 -21
- package/README.md +220 -33
- package/agents/architecture-advisor.md +6 -2
- package/agents/architecture-investigator.md +163 -0
- package/agents/architecture-planner.md +160 -0
- package/agents/external-resolver.md +97 -0
- package/dist/__tests__/agent-output.test.d.ts +5 -0
- package/dist/__tests__/agent-output.test.d.ts.map +1 -0
- package/dist/__tests__/agent-output.test.js +233 -0
- package/dist/__tests__/agent-output.test.js.map +1 -0
- package/dist/__tests__/architecture-insights-stack.test.d.ts +21 -0
- package/dist/__tests__/architecture-insights-stack.test.d.ts.map +1 -0
- package/dist/__tests__/architecture-insights-stack.test.js +86 -0
- package/dist/__tests__/architecture-insights-stack.test.js.map +1 -0
- package/dist/__tests__/architecture-insights.test.d.ts +2 -0
- package/dist/__tests__/architecture-insights.test.d.ts.map +1 -0
- package/dist/__tests__/architecture-insights.test.js +46 -0
- package/dist/__tests__/architecture-insights.test.js.map +1 -0
- package/dist/__tests__/audit-sampler.test.d.ts +10 -0
- package/dist/__tests__/audit-sampler.test.d.ts.map +1 -0
- package/dist/__tests__/audit-sampler.test.js +172 -0
- package/dist/__tests__/audit-sampler.test.js.map +1 -0
- package/dist/__tests__/audit-spc.test.d.ts +5 -0
- package/dist/__tests__/audit-spc.test.d.ts.map +1 -0
- package/dist/__tests__/audit-spc.test.js +94 -0
- package/dist/__tests__/audit-spc.test.js.map +1 -0
- package/dist/__tests__/audit-verifiers.test.d.ts +5 -0
- package/dist/__tests__/audit-verifiers.test.d.ts.map +1 -0
- package/dist/__tests__/audit-verifiers.test.js +248 -0
- package/dist/__tests__/audit-verifiers.test.js.map +1 -0
- package/dist/__tests__/auto-refresh.test.d.ts +12 -0
- package/dist/__tests__/auto-refresh.test.d.ts.map +1 -0
- package/dist/__tests__/auto-refresh.test.js +236 -0
- package/dist/__tests__/auto-refresh.test.js.map +1 -0
- package/dist/__tests__/bare-imports.test.d.ts +8 -0
- package/dist/__tests__/bare-imports.test.d.ts.map +1 -0
- package/dist/__tests__/bare-imports.test.js +176 -0
- package/dist/__tests__/bare-imports.test.js.map +1 -0
- package/dist/__tests__/classify.test.d.ts +5 -0
- package/dist/__tests__/classify.test.d.ts.map +1 -0
- package/dist/__tests__/classify.test.js +158 -0
- package/dist/__tests__/classify.test.js.map +1 -0
- package/dist/__tests__/cli-commands.test.d.ts +8 -0
- package/dist/__tests__/cli-commands.test.d.ts.map +1 -0
- package/dist/__tests__/cli-commands.test.js +207 -0
- package/dist/__tests__/cli-commands.test.js.map +1 -0
- package/dist/__tests__/consolidated-readers.test.d.ts +23 -0
- package/dist/__tests__/consolidated-readers.test.d.ts.map +1 -0
- package/dist/__tests__/consolidated-readers.test.js +200 -0
- package/dist/__tests__/consolidated-readers.test.js.map +1 -0
- package/dist/__tests__/coverage.test.d.ts +5 -0
- package/dist/__tests__/coverage.test.d.ts.map +1 -0
- package/dist/__tests__/coverage.test.js +120 -0
- package/dist/__tests__/coverage.test.js.map +1 -0
- package/dist/__tests__/deploy-scanner-runtime.test.d.ts +6 -0
- package/dist/__tests__/deploy-scanner-runtime.test.d.ts.map +1 -0
- package/dist/__tests__/deploy-scanner-runtime.test.js +168 -0
- package/dist/__tests__/deploy-scanner-runtime.test.js.map +1 -0
- package/dist/__tests__/env-scanner.test.d.ts +2 -0
- package/dist/__tests__/env-scanner.test.d.ts.map +1 -0
- package/dist/__tests__/env-scanner.test.js +191 -0
- package/dist/__tests__/env-scanner.test.js.map +1 -0
- package/dist/__tests__/freshness/cli-freshness.test.d.ts +2 -0
- package/dist/__tests__/freshness/cli-freshness.test.d.ts.map +1 -0
- package/dist/__tests__/freshness/cli-freshness.test.js +26 -0
- package/dist/__tests__/freshness/cli-freshness.test.js.map +1 -0
- package/dist/__tests__/freshness/dirty-ledger.test.d.ts +2 -0
- package/dist/__tests__/freshness/dirty-ledger.test.d.ts.map +1 -0
- package/dist/__tests__/freshness/dirty-ledger.test.js +39 -0
- package/dist/__tests__/freshness/dirty-ledger.test.js.map +1 -0
- package/dist/__tests__/freshness/drainer.test.d.ts +2 -0
- package/dist/__tests__/freshness/drainer.test.d.ts.map +1 -0
- package/dist/__tests__/freshness/drainer.test.js +103 -0
- package/dist/__tests__/freshness/drainer.test.js.map +1 -0
- package/dist/__tests__/freshness/paths.test.d.ts +2 -0
- package/dist/__tests__/freshness/paths.test.d.ts.map +1 -0
- package/dist/__tests__/freshness/paths.test.js +19 -0
- package/dist/__tests__/freshness/paths.test.js.map +1 -0
- package/dist/__tests__/freshness/scan-lock.test.d.ts +2 -0
- package/dist/__tests__/freshness/scan-lock.test.d.ts.map +1 -0
- package/dist/__tests__/freshness/scan-lock.test.js +40 -0
- package/dist/__tests__/freshness/scan-lock.test.js.map +1 -0
- package/dist/__tests__/freshness/stamp.test.d.ts +2 -0
- package/dist/__tests__/freshness/stamp.test.d.ts.map +1 -0
- package/dist/__tests__/freshness/stamp.test.js +36 -0
- package/dist/__tests__/freshness/stamp.test.js.map +1 -0
- package/dist/__tests__/gitignore-safety.test.d.ts +2 -0
- package/dist/__tests__/gitignore-safety.test.d.ts.map +1 -0
- package/dist/__tests__/gitignore-safety.test.js +110 -0
- package/dist/__tests__/gitignore-safety.test.js.map +1 -0
- package/dist/__tests__/helpers.d.ts +37 -0
- package/dist/__tests__/helpers.d.ts.map +1 -0
- package/dist/__tests__/helpers.js +134 -0
- package/dist/__tests__/helpers.js.map +1 -0
- package/dist/__tests__/impact.test.d.ts +5 -0
- package/dist/__tests__/impact.test.d.ts.map +1 -0
- package/dist/__tests__/impact.test.js +221 -0
- package/dist/__tests__/impact.test.js.map +1 -0
- package/dist/__tests__/lessons-store.test.d.ts +8 -0
- package/dist/__tests__/lessons-store.test.d.ts.map +1 -0
- package/dist/__tests__/lessons-store.test.js +232 -0
- package/dist/__tests__/lessons-store.test.js.map +1 -0
- package/dist/__tests__/llm-dedup.test.d.ts +2 -0
- package/dist/__tests__/llm-dedup.test.d.ts.map +1 -0
- package/dist/__tests__/llm-dedup.test.js +155 -0
- package/dist/__tests__/llm-dedup.test.js.map +1 -0
- package/dist/__tests__/mjs-frontend-fetch.test.d.ts +19 -0
- package/dist/__tests__/mjs-frontend-fetch.test.d.ts.map +1 -0
- package/dist/__tests__/mjs-frontend-fetch.test.js +179 -0
- package/dist/__tests__/mjs-frontend-fetch.test.js.map +1 -0
- package/dist/__tests__/multi-stack-discovery.test.d.ts +11 -0
- package/dist/__tests__/multi-stack-discovery.test.d.ts.map +1 -0
- package/dist/__tests__/multi-stack-discovery.test.js +75 -0
- package/dist/__tests__/multi-stack-discovery.test.js.map +1 -0
- package/dist/__tests__/per-entity-files-gate.test.d.ts +22 -0
- package/dist/__tests__/per-entity-files-gate.test.d.ts.map +1 -0
- package/dist/__tests__/per-entity-files-gate.test.js +160 -0
- package/dist/__tests__/per-entity-files-gate.test.js.map +1 -0
- package/dist/__tests__/prisma-calls.test.d.ts +2 -0
- package/dist/__tests__/prisma-calls.test.d.ts.map +1 -0
- package/dist/__tests__/prisma-calls.test.js +125 -0
- package/dist/__tests__/prisma-calls.test.js.map +1 -0
- package/dist/__tests__/prisma-parser.test.d.ts +2 -0
- package/dist/__tests__/prisma-parser.test.d.ts.map +1 -0
- package/dist/__tests__/prisma-parser.test.js +252 -0
- package/dist/__tests__/prisma-parser.test.js.map +1 -0
- package/dist/__tests__/prompt-detector.test.d.ts +5 -0
- package/dist/__tests__/prompt-detector.test.d.ts.map +1 -0
- package/dist/__tests__/prompt-detector.test.js +75 -0
- package/dist/__tests__/prompt-detector.test.js.map +1 -0
- package/dist/__tests__/queue-scanner.test.d.ts +5 -0
- package/dist/__tests__/queue-scanner.test.d.ts.map +1 -0
- package/dist/__tests__/queue-scanner.test.js +85 -0
- package/dist/__tests__/queue-scanner.test.js.map +1 -0
- package/dist/__tests__/resolve.test.d.ts +5 -0
- package/dist/__tests__/resolve.test.d.ts.map +1 -0
- package/dist/__tests__/resolve.test.js +196 -0
- package/dist/__tests__/resolve.test.js.map +1 -0
- package/dist/__tests__/rules.test.d.ts +2 -0
- package/dist/__tests__/rules.test.d.ts.map +1 -0
- package/dist/__tests__/rules.test.js +343 -0
- package/dist/__tests__/rules.test.js.map +1 -0
- package/dist/__tests__/sandbox.test.d.ts +5 -0
- package/dist/__tests__/sandbox.test.d.ts.map +1 -0
- package/dist/__tests__/sandbox.test.js +189 -0
- package/dist/__tests__/sandbox.test.js.map +1 -0
- package/dist/__tests__/scanner-audit.test.d.ts +9 -0
- package/dist/__tests__/scanner-audit.test.d.ts.map +1 -0
- package/dist/__tests__/scanner-audit.test.js +64 -0
- package/dist/__tests__/scanner-audit.test.js.map +1 -0
- package/dist/__tests__/scanner-characterization.test.d.ts +16 -0
- package/dist/__tests__/scanner-characterization.test.d.ts.map +1 -0
- package/dist/__tests__/scanner-characterization.test.js +167 -0
- package/dist/__tests__/scanner-characterization.test.js.map +1 -0
- package/dist/__tests__/scanner-incremental.test.d.ts +13 -0
- package/dist/__tests__/scanner-incremental.test.d.ts.map +1 -0
- package/dist/__tests__/scanner-incremental.test.js +725 -0
- package/dist/__tests__/scanner-incremental.test.js.map +1 -0
- package/dist/__tests__/scanner-integration.test.d.ts +7 -0
- package/dist/__tests__/scanner-integration.test.d.ts.map +1 -0
- package/dist/__tests__/scanner-integration.test.js +211 -0
- package/dist/__tests__/scanner-integration.test.js.map +1 -0
- package/dist/__tests__/scip-new-catches.test.d.ts +19 -0
- package/dist/__tests__/scip-new-catches.test.d.ts.map +1 -0
- package/dist/__tests__/scip-new-catches.test.js +90 -0
- package/dist/__tests__/scip-new-catches.test.js.map +1 -0
- package/dist/__tests__/subgraph.test.d.ts +5 -0
- package/dist/__tests__/subgraph.test.d.ts.map +1 -0
- package/dist/__tests__/subgraph.test.js +145 -0
- package/dist/__tests__/subgraph.test.js.map +1 -0
- package/dist/__tests__/trace.test.d.ts +5 -0
- package/dist/__tests__/trace.test.d.ts.map +1 -0
- package/dist/__tests__/trace.test.js +221 -0
- package/dist/__tests__/trace.test.js.map +1 -0
- package/dist/agent-output.d.ts +16 -0
- package/dist/agent-output.d.ts.map +1 -0
- package/dist/agent-output.js +142 -0
- package/dist/agent-output.js.map +1 -0
- package/dist/architecture-insights.d.ts +17 -0
- package/dist/architecture-insights.d.ts.map +1 -0
- package/dist/architecture-insights.js +178 -0
- package/dist/architecture-insights.js.map +1 -0
- package/dist/audit/index.d.ts +69 -0
- package/dist/audit/index.d.ts.map +1 -0
- package/dist/audit/index.js +255 -0
- package/dist/audit/index.js.map +1 -0
- package/dist/audit/sampler.d.ts +98 -0
- package/dist/audit/sampler.d.ts.map +1 -0
- package/dist/audit/sampler.js +298 -0
- package/dist/audit/sampler.js.map +1 -0
- package/dist/audit/spc.d.ts +62 -0
- package/dist/audit/spc.d.ts.map +1 -0
- package/dist/audit/spc.js +81 -0
- package/dist/audit/spc.js.map +1 -0
- package/dist/audit/verifiers.d.ts +81 -0
- package/dist/audit/verifiers.d.ts.map +1 -0
- package/dist/audit/verifiers.js +366 -0
- package/dist/audit/verifiers.js.map +1 -0
- package/dist/classify.d.ts +19 -0
- package/dist/classify.d.ts.map +1 -0
- package/dist/classify.js +124 -0
- package/dist/classify.js.map +1 -0
- package/dist/cli/commands/connections.d.ts +3 -0
- package/dist/cli/commands/connections.d.ts.map +1 -0
- package/dist/cli/commands/connections.js +125 -0
- package/dist/cli/commands/connections.js.map +1 -0
- package/dist/cli/commands/coverage.d.ts +3 -0
- package/dist/cli/commands/coverage.d.ts.map +1 -0
- package/dist/cli/commands/coverage.js +94 -0
- package/dist/cli/commands/coverage.js.map +1 -0
- package/dist/cli/commands/dead.d.ts +3 -0
- package/dist/cli/commands/dead.d.ts.map +1 -0
- package/dist/cli/commands/dead.js +80 -0
- package/dist/cli/commands/dead.js.map +1 -0
- package/dist/cli/commands/diagram.d.ts +3 -0
- package/dist/cli/commands/diagram.d.ts.map +1 -0
- package/dist/cli/commands/diagram.js +102 -0
- package/dist/cli/commands/diagram.js.map +1 -0
- package/dist/cli/commands/find.d.ts +3 -0
- package/dist/cli/commands/find.d.ts.map +1 -0
- package/dist/cli/commands/find.js +128 -0
- package/dist/cli/commands/find.js.map +1 -0
- package/dist/cli/commands/freshness.d.ts +20 -0
- package/dist/cli/commands/freshness.d.ts.map +1 -0
- package/dist/cli/commands/freshness.js +90 -0
- package/dist/cli/commands/freshness.js.map +1 -0
- package/dist/cli/commands/helpers.d.ts +7 -0
- package/dist/cli/commands/helpers.d.ts.map +1 -0
- package/dist/cli/commands/helpers.js +30 -0
- package/dist/cli/commands/helpers.js.map +1 -0
- package/dist/cli/commands/impact.d.ts +3 -0
- package/dist/cli/commands/impact.d.ts.map +1 -0
- package/dist/cli/commands/impact.js +172 -0
- package/dist/cli/commands/impact.js.map +1 -0
- package/dist/cli/commands/lessons.d.ts +6 -0
- package/dist/cli/commands/lessons.d.ts.map +1 -0
- package/dist/cli/commands/lessons.js +279 -0
- package/dist/cli/commands/lessons.js.map +1 -0
- package/dist/cli/commands/list.d.ts +3 -0
- package/dist/cli/commands/list.d.ts.map +1 -0
- package/dist/cli/commands/list.js +91 -0
- package/dist/cli/commands/list.js.map +1 -0
- package/dist/cli/commands/llm-map.d.ts +3 -0
- package/dist/cli/commands/llm-map.d.ts.map +1 -0
- package/dist/cli/commands/llm-map.js +121 -0
- package/dist/cli/commands/llm-map.js.map +1 -0
- package/dist/cli/commands/misc.d.ts +17 -0
- package/dist/cli/commands/misc.d.ts.map +1 -0
- package/dist/cli/commands/misc.js +495 -0
- package/dist/cli/commands/misc.js.map +1 -0
- package/dist/cli/commands/prompts.d.ts +3 -0
- package/dist/cli/commands/prompts.d.ts.map +1 -0
- package/dist/cli/commands/prompts.js +74 -0
- package/dist/cli/commands/prompts.js.map +1 -0
- package/dist/cli/commands/rules.d.ts +3 -0
- package/dist/cli/commands/rules.d.ts.map +1 -0
- package/dist/cli/commands/rules.js +61 -0
- package/dist/cli/commands/rules.js.map +1 -0
- package/dist/cli/commands/scan.d.ts +3 -0
- package/dist/cli/commands/scan.d.ts.map +1 -0
- package/dist/cli/commands/scan.js +177 -0
- package/dist/cli/commands/scan.js.map +1 -0
- package/dist/cli/commands/schema.d.ts +3 -0
- package/dist/cli/commands/schema.d.ts.map +1 -0
- package/dist/cli/commands/schema.js +126 -0
- package/dist/cli/commands/schema.js.map +1 -0
- package/dist/cli/commands/status.d.ts +3 -0
- package/dist/cli/commands/status.d.ts.map +1 -0
- package/dist/cli/commands/status.js +340 -0
- package/dist/cli/commands/status.js.map +1 -0
- package/dist/cli/commands/subgraph.d.ts +3 -0
- package/dist/cli/commands/subgraph.d.ts.map +1 -0
- package/dist/cli/commands/subgraph.js +55 -0
- package/dist/cli/commands/subgraph.js.map +1 -0
- package/dist/cli/commands/temporal.d.ts +3 -0
- package/dist/cli/commands/temporal.d.ts.map +1 -0
- package/dist/cli/commands/temporal.js +112 -0
- package/dist/cli/commands/temporal.js.map +1 -0
- package/dist/cli/commands/trace.d.ts +3 -0
- package/dist/cli/commands/trace.d.ts.map +1 -0
- package/dist/cli/commands/trace.js +65 -0
- package/dist/cli/commands/trace.js.map +1 -0
- package/dist/cli/index.js +88 -825
- package/dist/cli/index.js.map +1 -1
- package/dist/config.d.ts +13 -2
- package/dist/config.d.ts.map +1 -1
- package/dist/config.js +106 -12
- package/dist/config.js.map +1 -1
- package/dist/coverage.d.ts +37 -0
- package/dist/coverage.d.ts.map +1 -0
- package/dist/coverage.js +177 -0
- package/dist/coverage.js.map +1 -0
- package/dist/diagram.d.ts.map +1 -1
- package/dist/diagram.js +41 -0
- package/dist/diagram.js.map +1 -1
- package/dist/diff.d.ts +57 -0
- package/dist/diff.d.ts.map +1 -0
- package/dist/diff.js +527 -0
- package/dist/diff.js.map +1 -0
- package/dist/enrich/cache.d.ts +41 -0
- package/dist/enrich/cache.d.ts.map +1 -0
- package/dist/enrich/cache.js +97 -0
- package/dist/enrich/cache.js.map +1 -0
- package/dist/enrich/external-enrichment.types.d.ts +91 -0
- package/dist/enrich/external-enrichment.types.d.ts.map +1 -0
- package/dist/enrich/external-enrichment.types.js +38 -0
- package/dist/enrich/external-enrichment.types.js.map +1 -0
- package/dist/enrich/external-resolver.d.ts +95 -0
- package/dist/enrich/external-resolver.d.ts.map +1 -0
- package/dist/enrich/external-resolver.js +222 -0
- package/dist/enrich/external-resolver.js.map +1 -0
- package/dist/enrich/fetchers.d.ts +30 -0
- package/dist/enrich/fetchers.d.ts.map +1 -0
- package/dist/enrich/fetchers.js +89 -0
- package/dist/enrich/fetchers.js.map +1 -0
- package/dist/file-resolve.d.ts +35 -0
- package/dist/file-resolve.d.ts.map +1 -0
- package/dist/file-resolve.js +159 -0
- package/dist/file-resolve.js.map +1 -0
- package/dist/freshness/dirty-ledger.d.ts +7 -0
- package/dist/freshness/dirty-ledger.d.ts.map +1 -0
- package/dist/freshness/dirty-ledger.js +56 -0
- package/dist/freshness/dirty-ledger.js.map +1 -0
- package/dist/freshness/drainer.d.ts +38 -0
- package/dist/freshness/drainer.d.ts.map +1 -0
- package/dist/freshness/drainer.js +88 -0
- package/dist/freshness/drainer.js.map +1 -0
- package/dist/freshness/paths.d.ts +9 -0
- package/dist/freshness/paths.d.ts.map +1 -0
- package/dist/freshness/paths.js +24 -0
- package/dist/freshness/paths.js.map +1 -0
- package/dist/freshness/scan-lock.d.ts +8 -0
- package/dist/freshness/scan-lock.d.ts.map +1 -0
- package/dist/freshness/scan-lock.js +68 -0
- package/dist/freshness/scan-lock.js.map +1 -0
- package/dist/freshness/stamp.d.ts +25 -0
- package/dist/freshness/stamp.d.ts.map +1 -0
- package/dist/freshness/stamp.js +50 -0
- package/dist/freshness/stamp.js.map +1 -0
- package/dist/git.d.ts +12 -0
- package/dist/git.d.ts.map +1 -0
- package/dist/git.js +42 -0
- package/dist/git.js.map +1 -0
- package/dist/gitignore-safety.d.ts +41 -0
- package/dist/gitignore-safety.d.ts.map +1 -0
- package/dist/gitignore-safety.js +107 -0
- package/dist/gitignore-safety.js.map +1 -0
- package/dist/impact.d.ts +20 -0
- package/dist/impact.d.ts.map +1 -0
- package/dist/impact.js +89 -0
- package/dist/impact.js.map +1 -0
- package/dist/index.d.ts +24 -2
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +31 -1
- package/dist/index.js.map +1 -1
- package/dist/lessons-store.d.ts +93 -0
- package/dist/lessons-store.d.ts.map +1 -0
- package/dist/lessons-store.js +265 -0
- package/dist/lessons-store.js.map +1 -0
- package/dist/llm-dedup.d.ts +40 -0
- package/dist/llm-dedup.d.ts.map +1 -0
- package/dist/llm-dedup.js +373 -0
- package/dist/llm-dedup.js.map +1 -0
- package/dist/mcp/server.d.ts +9 -0
- package/dist/mcp/server.d.ts.map +1 -0
- package/dist/mcp/server.js +87 -0
- package/dist/mcp/server.js.map +1 -0
- package/dist/mcp/tools.d.ts +198 -0
- package/dist/mcp/tools.d.ts.map +1 -0
- package/dist/mcp/tools.js +744 -0
- package/dist/mcp/tools.js.map +1 -0
- package/dist/metrics/pagerank-louvain.d.ts +44 -0
- package/dist/metrics/pagerank-louvain.d.ts.map +1 -0
- package/dist/metrics/pagerank-louvain.js +128 -0
- package/dist/metrics/pagerank-louvain.js.map +1 -0
- package/dist/parsers/scip-runner.d.ts +63 -0
- package/dist/parsers/scip-runner.d.ts.map +1 -0
- package/dist/parsers/scip-runner.js +179 -0
- package/dist/parsers/scip-runner.js.map +1 -0
- package/dist/projects.d.ts +54 -0
- package/dist/projects.d.ts.map +1 -0
- package/dist/projects.js +153 -0
- package/dist/projects.js.map +1 -0
- package/dist/resolve.d.ts +22 -0
- package/dist/resolve.d.ts.map +1 -0
- package/dist/resolve.js +128 -0
- package/dist/resolve.js.map +1 -0
- package/dist/rules.d.ts +36 -0
- package/dist/rules.d.ts.map +1 -0
- package/dist/rules.js +484 -0
- package/dist/rules.js.map +1 -0
- package/dist/sandbox.d.ts +33 -0
- package/dist/sandbox.d.ts.map +1 -0
- package/dist/sandbox.js +91 -0
- package/dist/sandbox.js.map +1 -0
- package/dist/scan-lock.d.ts +37 -0
- package/dist/scan-lock.d.ts.map +1 -0
- package/dist/scan-lock.js +145 -0
- package/dist/scan-lock.js.map +1 -0
- package/dist/scanner.d.ts +126 -1
- package/dist/scanner.d.ts.map +1 -1
- package/dist/scanner.js +1711 -235
- package/dist/scanner.js.map +1 -1
- package/dist/scanners/connections/ast-scanner.d.ts +9 -2
- package/dist/scanners/connections/ast-scanner.d.ts.map +1 -1
- package/dist/scanners/connections/ast-scanner.js +19 -4
- package/dist/scanners/connections/ast-scanner.js.map +1 -1
- package/dist/scanners/connections/import-scanner.d.ts +27 -0
- package/dist/scanners/connections/import-scanner.d.ts.map +1 -0
- package/dist/scanners/connections/import-scanner.js +537 -0
- package/dist/scanners/connections/import-scanner.js.map +1 -0
- package/dist/scanners/connections/llm-call-tracer.d.ts +1 -1
- package/dist/scanners/connections/llm-call-tracer.d.ts.map +1 -1
- package/dist/scanners/connections/llm-call-tracer.js +6 -2
- package/dist/scanners/connections/llm-call-tracer.js.map +1 -1
- package/dist/scanners/connections/prisma-calls.d.ts +11 -0
- package/dist/scanners/connections/prisma-calls.d.ts.map +1 -0
- package/dist/scanners/connections/prisma-calls.js +237 -0
- package/dist/scanners/connections/prisma-calls.js.map +1 -0
- package/dist/scanners/connections/service-calls.d.ts +1 -1
- package/dist/scanners/connections/service-calls.d.ts.map +1 -1
- package/dist/scanners/connections/service-calls.js +35 -3
- package/dist/scanners/connections/service-calls.js.map +1 -1
- package/dist/scanners/infrastructure/cron-scanner.d.ts +14 -0
- package/dist/scanners/infrastructure/cron-scanner.d.ts.map +1 -0
- package/dist/scanners/infrastructure/cron-scanner.js +383 -0
- package/dist/scanners/infrastructure/cron-scanner.js.map +1 -0
- package/dist/scanners/infrastructure/deploy-scanner.d.ts +11 -0
- package/dist/scanners/infrastructure/deploy-scanner.d.ts.map +1 -0
- package/dist/scanners/infrastructure/deploy-scanner.js +508 -0
- package/dist/scanners/infrastructure/deploy-scanner.js.map +1 -0
- package/dist/scanners/infrastructure/env-scanner.d.ts +55 -0
- package/dist/scanners/infrastructure/env-scanner.d.ts.map +1 -0
- package/dist/scanners/infrastructure/env-scanner.js +431 -0
- package/dist/scanners/infrastructure/env-scanner.js.map +1 -0
- package/dist/scanners/infrastructure/field-usage-analyzer.d.ts +52 -0
- package/dist/scanners/infrastructure/field-usage-analyzer.d.ts.map +1 -0
- package/dist/scanners/infrastructure/field-usage-analyzer.js +480 -0
- package/dist/scanners/infrastructure/field-usage-analyzer.js.map +1 -0
- package/dist/scanners/infrastructure/prisma-parser.d.ts +21 -0
- package/dist/scanners/infrastructure/prisma-parser.d.ts.map +1 -0
- package/dist/scanners/infrastructure/prisma-parser.js +58 -0
- package/dist/scanners/infrastructure/prisma-parser.js.map +1 -0
- package/dist/scanners/infrastructure/prisma-scanner.d.ts +30 -0
- package/dist/scanners/infrastructure/prisma-scanner.d.ts.map +1 -0
- package/dist/scanners/infrastructure/prisma-scanner.js +329 -0
- package/dist/scanners/infrastructure/prisma-scanner.js.map +1 -0
- package/dist/scanners/infrastructure/queue-scanner.d.ts +14 -0
- package/dist/scanners/infrastructure/queue-scanner.d.ts.map +1 -0
- package/dist/scanners/infrastructure/queue-scanner.js +455 -0
- package/dist/scanners/infrastructure/queue-scanner.js.map +1 -0
- package/dist/scanners/infrastructure/typespec-validator.d.ts +50 -0
- package/dist/scanners/infrastructure/typespec-validator.d.ts.map +1 -0
- package/dist/scanners/infrastructure/typespec-validator.js +407 -0
- package/dist/scanners/infrastructure/typespec-validator.js.map +1 -0
- package/dist/scanners/packages/swift.d.ts +5 -0
- package/dist/scanners/packages/swift.d.ts.map +1 -1
- package/dist/scanners/packages/swift.js +23 -0
- package/dist/scanners/packages/swift.js.map +1 -1
- package/dist/scanners/prompts/detector.d.ts +13 -2
- package/dist/scanners/prompts/detector.d.ts.map +1 -1
- package/dist/scanners/prompts/detector.js +97 -46
- package/dist/scanners/prompts/detector.js.map +1 -1
- package/dist/scanners/prompts/index.d.ts +1 -1
- package/dist/scanners/prompts/index.d.ts.map +1 -1
- package/dist/scanners/prompts/index.js +2 -2
- package/dist/scanners/prompts/index.js.map +1 -1
- package/dist/scanners/swift/code-scanner.d.ts +1 -1
- package/dist/scanners/swift/code-scanner.d.ts.map +1 -1
- package/dist/scanners/swift/code-scanner.js +216 -2
- package/dist/scanners/swift/code-scanner.js.map +1 -1
- package/dist/scanners/swift/swiftui-scanner.d.ts +45 -0
- package/dist/scanners/swift/swiftui-scanner.d.ts.map +1 -0
- package/dist/scanners/swift/swiftui-scanner.js +606 -0
- package/dist/scanners/swift/swiftui-scanner.js.map +1 -0
- package/dist/scanners/xcode/pbxproj-parser.d.ts +32 -0
- package/dist/scanners/xcode/pbxproj-parser.d.ts.map +1 -0
- package/dist/scanners/xcode/pbxproj-parser.js +407 -0
- package/dist/scanners/xcode/pbxproj-parser.js.map +1 -0
- package/dist/scanners/xcode/storyboard-scanner.d.ts +13 -0
- package/dist/scanners/xcode/storyboard-scanner.d.ts.map +1 -0
- package/dist/scanners/xcode/storyboard-scanner.js +236 -0
- package/dist/scanners/xcode/storyboard-scanner.js.map +1 -0
- package/dist/storage/markdown-view.d.ts +49 -0
- package/dist/storage/markdown-view.d.ts.map +1 -0
- package/dist/storage/markdown-view.js +233 -0
- package/dist/storage/markdown-view.js.map +1 -0
- package/dist/storage.d.ts +225 -9
- package/dist/storage.d.ts.map +1 -1
- package/dist/storage.js +945 -86
- package/dist/storage.js.map +1 -1
- package/dist/subgraph.d.ts +30 -0
- package/dist/subgraph.d.ts.map +1 -0
- package/dist/subgraph.js +106 -0
- package/dist/subgraph.js.map +1 -0
- package/dist/temporal/git-store.d.ts +65 -0
- package/dist/temporal/git-store.d.ts.map +1 -0
- package/dist/temporal/git-store.js +166 -0
- package/dist/temporal/git-store.js.map +1 -0
- package/dist/trace.d.ts +38 -0
- package/dist/trace.d.ts.map +1 -0
- package/dist/trace.js +292 -0
- package/dist/trace.js.map +1 -0
- package/dist/types.d.ts +322 -2
- package/dist/types.d.ts.map +1 -1
- package/dist/types.js +53 -0
- package/dist/types.js.map +1 -1
- package/hooks/hooks.json +1 -55
- package/hooks/mark-dirty.sh +20 -0
- package/hooks/post-bash-suggest.sh +30 -0
- package/hooks/post-edit-suggest.sh +29 -0
- package/hooks/pre-edit-warn.sh +28 -0
- package/hooks/session-start.sh +19 -0
- package/hooks/stop-suggest.sh +49 -0
- package/package.json +30 -11
- package/scripts/install-codex-plugin.sh +119 -0
- package/scripts/install-plugin.sh +29 -24
- package/skills/architecture-export/SKILL.md +79 -0
- package/skills/architecture-scan/SKILL.md +64 -0
- package/skills/code-review/SKILL.md +368 -0
- package/skills/impact-analysis/SKILL.md +80 -0
- package/skills/infrastructure-scanning.md +42 -0
- package/skills/navgator-setup/SKILL.md +108 -0
- package/web/.next/standalone/web/.next/BUILD_ID +1 -1
- package/web/.next/standalone/web/.next/app-path-routes-manifest.json +3 -0
- package/web/.next/standalone/web/.next/build-manifest.json +2 -2
- package/web/.next/standalone/web/.next/prerender-manifest.json +3 -3
- package/web/.next/standalone/web/.next/required-server-files.json +4 -4
- package/web/.next/standalone/web/.next/routes-manifest.json +18 -0
- package/web/.next/standalone/web/.next/server/app/_global-error.html +2 -2
- package/web/.next/standalone/web/.next/server/app/_global-error.rsc +1 -1
- package/web/.next/standalone/web/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +1 -1
- package/web/.next/standalone/web/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/web/.next/standalone/web/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/web/.next/standalone/web/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/web/.next/standalone/web/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/web/.next/standalone/web/.next/server/app/_not-found/page/next-font-manifest.json +2 -2
- package/web/.next/standalone/web/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
- package/web/.next/standalone/web/.next/server/app/_not-found.html +1 -1
- package/web/.next/standalone/web/.next/server/app/_not-found.rsc +3 -3
- package/web/.next/standalone/web/.next/server/app/_not-found.segments/_full.segment.rsc +3 -3
- package/web/.next/standalone/web/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/web/.next/standalone/web/.next/server/app/_not-found.segments/_index.segment.rsc +3 -3
- package/web/.next/standalone/web/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/web/.next/standalone/web/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/web/.next/standalone/web/.next/server/app/_not-found.segments/_tree.segment.rsc +3 -3
- package/web/.next/standalone/web/.next/server/app/api/prompts/route.js.nft.json +1 -1
- package/web/.next/standalone/web/.next/server/app/api/rules/route/app-paths-manifest.json +3 -0
- package/web/.next/standalone/web/.next/server/app/api/rules/route/build-manifest.json +11 -0
- package/web/.next/standalone/web/.next/server/app/api/rules/route/server-reference-manifest.json +4 -0
- package/web/.next/standalone/web/.next/server/app/api/rules/route.js +6 -0
- package/web/.next/standalone/web/.next/server/app/api/rules/route.js.map +5 -0
- package/web/.next/standalone/web/.next/server/app/api/rules/route.js.nft.json +1 -0
- package/web/.next/standalone/web/.next/server/app/api/rules/route_client-reference-manifest.js +2 -0
- package/web/.next/standalone/web/.next/server/app/api/subgraph/route/app-paths-manifest.json +3 -0
- package/web/.next/standalone/web/.next/server/app/api/subgraph/route/build-manifest.json +11 -0
- package/web/.next/standalone/web/.next/server/app/api/subgraph/route/server-reference-manifest.json +4 -0
- package/web/.next/standalone/web/.next/server/app/api/subgraph/route.js +6 -0
- package/web/.next/standalone/web/.next/server/app/api/subgraph/route.js.map +5 -0
- package/web/.next/standalone/web/.next/server/app/api/subgraph/route.js.nft.json +1 -0
- package/web/.next/standalone/web/.next/server/app/api/subgraph/route_client-reference-manifest.js +2 -0
- package/web/.next/standalone/web/.next/server/app/api/trace/route/app-paths-manifest.json +3 -0
- package/web/.next/standalone/web/.next/server/app/api/trace/route/build-manifest.json +11 -0
- package/web/.next/standalone/web/.next/server/app/api/trace/route/server-reference-manifest.json +4 -0
- package/web/.next/standalone/web/.next/server/app/api/trace/route.js +6 -0
- package/web/.next/standalone/web/.next/server/app/api/trace/route.js.map +5 -0
- package/web/.next/standalone/web/.next/server/app/api/trace/route.js.nft.json +1 -0
- package/web/.next/standalone/web/.next/server/app/api/trace/route_client-reference-manifest.js +2 -0
- package/web/.next/standalone/web/.next/server/app/index.html +1 -1
- package/web/.next/standalone/web/.next/server/app/index.rsc +6 -6
- package/web/.next/standalone/web/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
- package/web/.next/standalone/web/.next/server/app/index.segments/_full.segment.rsc +6 -6
- package/web/.next/standalone/web/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/web/.next/standalone/web/.next/server/app/index.segments/_index.segment.rsc +3 -3
- package/web/.next/standalone/web/.next/server/app/index.segments/_tree.segment.rsc +5 -5
- package/web/.next/standalone/web/.next/server/app/page/next-font-manifest.json +2 -2
- package/web/.next/standalone/web/.next/server/app/page_client-reference-manifest.js +1 -1
- package/web/.next/standalone/web/.next/server/app-paths-manifest.json +3 -0
- package/web/.next/standalone/web/.next/server/chunks/2374f_next_dist_esm_build_templates_app-route_0bb4e66a.js +1 -1
- package/web/.next/standalone/web/.next/server/chunks/[root-of-the-server]__006b837d._.js +1 -1
- package/web/.next/standalone/web/.next/server/chunks/[root-of-the-server]__0426efe8._.js +3 -0
- package/web/.next/standalone/web/.next/server/chunks/[root-of-the-server]__2e09fec9._.js +1 -1
- package/web/.next/standalone/web/.next/server/chunks/[root-of-the-server]__38d0390f._.js +1 -1
- package/web/.next/standalone/web/.next/server/chunks/[root-of-the-server]__594bcf20._.js +1 -1
- package/web/.next/standalone/web/.next/server/chunks/[root-of-the-server]__b888fadf._.js +1 -1
- package/web/.next/standalone/web/.next/server/chunks/[root-of-the-server]__cd5f36ce._.js +3 -0
- package/web/.next/standalone/web/.next/server/chunks/[root-of-the-server]__ee6fc95f._.js +3 -0
- package/web/.next/standalone/web/.next/server/chunks/[root-of-the-server]__fa2ec862._.js +1 -1
- package/web/.next/standalone/web/.next/server/chunks/ssr/web_171de0df._.js +9 -4
- package/web/.next/standalone/web/.next/server/chunks/web__next-internal_server_app_api_rules_route_actions_3de01bd5.js +3 -0
- package/web/.next/standalone/web/.next/server/chunks/web__next-internal_server_app_api_subgraph_route_actions_d8b5a63f.js +3 -0
- package/web/.next/standalone/web/.next/server/chunks/web__next-internal_server_app_api_trace_route_actions_b0703ae2.js +3 -0
- package/web/.next/standalone/web/.next/server/next-font-manifest.js +1 -1
- package/web/.next/standalone/web/.next/server/next-font-manifest.json +4 -4
- package/web/.next/standalone/web/.next/server/pages/404.html +1 -1
- package/web/.next/standalone/web/.next/server/pages/500.html +2 -2
- package/web/.next/standalone/web/.next/server/server-reference-manifest.js +1 -1
- package/web/.next/standalone/web/.next/server/server-reference-manifest.json +1 -1
- package/web/.next/standalone/web/.next/static/chunks/22a09ecf6ba35cfd.js +17 -0
- package/web/.next/standalone/web/.next/static/chunks/9857ba86ce4e82d8.css +2 -0
- package/web/.next/standalone/web/.next/static/chunks/f899547f99ef4b76.css +1 -0
- package/web/.next/standalone/web/.next/static/media/4fa387ec64143e14-s.c36e1862.woff2 +0 -0
- package/web/.next/standalone/web/.next/static/media/53b9e256198e5412-s.853d50a3.woff2 +0 -0
- package/web/.next/standalone/web/.next/static/media/5ce348bf30bf5439-s.ebceb24d.woff2 +0 -0
- package/web/.next/standalone/web/.next/static/media/6306c77e7c8268e4-s.ff4a2084.woff2 +0 -0
- package/web/.next/standalone/web/.next/static/media/7178b3e590c64307-s.55554cd0.woff2 +0 -0
- package/web/.next/standalone/web/.next/static/media/797e433ab948586e-s.p.479bea2b.woff2 +0 -0
- package/web/.next/standalone/web/.next/static/media/7d817b4c03b0c5f1-s.f377b9c4.woff2 +0 -0
- package/web/.next/standalone/web/.next/static/media/8a480f0b521d4e75-s.ea323500.woff2 +0 -0
- package/web/.next/standalone/web/.next/static/media/bbc41e54d2fcbd21-s.d1207556.woff2 +0 -0
- package/web/.next/standalone/web/.next/static/media/caa3a2e1cccd8315-s.p.3b6cae6d.woff2 +0 -0
- package/web/.next/standalone/web/.next/static/media/fef07dbb0973bf53-s.518e079e.woff2 +0 -0
- package/web/.next/standalone/web/app/api/components/route.ts +1 -1
- package/web/.next/standalone/web/app/api/connections/route.ts +3 -1
- package/web/.next/standalone/web/app/api/graph/route.ts +3 -3
- package/web/.next/standalone/web/app/api/projects/route.ts +1 -1
- package/web/.next/standalone/web/app/api/prompts/route.ts +2 -2
- package/web/.next/standalone/web/app/api/rules/route.ts +213 -0
- package/web/.next/standalone/web/app/api/settings/route.ts +2 -2
- package/web/.next/standalone/web/app/api/status/route.ts +1 -1
- package/web/.next/standalone/web/app/api/subgraph/route.ts +267 -0
- package/web/.next/standalone/web/app/api/trace/route.ts +321 -0
- package/web/.next/standalone/web/app/page.tsx +9 -1
- package/web/.next/standalone/web/components/connections-panel.tsx +23 -6
- package/web/.next/standalone/web/components/coverage-panel.tsx +309 -0
- package/web/.next/standalone/web/components/rules-panel.tsx +156 -0
- package/web/.next/standalone/web/components/sidebar.tsx +8 -0
- package/web/.next/standalone/web/components/status-overview.tsx +24 -1
- package/web/.next/standalone/web/components/subgraph-panel.tsx +382 -0
- package/web/.next/standalone/web/components/trace-panel.tsx +325 -0
- package/web/.next/standalone/web/lib/hooks/index.ts +3 -0
- package/web/.next/standalone/web/lib/hooks/use-coverage.ts +68 -0
- package/web/.next/standalone/web/lib/hooks/use-subgraph.ts +85 -0
- package/web/.next/standalone/web/lib/hooks/use-trace.ts +84 -0
- package/web/.next/standalone/web/lib/types.ts +108 -0
- package/web/.next/standalone/web/package-lock.json +218 -0
- package/web/.next/standalone/web/package.json +4 -2
- package/web/.next/standalone/web/server.js +1 -1
- package/web/.next/static/chunks/22a09ecf6ba35cfd.js +17 -0
- package/web/.next/static/chunks/9857ba86ce4e82d8.css +2 -0
- package/web/.next/static/chunks/f899547f99ef4b76.css +1 -0
- package/web/.next/static/media/4fa387ec64143e14-s.c36e1862.woff2 +0 -0
- package/web/.next/static/media/53b9e256198e5412-s.853d50a3.woff2 +0 -0
- package/web/.next/static/media/5ce348bf30bf5439-s.ebceb24d.woff2 +0 -0
- package/web/.next/static/media/6306c77e7c8268e4-s.ff4a2084.woff2 +0 -0
- package/web/.next/static/media/7178b3e590c64307-s.55554cd0.woff2 +0 -0
- package/web/.next/static/media/797e433ab948586e-s.p.479bea2b.woff2 +0 -0
- package/web/.next/static/media/7d817b4c03b0c5f1-s.f377b9c4.woff2 +0 -0
- package/web/.next/static/media/8a480f0b521d4e75-s.ea323500.woff2 +0 -0
- package/web/.next/static/media/bbc41e54d2fcbd21-s.d1207556.woff2 +0 -0
- package/web/.next/static/media/caa3a2e1cccd8315-s.p.3b6cae6d.woff2 +0 -0
- package/web/.next/static/media/fef07dbb0973bf53-s.518e079e.woff2 +0 -0
- package/skills/check/SKILL.md +0 -64
- package/skills/connections/SKILL.md +0 -54
- package/skills/diagram/SKILL.md +0 -64
- package/skills/export/SKILL.md +0 -49
- package/skills/impact/SKILL.md +0 -58
- package/skills/install/SKILL.md +0 -94
- package/skills/scan/SKILL.md +0 -75
- package/skills/status/SKILL.md +0 -37
- package/skills/ui/SKILL.md +0 -43
- package/skills/update/SKILL.md +0 -43
- package/web/.next/standalone/web/.next/static/chunks/8a80e7184ad3a13f.css +0 -2
- package/web/.next/standalone/web/.next/static/chunks/c056475f5f4424b6.css +0 -1
- package/web/.next/standalone/web/.next/static/chunks/cb3513192b63e480.js +0 -12
- package/web/.next/standalone/web/.next/static/media/4fa387ec64143e14-s.c1fdd6c2.woff2 +0 -0
- package/web/.next/standalone/web/.next/static/media/7178b3e590c64307-s.b97b3418.woff2 +0 -0
- package/web/.next/standalone/web/.next/static/media/797e433ab948586e-s.p.dbea232f.woff2 +0 -0
- package/web/.next/standalone/web/.next/static/media/8a480f0b521d4e75-s.8e0177b5.woff2 +0 -0
- package/web/.next/standalone/web/.next/static/media/bbc41e54d2fcbd21-s.799d8ef8.woff2 +0 -0
- package/web/.next/standalone/web/.next/static/media/caa3a2e1cccd8315-s.p.853070df.woff2 +0 -0
- package/web/.next/static/chunks/8a80e7184ad3a13f.css +0 -2
- package/web/.next/static/chunks/c056475f5f4424b6.css +0 -1
- package/web/.next/static/chunks/cb3513192b63e480.js +0 -12
- package/web/.next/static/media/4fa387ec64143e14-s.c1fdd6c2.woff2 +0 -0
- package/web/.next/static/media/7178b3e590c64307-s.b97b3418.woff2 +0 -0
- package/web/.next/static/media/797e433ab948586e-s.p.dbea232f.woff2 +0 -0
- package/web/.next/static/media/8a480f0b521d4e75-s.8e0177b5.woff2 +0 -0
- package/web/.next/static/media/bbc41e54d2fcbd21-s.799d8ef8.woff2 +0 -0
- package/web/.next/static/media/caa3a2e1cccd8315-s.p.853070df.woff2 +0 -0
- /package/web/.next/standalone/web/.next/static/{P-ZMQO7_Wnj487ks3guqa → qZVrJ4kmwXfw4Ikgj1oXR}/_buildManifest.js +0 -0
- /package/web/.next/standalone/web/.next/static/{P-ZMQO7_Wnj487ks3guqa → qZVrJ4kmwXfw4Ikgj1oXR}/_clientMiddlewareManifest.json +0 -0
- /package/web/.next/standalone/web/.next/static/{P-ZMQO7_Wnj487ks3guqa → qZVrJ4kmwXfw4Ikgj1oXR}/_ssgManifest.js +0 -0
- /package/web/.next/static/{P-ZMQO7_Wnj487ks3guqa → qZVrJ4kmwXfw4Ikgj1oXR}/_buildManifest.js +0 -0
- /package/web/.next/static/{P-ZMQO7_Wnj487ks3guqa → qZVrJ4kmwXfw4Ikgj1oXR}/_clientMiddlewareManifest.json +0 -0
- /package/web/.next/static/{P-ZMQO7_Wnj487ks3guqa → qZVrJ4kmwXfw4Ikgj1oXR}/_ssgManifest.js +0 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"architecture-insights.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/architecture-insights.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import { describe, expect, it } from 'vitest';
|
|
2
|
+
import { detectImportCycles, detectLayerViolations, getTopFanOut, getTopHotspots, } from '../architecture-insights.js';
|
|
3
|
+
import { createComponent, createConnection } from './helpers.js';
|
|
4
|
+
describe('architecture insights', () => {
|
|
5
|
+
it('surfaces internal hotspots by import fan-in', () => {
|
|
6
|
+
const core = createComponent({ name: 'core/types', type: 'component', file: 'src/core/types.ts' });
|
|
7
|
+
const a = createComponent({ name: 'cdp/driver', type: 'component', file: 'src/cdp/driver.ts' });
|
|
8
|
+
const b = createComponent({ name: 'mcp/server', type: 'component', file: 'src/mcp/server.ts' });
|
|
9
|
+
const c = createComponent({ name: 'media/session', type: 'component', file: 'src/media/session.ts' });
|
|
10
|
+
const hotspots = getTopHotspots([core, a, b, c], [
|
|
11
|
+
createConnection(a, core, { connection_type: 'imports' }),
|
|
12
|
+
createConnection(b, core, { connection_type: 'imports' }),
|
|
13
|
+
createConnection(c, core, { connection_type: 'imports' }),
|
|
14
|
+
]);
|
|
15
|
+
expect(hotspots[0].component.name).toBe('core/types');
|
|
16
|
+
expect(hotspots[0].count).toBe(3);
|
|
17
|
+
});
|
|
18
|
+
it('surfaces high fan-out internal modules', () => {
|
|
19
|
+
const driver = createComponent({ name: 'cdp/driver', type: 'component', file: 'src/cdp/driver.ts' });
|
|
20
|
+
const deps = Array.from({ length: 8 }, (_, index) => createComponent({ name: `core/dep-${index}`, type: 'component', file: `src/core/dep-${index}.ts` }));
|
|
21
|
+
const fanOut = getTopFanOut([driver, ...deps], deps.map((dep) => createConnection(driver, dep, { connection_type: 'imports' })));
|
|
22
|
+
expect(fanOut[0].component.name).toBe('cdp/driver');
|
|
23
|
+
expect(fanOut[0].count).toBe(8);
|
|
24
|
+
});
|
|
25
|
+
it('detects upward inferred layer imports', () => {
|
|
26
|
+
const core = createComponent({ name: 'core/types', type: 'component', file: 'src/core/types.ts' });
|
|
27
|
+
const cdp = createComponent({ name: 'cdp/driver', type: 'component', file: 'src/cdp/driver.ts' });
|
|
28
|
+
const violations = detectLayerViolations([core, cdp], [createConnection(cdp, core, { connection_type: 'imports' }), createConnection(core, cdp, { connection_type: 'imports' })]);
|
|
29
|
+
expect(violations).toHaveLength(1);
|
|
30
|
+
expect(violations[0].from.name).toBe('cdp/driver');
|
|
31
|
+
expect(violations[0].to.name).toBe('core/types');
|
|
32
|
+
});
|
|
33
|
+
it('detects import cycles', () => {
|
|
34
|
+
const a = createComponent({ name: 'core/a', type: 'component', file: 'src/core/a.ts' });
|
|
35
|
+
const b = createComponent({ name: 'core/b', type: 'component', file: 'src/core/b.ts' });
|
|
36
|
+
const c = createComponent({ name: 'core/c', type: 'component', file: 'src/core/c.ts' });
|
|
37
|
+
const cycles = detectImportCycles([a, b, c], [
|
|
38
|
+
createConnection(a, b, { connection_type: 'imports' }),
|
|
39
|
+
createConnection(b, c, { connection_type: 'imports' }),
|
|
40
|
+
createConnection(c, a, { connection_type: 'imports' }),
|
|
41
|
+
]);
|
|
42
|
+
expect(cycles).toHaveLength(1);
|
|
43
|
+
expect(cycles[0]).toEqual(['core/a', 'core/b', 'core/c', 'core/a']);
|
|
44
|
+
});
|
|
45
|
+
});
|
|
46
|
+
//# sourceMappingURL=architecture-insights.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"architecture-insights.test.js","sourceRoot":"","sources":["../../src/__tests__/architecture-insights.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* NavGator audit sampler tests — Run 2 / D6
|
|
3
|
+
*
|
|
4
|
+
* Math values verified against:
|
|
5
|
+
* - NIST/SEMATECH e-Handbook §6.2.2-3 (AQL=2.5%, single sampling, GIL II)
|
|
6
|
+
* - Wald 1945 — SPRT bounds A=(1-β)/α, B=β/(1-α)
|
|
7
|
+
* - Cochran 1977 — sample-size formula + FPC
|
|
8
|
+
*/
|
|
9
|
+
export {};
|
|
10
|
+
//# sourceMappingURL=audit-sampler.test.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"audit-sampler.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/audit-sampler.test.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG"}
|
|
@@ -0,0 +1,172 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* NavGator audit sampler tests — Run 2 / D6
|
|
3
|
+
*
|
|
4
|
+
* Math values verified against:
|
|
5
|
+
* - NIST/SEMATECH e-Handbook §6.2.2-3 (AQL=2.5%, single sampling, GIL II)
|
|
6
|
+
* - Wald 1945 — SPRT bounds A=(1-β)/α, B=β/(1-α)
|
|
7
|
+
* - Cochran 1977 — sample-size formula + FPC
|
|
8
|
+
*/
|
|
9
|
+
import { describe, it, expect } from 'vitest';
|
|
10
|
+
import { binomialCDF, chooseAQLPlan, cochranSize, neymanAllocate, selectAuditSample, sprtNext, Z, } from '../audit/sampler.js';
|
|
11
|
+
describe('binomialCDF', () => {
|
|
12
|
+
it('returns 1 when c >= n', () => {
|
|
13
|
+
expect(binomialCDF(10, 0.5, 10)).toBe(1);
|
|
14
|
+
});
|
|
15
|
+
it('returns 0 when c < 0', () => {
|
|
16
|
+
expect(binomialCDF(10, 0.5, -1)).toBe(0);
|
|
17
|
+
});
|
|
18
|
+
it('matches manual calc for small n', () => {
|
|
19
|
+
// P(X ≤ 1 | n=5, p=0.2) = (0.8)^5 + 5·0.2·(0.8)^4
|
|
20
|
+
const expected = Math.pow(0.8, 5) + 5 * 0.2 * Math.pow(0.8, 4);
|
|
21
|
+
expect(binomialCDF(5, 0.2, 1)).toBeCloseTo(expected, 10);
|
|
22
|
+
});
|
|
23
|
+
it('AQL n=80, c=5, p=0.025 has high acceptance probability (≥ 0.99)', () => {
|
|
24
|
+
// OC curve: at AQL itself, the plan should accept with high probability.
|
|
25
|
+
// For n=80, c=5, p=0.025, P(X≤5) ≈ 0.987 — comfortably above 0.95.
|
|
26
|
+
const p = binomialCDF(80, 0.025, 5);
|
|
27
|
+
expect(p).toBeGreaterThan(0.95);
|
|
28
|
+
});
|
|
29
|
+
it('AQL n=80, c=5 rejects high defect rates (p=0.15)', () => {
|
|
30
|
+
const p = binomialCDF(80, 0.15, 5);
|
|
31
|
+
expect(p).toBeLessThan(0.05);
|
|
32
|
+
});
|
|
33
|
+
});
|
|
34
|
+
describe('chooseAQLPlan (MIL-STD-105E lookup)', () => {
|
|
35
|
+
it('lot of 100 → code letter F: n=20, c=1', () => {
|
|
36
|
+
const plan = chooseAQLPlan(100);
|
|
37
|
+
expect(plan.codeLetter).toBe('F');
|
|
38
|
+
expect(plan.n).toBe(20);
|
|
39
|
+
expect(plan.c).toBe(1);
|
|
40
|
+
});
|
|
41
|
+
it('lot of 1000 → code letter J: n=80, c=5', () => {
|
|
42
|
+
const plan = chooseAQLPlan(1000);
|
|
43
|
+
expect(plan.codeLetter).toBe('J');
|
|
44
|
+
expect(plan.n).toBe(80);
|
|
45
|
+
expect(plan.c).toBe(5);
|
|
46
|
+
});
|
|
47
|
+
it('lot of 5000 → code letter L: n=200, c=10', () => {
|
|
48
|
+
const plan = chooseAQLPlan(5000);
|
|
49
|
+
expect(plan.codeLetter).toBe('L');
|
|
50
|
+
expect(plan.n).toBe(200);
|
|
51
|
+
expect(plan.c).toBe(10);
|
|
52
|
+
});
|
|
53
|
+
it('caps n to lotSize for tiny populations', () => {
|
|
54
|
+
const plan = chooseAQLPlan(2);
|
|
55
|
+
// Table row for N≤8 has n=2; we already lot-size cap.
|
|
56
|
+
expect(plan.n).toBe(2);
|
|
57
|
+
});
|
|
58
|
+
});
|
|
59
|
+
describe('sprtNext (Wald 1945)', () => {
|
|
60
|
+
it('A=19, B≈0.0526 for α=β=0.05', () => {
|
|
61
|
+
const step = sprtNext([0], 0.01, 0.05, 0.05, 0.05);
|
|
62
|
+
expect(step.A).toBeCloseTo(19, 6);
|
|
63
|
+
expect(step.B).toBeCloseTo(0.0526315789, 6);
|
|
64
|
+
});
|
|
65
|
+
it('continues with no observations', () => {
|
|
66
|
+
const step = sprtNext([], 0.01, 0.05);
|
|
67
|
+
expect(step.verdict).toBe('continue');
|
|
68
|
+
expect(step.logLR).toBe(0);
|
|
69
|
+
});
|
|
70
|
+
it('accepts H0 when many cleans observed (negligible defects)', () => {
|
|
71
|
+
// With α=β=0.05 and p0=0.01, p1=0.05: each clean contributes
|
|
72
|
+
// ln((1-p1)/(1-p0)) ≈ -0.0412 to logLR. Need ≥ |ln B| / 0.0412 ≈ 72 cleans
|
|
73
|
+
// to drop below ln B = ln(0.0526) ≈ -2.944 and accept.
|
|
74
|
+
const obs = Array(80).fill(0);
|
|
75
|
+
const step = sprtNext(obs, 0.01, 0.05);
|
|
76
|
+
expect(step.verdict).toBe('accept');
|
|
77
|
+
});
|
|
78
|
+
it('rejects (i.e., supports H1) when many defects observed', () => {
|
|
79
|
+
// 20 defects in 20 → strongly supports H1
|
|
80
|
+
const obs = Array(20).fill(1);
|
|
81
|
+
const step = sprtNext(obs, 0.01, 0.05);
|
|
82
|
+
expect(step.verdict).toBe('reject');
|
|
83
|
+
});
|
|
84
|
+
it('throws on invalid p0/p1', () => {
|
|
85
|
+
expect(() => sprtNext([], 0.5, 0.3)).toThrow();
|
|
86
|
+
expect(() => sprtNext([], 0, 0.5)).toThrow();
|
|
87
|
+
expect(() => sprtNext([], 0.5, 1)).toThrow();
|
|
88
|
+
});
|
|
89
|
+
});
|
|
90
|
+
describe('cochranSize', () => {
|
|
91
|
+
it('p=0.5 e=0.05 Z=1.96 → 385 (no FPC)', () => {
|
|
92
|
+
// Textbook: n = 1.96² · 0.25 / 0.0025 = 384.16 → ceil = 385
|
|
93
|
+
expect(cochranSize(0.5, 0.05, 1.96)).toBe(385);
|
|
94
|
+
});
|
|
95
|
+
it('p=0.5 e=0.05 Z=1.96 N=2000 → 322 (textbook FPC value)', () => {
|
|
96
|
+
// Cochran (1977): n_adj = 384.16 / (1 + 383.16/2000) = 384.16 / 1.1916 = 322.55 → ceil = 323
|
|
97
|
+
// Textbook frequently rounds intermediate to 384, giving exactly 322.
|
|
98
|
+
// Either 322 or 323 is acceptable depending on rounding strategy; we ceil
|
|
99
|
+
// the FPC-adjusted real value, which yields 323.
|
|
100
|
+
const n = cochranSize(0.5, 0.05, 1.96, 2000);
|
|
101
|
+
expect(n).toBeGreaterThanOrEqual(322);
|
|
102
|
+
expect(n).toBeLessThanOrEqual(323);
|
|
103
|
+
});
|
|
104
|
+
it('p=0.5 e=0.10 Z=1.96 → 97', () => {
|
|
105
|
+
// 1.96² · 0.25 / 0.01 = 96.04 → ceil = 97
|
|
106
|
+
expect(cochranSize(0.5, 0.10, 1.96)).toBe(97);
|
|
107
|
+
});
|
|
108
|
+
it('Z constants match standard normal quantiles', () => {
|
|
109
|
+
expect(Z.Z_95).toBeCloseTo(1.96, 3);
|
|
110
|
+
expect(Z.Z_99).toBeCloseTo(2.576, 3);
|
|
111
|
+
expect(Z.Z_90).toBeCloseTo(1.645, 3);
|
|
112
|
+
});
|
|
113
|
+
it('throws on invalid inputs', () => {
|
|
114
|
+
expect(() => cochranSize(-0.1, 0.05)).toThrow();
|
|
115
|
+
expect(() => cochranSize(0.5, 0)).toThrow();
|
|
116
|
+
expect(() => cochranSize(0.5, 1.5)).toThrow();
|
|
117
|
+
expect(() => cochranSize(0.5, 0.05, -1)).toThrow();
|
|
118
|
+
});
|
|
119
|
+
});
|
|
120
|
+
describe('neymanAllocate', () => {
|
|
121
|
+
it('preserves Σ n_h = n', () => {
|
|
122
|
+
const out = neymanAllocate(100, [200, 300, 500], [0.1, 0.2, 0.3]);
|
|
123
|
+
const total = out.reduce((a, b) => a + b, 0);
|
|
124
|
+
expect(total).toBe(100);
|
|
125
|
+
});
|
|
126
|
+
it('allocates more to higher-variance strata', () => {
|
|
127
|
+
// Equal stratum sizes; only σ differs.
|
|
128
|
+
const out = neymanAllocate(60, [100, 100, 100], [0.1, 0.3, 0.5]);
|
|
129
|
+
expect(out[0]).toBeLessThan(out[1]);
|
|
130
|
+
expect(out[1]).toBeLessThan(out[2]);
|
|
131
|
+
});
|
|
132
|
+
it('handles all-zero σ (returns equal split)', () => {
|
|
133
|
+
const out = neymanAllocate(9, [10, 10, 10], [0, 0, 0]);
|
|
134
|
+
expect(out.reduce((a, b) => a + b, 0)).toBe(9);
|
|
135
|
+
// Equal-ish split (3,3,3)
|
|
136
|
+
expect(Math.max(...out) - Math.min(...out)).toBeLessThanOrEqual(1);
|
|
137
|
+
});
|
|
138
|
+
it('caps allocation at stratum size', () => {
|
|
139
|
+
// Want 100 from a stratum that only has 10 elements
|
|
140
|
+
const out = neymanAllocate(50, [10, 1000], [0.5, 0.1]);
|
|
141
|
+
expect(out[0]).toBeLessThanOrEqual(10);
|
|
142
|
+
});
|
|
143
|
+
it('returns empty for empty inputs', () => {
|
|
144
|
+
expect(neymanAllocate(10, [], [])).toEqual([]);
|
|
145
|
+
});
|
|
146
|
+
});
|
|
147
|
+
describe('selectAuditSample', () => {
|
|
148
|
+
it('respects total N and stratifies by key', () => {
|
|
149
|
+
const items = Array(200).fill(0).map((_, i) => ({ id: i, kind: i < 50 ? 'A' : i < 150 ? 'B' : 'C' }));
|
|
150
|
+
const { samples, byStratum } = selectAuditSample(items, 30, (it) => it.kind);
|
|
151
|
+
expect(samples.length).toBeLessThanOrEqual(30);
|
|
152
|
+
expect(samples.length).toBeGreaterThan(0);
|
|
153
|
+
expect(Object.keys(byStratum).sort()).toEqual(['A', 'B', 'C']);
|
|
154
|
+
});
|
|
155
|
+
it('returns empty when totalN=0', () => {
|
|
156
|
+
const items = [{ id: 1 }];
|
|
157
|
+
const { samples } = selectAuditSample(items, 0, () => 'x');
|
|
158
|
+
expect(samples).toEqual([]);
|
|
159
|
+
});
|
|
160
|
+
it('returns empty when items empty', () => {
|
|
161
|
+
const { samples } = selectAuditSample([], 10, () => 'x');
|
|
162
|
+
expect(samples).toEqual([]);
|
|
163
|
+
});
|
|
164
|
+
it('honors prior defect rates for variance estimation', () => {
|
|
165
|
+
const items = Array(100).fill(0).map((_, i) => ({ id: i, kind: i < 50 ? 'safe' : 'risky' }));
|
|
166
|
+
// Risky stratum has higher prior defect rate → higher σ → more samples.
|
|
167
|
+
const { byStratum } = selectAuditSample(items, 20, (it) => it.kind, { safe: 0.01, risky: 0.40 });
|
|
168
|
+
// We don't pin exact values — RNG-dependent — but risky should get >= safe.
|
|
169
|
+
expect(byStratum.risky.sampled).toBeGreaterThanOrEqual(byStratum.safe.sampled);
|
|
170
|
+
});
|
|
171
|
+
});
|
|
172
|
+
//# sourceMappingURL=audit-sampler.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"audit-sampler.test.js","sourceRoot":"","sources":["../../src/__tests__/audit-sampler.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"audit-spc.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/audit-spc.test.ts"],"names":[],"mappings":"AAAA;;GAEG"}
|
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* NavGator audit SPC tests — Run 2 / D6
|
|
3
|
+
*/
|
|
4
|
+
import { describe, it, expect } from 'vitest';
|
|
5
|
+
import { newEwmaState, updateEwma } from '../audit/spc.js';
|
|
6
|
+
describe('newEwmaState', () => {
|
|
7
|
+
it('default lambda=0.2, L=2.7', () => {
|
|
8
|
+
const s = newEwmaState();
|
|
9
|
+
expect(s.lambda).toBeCloseTo(0.2, 6);
|
|
10
|
+
expect(s.L).toBeCloseTo(2.7, 6);
|
|
11
|
+
expect(s.n).toBe(0);
|
|
12
|
+
expect(s.points).toEqual([]);
|
|
13
|
+
});
|
|
14
|
+
it('honors overrides', () => {
|
|
15
|
+
const s = newEwmaState(0.1, 3);
|
|
16
|
+
expect(s.lambda).toBe(0.1);
|
|
17
|
+
expect(s.L).toBe(3);
|
|
18
|
+
});
|
|
19
|
+
});
|
|
20
|
+
describe('updateEwma — warm-up phase', () => {
|
|
21
|
+
it('does not breach during the first 5 observations even with extreme values', () => {
|
|
22
|
+
let s = newEwmaState();
|
|
23
|
+
for (const x of [0.5, 0.6, 0.7, 0.8, 0.9]) {
|
|
24
|
+
const r = updateEwma(s, x);
|
|
25
|
+
expect(r.breach).toBe(false);
|
|
26
|
+
s = r.state;
|
|
27
|
+
}
|
|
28
|
+
expect(s.n).toBe(5);
|
|
29
|
+
});
|
|
30
|
+
it('estimates mean and variance from the warm-up window', () => {
|
|
31
|
+
let s = newEwmaState();
|
|
32
|
+
for (const x of [0.02, 0.02, 0.02, 0.02, 0.02]) {
|
|
33
|
+
s = updateEwma(s, x).state;
|
|
34
|
+
}
|
|
35
|
+
expect(s.mean).toBeCloseTo(0.02, 6);
|
|
36
|
+
expect(s.variance).toBeCloseTo(0, 6);
|
|
37
|
+
});
|
|
38
|
+
});
|
|
39
|
+
describe('updateEwma — control phase', () => {
|
|
40
|
+
it('stays in control for stable observations', () => {
|
|
41
|
+
let s = newEwmaState();
|
|
42
|
+
// Warm-up with mean=0.02, σ≈0.005
|
|
43
|
+
const warm = [0.015, 0.020, 0.025, 0.020, 0.020];
|
|
44
|
+
for (const x of warm)
|
|
45
|
+
s = updateEwma(s, x).state;
|
|
46
|
+
// 10 more stable points
|
|
47
|
+
for (let i = 0; i < 10; i++) {
|
|
48
|
+
const r = updateEwma(s, 0.02 + (Math.random() - 0.5) * 0.002);
|
|
49
|
+
expect(r.breach).toBe(false);
|
|
50
|
+
s = r.state;
|
|
51
|
+
}
|
|
52
|
+
});
|
|
53
|
+
it('breaches on a sustained large shift', () => {
|
|
54
|
+
let s = newEwmaState();
|
|
55
|
+
const warm = [0.020, 0.020, 0.020, 0.020, 0.020];
|
|
56
|
+
for (const x of warm)
|
|
57
|
+
s = updateEwma(s, x).state;
|
|
58
|
+
// Even with σ→0 (perfectly stable warm-up) the variance floors at 1e-12,
|
|
59
|
+
// so any meaningful shift breaches. We force a shift to 0.10.
|
|
60
|
+
let breached = false;
|
|
61
|
+
for (let i = 0; i < 5; i++) {
|
|
62
|
+
const r = updateEwma(s, 0.10);
|
|
63
|
+
s = r.state;
|
|
64
|
+
if (r.breach) {
|
|
65
|
+
breached = true;
|
|
66
|
+
break;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
expect(breached).toBe(true);
|
|
70
|
+
});
|
|
71
|
+
it('records control limits as ±L·σ·√(λ/(2-λ))·stabilizer', () => {
|
|
72
|
+
let s = newEwmaState();
|
|
73
|
+
// Warm-up with non-zero variance.
|
|
74
|
+
const warm = [0.02, 0.04, 0.02, 0.04, 0.02];
|
|
75
|
+
for (const x of warm)
|
|
76
|
+
s = updateEwma(s, x).state;
|
|
77
|
+
const r = updateEwma(s, 0.03);
|
|
78
|
+
// After warm-up + 1 obs, k=1, stabilizer=1 - (0.8)^2 = 0.36
|
|
79
|
+
// se = σ · √(0.2/1.8 · 0.36) = σ · √0.04 = σ · 0.2
|
|
80
|
+
// ucl = 2.7 · σ · 0.2
|
|
81
|
+
const sigma = Math.sqrt(s.variance);
|
|
82
|
+
const expectedUcl = 2.7 * sigma * Math.sqrt((0.2 / 1.8) * (1 - Math.pow(0.8, 2)));
|
|
83
|
+
expect(r.ucl).toBeCloseTo(expectedUcl, 6);
|
|
84
|
+
expect(r.lcl).toBeCloseTo(-expectedUcl, 6);
|
|
85
|
+
});
|
|
86
|
+
it('caps points history at 50', () => {
|
|
87
|
+
let s = newEwmaState();
|
|
88
|
+
for (let i = 0; i < 80; i++) {
|
|
89
|
+
s = updateEwma(s, 0.02).state;
|
|
90
|
+
}
|
|
91
|
+
expect(s.points.length).toBe(50);
|
|
92
|
+
});
|
|
93
|
+
});
|
|
94
|
+
//# sourceMappingURL=audit-spc.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"audit-spc.test.js","sourceRoot":"","sources":["../../src/__tests__/audit-spc.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"audit-verifiers.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/audit-verifiers.test.ts"],"names":[],"mappings":"AAAA;;GAEG"}
|
|
@@ -0,0 +1,248 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* NavGator audit verifiers tests — Run 2 / D6
|
|
3
|
+
*/
|
|
4
|
+
import * as fs from 'fs';
|
|
5
|
+
import * as os from 'os';
|
|
6
|
+
import * as path from 'path';
|
|
7
|
+
import * as crypto from 'crypto';
|
|
8
|
+
import { afterEach, beforeEach, describe, expect, it } from 'vitest';
|
|
9
|
+
import { verifyDedupCollision, verifyHallucinatedComponent, verifyHallucinatedEdge, verifyMissedEdge, verifyStaleReference, verifyWrongEndpoint, } from '../audit/verifiers.js';
|
|
10
|
+
// ============================================================================
|
|
11
|
+
// FIXTURES
|
|
12
|
+
// ============================================================================
|
|
13
|
+
let workDir;
|
|
14
|
+
beforeEach(() => {
|
|
15
|
+
workDir = fs.mkdtempSync(path.join(os.tmpdir(), 'navgator-audit-test-'));
|
|
16
|
+
});
|
|
17
|
+
afterEach(() => {
|
|
18
|
+
fs.rmSync(workDir, { recursive: true, force: true });
|
|
19
|
+
});
|
|
20
|
+
function makeComponent(overrides = {}) {
|
|
21
|
+
return {
|
|
22
|
+
component_id: 'COMP_npm_x_1234',
|
|
23
|
+
name: 'x',
|
|
24
|
+
type: 'npm',
|
|
25
|
+
role: { purpose: 'pkg', layer: 'backend', critical: false },
|
|
26
|
+
source: { detection_method: 'auto', config_files: ['package.json'], confidence: 1 },
|
|
27
|
+
...overrides,
|
|
28
|
+
};
|
|
29
|
+
}
|
|
30
|
+
function makeConnection(overrides = {}) {
|
|
31
|
+
return {
|
|
32
|
+
connection_id: 'CONN_imports_abc',
|
|
33
|
+
from: { component_id: 'COMP_a', location: { file: 'a.ts', line: 1 } },
|
|
34
|
+
to: { component_id: 'COMP_b', location: { file: 'b.ts', line: 1 } },
|
|
35
|
+
connection_type: 'imports',
|
|
36
|
+
code_reference: { file: 'a.ts', symbol: 'foo', line_start: 1, line_end: 1 },
|
|
37
|
+
detected_from: 'test',
|
|
38
|
+
confidence: 0.9,
|
|
39
|
+
timestamp: 0,
|
|
40
|
+
last_verified: 0,
|
|
41
|
+
...overrides,
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
function ctxFor(overrides = {}) {
|
|
45
|
+
return {
|
|
46
|
+
projectRoot: workDir,
|
|
47
|
+
hashes: null,
|
|
48
|
+
componentById: new Map(),
|
|
49
|
+
isMcpMode: false,
|
|
50
|
+
...overrides,
|
|
51
|
+
};
|
|
52
|
+
}
|
|
53
|
+
// ============================================================================
|
|
54
|
+
// V1 — HALLUCINATED_COMPONENT
|
|
55
|
+
// ============================================================================
|
|
56
|
+
describe('verifyHallucinatedComponent', () => {
|
|
57
|
+
it('positive case: config file does not exist on disk → defect', async () => {
|
|
58
|
+
const comp = makeComponent({ component_id: 'COMP_npm_ghost', source: { detection_method: 'auto', config_files: ['does-not-exist.json'], confidence: 1 } });
|
|
59
|
+
const out = await verifyHallucinatedComponent([comp], ctxFor());
|
|
60
|
+
expect(out.defectCount).toBe(1);
|
|
61
|
+
expect(out.samples[0].ok).toBe(false);
|
|
62
|
+
});
|
|
63
|
+
it('negative case: config file exists → clean', async () => {
|
|
64
|
+
fs.writeFileSync(path.join(workDir, 'package.json'), '{}');
|
|
65
|
+
const comp = makeComponent();
|
|
66
|
+
const out = await verifyHallucinatedComponent([comp], ctxFor());
|
|
67
|
+
expect(out.defectCount).toBe(0);
|
|
68
|
+
expect(out.samples[0].ok).toBe(true);
|
|
69
|
+
});
|
|
70
|
+
});
|
|
71
|
+
// ============================================================================
|
|
72
|
+
// V2 — HALLUCINATED_EDGE
|
|
73
|
+
// ============================================================================
|
|
74
|
+
describe('verifyHallucinatedEdge', () => {
|
|
75
|
+
it('positive case: from/to component_ids unresolved → defect', () => {
|
|
76
|
+
const conn = makeConnection({
|
|
77
|
+
from: { component_id: 'GHOST_FROM', location: { file: 'a.ts', line: 1 } },
|
|
78
|
+
to: { component_id: 'GHOST_TO' },
|
|
79
|
+
});
|
|
80
|
+
const out = verifyHallucinatedEdge([conn], ctxFor());
|
|
81
|
+
expect(out.defectCount).toBe(1);
|
|
82
|
+
expect(out.samples[0].reason).toContain('both endpoints unresolved');
|
|
83
|
+
});
|
|
84
|
+
it('negative case: both endpoints resolve → clean', () => {
|
|
85
|
+
const a = makeComponent({ component_id: 'COMP_a' });
|
|
86
|
+
const b = makeComponent({ component_id: 'COMP_b' });
|
|
87
|
+
const map = new Map();
|
|
88
|
+
map.set('COMP_a', a);
|
|
89
|
+
map.set('COMP_b', b);
|
|
90
|
+
const out = verifyHallucinatedEdge([makeConnection()], ctxFor({ componentById: map }));
|
|
91
|
+
expect(out.defectCount).toBe(0);
|
|
92
|
+
});
|
|
93
|
+
});
|
|
94
|
+
// ============================================================================
|
|
95
|
+
// V3 — WRONG_ENDPOINT
|
|
96
|
+
// ============================================================================
|
|
97
|
+
describe('verifyWrongEndpoint', () => {
|
|
98
|
+
it('positive case: file present but symbol absent → defect', async () => {
|
|
99
|
+
fs.writeFileSync(path.join(workDir, 'a.ts'), '// nothing relevant here');
|
|
100
|
+
const target = makeComponent({ component_id: 'COMP_b', name: 'targetThingDef' });
|
|
101
|
+
const map = new Map([['COMP_b', target]]);
|
|
102
|
+
const conn = makeConnection({ code_reference: { file: 'a.ts', symbol: 'fooMissingSymbol' } });
|
|
103
|
+
const out = await verifyWrongEndpoint([conn], ctxFor({ componentById: map }));
|
|
104
|
+
expect(out.defectCount).toBe(1);
|
|
105
|
+
});
|
|
106
|
+
it('negative case: symbol present in file → clean', async () => {
|
|
107
|
+
fs.writeFileSync(path.join(workDir, 'a.ts'), 'export function foo() {}');
|
|
108
|
+
const target = makeComponent({ component_id: 'COMP_b', name: 'foo' });
|
|
109
|
+
const map = new Map([['COMP_b', target]]);
|
|
110
|
+
const conn = makeConnection({ code_reference: { file: 'a.ts', symbol: 'foo' } });
|
|
111
|
+
const out = await verifyWrongEndpoint([conn], ctxFor({ componentById: map }));
|
|
112
|
+
expect(out.defectCount).toBe(0);
|
|
113
|
+
});
|
|
114
|
+
// Run 3 D2a: path-style symbol regression — \b fails on non-word boundaries
|
|
115
|
+
it('Run 3 path-style symbol present in file → clean (no false defect)', async () => {
|
|
116
|
+
fs.writeFileSync(path.join(workDir, 'caller.ts'), "import { foo } from './entity-analysis-service';\n");
|
|
117
|
+
const target = makeComponent({ component_id: 'COMP_t', name: 'services/entity-analysis-service' });
|
|
118
|
+
const map = new Map([['COMP_t', target]]);
|
|
119
|
+
const conn = makeConnection({
|
|
120
|
+
code_reference: { file: 'caller.ts', symbol: './entity-analysis-service' },
|
|
121
|
+
});
|
|
122
|
+
const out = await verifyWrongEndpoint([conn], ctxFor({ componentById: map }));
|
|
123
|
+
expect(out.defectCount).toBe(0);
|
|
124
|
+
});
|
|
125
|
+
it('Run 3 path-style symbol with hyphens (../fixtures/factories) → clean', async () => {
|
|
126
|
+
fs.writeFileSync(path.join(workDir, 'test.ts'), "import { make } from '../fixtures/factories';\n");
|
|
127
|
+
const map = new Map();
|
|
128
|
+
const conn = makeConnection({
|
|
129
|
+
code_reference: { file: 'test.ts', symbol: '../fixtures/factories' },
|
|
130
|
+
});
|
|
131
|
+
const out = await verifyWrongEndpoint([conn], ctxFor({ componentById: map }));
|
|
132
|
+
expect(out.defectCount).toBe(0);
|
|
133
|
+
});
|
|
134
|
+
it('Run 3 scoped package symbol (@scope/pkg) present → clean', async () => {
|
|
135
|
+
fs.writeFileSync(path.join(workDir, 'a.ts'), "import x from '@radix-ui/react-dialog';\n");
|
|
136
|
+
const target = makeComponent({ component_id: 'COMP_pkg', name: '@radix-ui/react-dialog' });
|
|
137
|
+
const map = new Map([['COMP_pkg', target]]);
|
|
138
|
+
const conn = makeConnection({
|
|
139
|
+
code_reference: { file: 'a.ts', symbol: '@radix-ui/react-dialog' },
|
|
140
|
+
});
|
|
141
|
+
const out = await verifyWrongEndpoint([conn], ctxFor({ componentById: map }));
|
|
142
|
+
expect(out.defectCount).toBe(0);
|
|
143
|
+
});
|
|
144
|
+
it('Run 3 identifier symbol still uses \\b — substring of larger word does NOT match', async () => {
|
|
145
|
+
// File contains `foobar` only; symbol `foo` should NOT match (false-positive
|
|
146
|
+
// resistance for plain identifiers — keeps the safety the original regex provided).
|
|
147
|
+
fs.writeFileSync(path.join(workDir, 'a.ts'), 'const x = foobar;\n');
|
|
148
|
+
const map = new Map();
|
|
149
|
+
const conn = makeConnection({ code_reference: { file: 'a.ts', symbol: 'foo' } });
|
|
150
|
+
const out = await verifyWrongEndpoint([conn], ctxFor({ componentById: map }));
|
|
151
|
+
expect(out.defectCount).toBe(1);
|
|
152
|
+
});
|
|
153
|
+
});
|
|
154
|
+
// ============================================================================
|
|
155
|
+
// V4 — STALE_REFERENCE
|
|
156
|
+
// ============================================================================
|
|
157
|
+
describe('verifyStaleReference', () => {
|
|
158
|
+
it('positive case: hash mismatch → defect', async () => {
|
|
159
|
+
const file = path.join(workDir, 'src/foo.ts');
|
|
160
|
+
fs.mkdirSync(path.dirname(file), { recursive: true });
|
|
161
|
+
fs.writeFileSync(file, 'current content');
|
|
162
|
+
const recorded = crypto.createHash('sha256').update('OLD content').digest('hex');
|
|
163
|
+
const hashes = {
|
|
164
|
+
version: '1.0',
|
|
165
|
+
generatedAt: 0,
|
|
166
|
+
projectPath: workDir,
|
|
167
|
+
files: { 'src/foo.ts': { hash: recorded, lastScanned: 0, size: 0 } },
|
|
168
|
+
};
|
|
169
|
+
const out = await verifyStaleReference(['src/foo.ts'], ctxFor({ hashes }));
|
|
170
|
+
expect(out.defectCount).toBe(1);
|
|
171
|
+
expect(out.samples[0].reason).toContain('hash mismatch');
|
|
172
|
+
});
|
|
173
|
+
it('positive case: file deleted → defect', async () => {
|
|
174
|
+
const recorded = crypto.createHash('sha256').update('was here').digest('hex');
|
|
175
|
+
const hashes = {
|
|
176
|
+
version: '1.0',
|
|
177
|
+
generatedAt: 0,
|
|
178
|
+
projectPath: workDir,
|
|
179
|
+
files: { 'gone.ts': { hash: recorded, lastScanned: 0, size: 0 } },
|
|
180
|
+
};
|
|
181
|
+
const out = await verifyStaleReference(['gone.ts'], ctxFor({ hashes }));
|
|
182
|
+
expect(out.defectCount).toBe(1);
|
|
183
|
+
});
|
|
184
|
+
it('negative case: hash matches → clean', async () => {
|
|
185
|
+
const file = path.join(workDir, 'foo.ts');
|
|
186
|
+
fs.writeFileSync(file, 'stable content');
|
|
187
|
+
const recorded = crypto.createHash('sha256').update('stable content').digest('hex');
|
|
188
|
+
const hashes = {
|
|
189
|
+
version: '1.0',
|
|
190
|
+
generatedAt: 0,
|
|
191
|
+
projectPath: workDir,
|
|
192
|
+
files: { 'foo.ts': { hash: recorded, lastScanned: 0, size: 0 } },
|
|
193
|
+
};
|
|
194
|
+
const out = await verifyStaleReference(['foo.ts'], ctxFor({ hashes }));
|
|
195
|
+
expect(out.defectCount).toBe(0);
|
|
196
|
+
});
|
|
197
|
+
it('handles missing hashes.json (no defect)', async () => {
|
|
198
|
+
const out = await verifyStaleReference(['x.ts'], ctxFor({ hashes: null }));
|
|
199
|
+
expect(out.defectCount).toBe(0);
|
|
200
|
+
});
|
|
201
|
+
});
|
|
202
|
+
// ============================================================================
|
|
203
|
+
// V5 — DEDUP_COLLISION
|
|
204
|
+
// ============================================================================
|
|
205
|
+
describe('verifyDedupCollision', () => {
|
|
206
|
+
it('positive case: duplicate (type,name,primary-config) triple → defect', () => {
|
|
207
|
+
const a = makeComponent({ component_id: 'COMP_1', type: 'npm', name: 'react', source: { detection_method: 'auto', config_files: ['package.json'], confidence: 1 } });
|
|
208
|
+
const b = makeComponent({ component_id: 'COMP_2', type: 'npm', name: 'react', source: { detection_method: 'auto', config_files: ['package.json'], confidence: 1 } });
|
|
209
|
+
const out = verifyDedupCollision([a, b]);
|
|
210
|
+
expect(out.defectCount).toBe(1);
|
|
211
|
+
expect(out.samples[0].reason).toContain('dedup-key collision');
|
|
212
|
+
});
|
|
213
|
+
it('negative case: same name, different type → clean', () => {
|
|
214
|
+
const a = makeComponent({ component_id: 'COMP_1', type: 'npm', name: 'prisma' });
|
|
215
|
+
const b = makeComponent({ component_id: 'COMP_2', type: 'database', name: 'prisma' });
|
|
216
|
+
const out = verifyDedupCollision([a, b]);
|
|
217
|
+
expect(out.defectCount).toBe(0);
|
|
218
|
+
});
|
|
219
|
+
it('negative case: same name + type, different file → clean', () => {
|
|
220
|
+
const a = makeComponent({ component_id: 'COMP_1', source: { detection_method: 'auto', config_files: ['app/proxy.ts'], confidence: 1 } });
|
|
221
|
+
const b = makeComponent({ component_id: 'COMP_2', source: { detection_method: 'auto', config_files: ['lib/proxy.ts'], confidence: 1 } });
|
|
222
|
+
const out = verifyDedupCollision([a, b]);
|
|
223
|
+
expect(out.defectCount).toBe(0);
|
|
224
|
+
});
|
|
225
|
+
});
|
|
226
|
+
// ============================================================================
|
|
227
|
+
// V6 — MISSED_EDGE (LLM-judge)
|
|
228
|
+
// ============================================================================
|
|
229
|
+
describe('verifyMissedEdge', () => {
|
|
230
|
+
it('CLI mode: returns llm_skipped=true with zero defects', () => {
|
|
231
|
+
const out = verifyMissedEdge(['foo.ts'], [], ctxFor({ isMcpMode: false }));
|
|
232
|
+
expect(out.llm_skipped).toBe(true);
|
|
233
|
+
expect(out.defectCount).toBe(0);
|
|
234
|
+
});
|
|
235
|
+
it('MCP mode: returns structured llm_payload with recorded edges', () => {
|
|
236
|
+
const conn = makeConnection({
|
|
237
|
+
connection_id: 'CONN_abc',
|
|
238
|
+
code_reference: { file: 'src/index.ts', symbol: 'doThing' },
|
|
239
|
+
});
|
|
240
|
+
const out = verifyMissedEdge(['src/index.ts'], [conn], ctxFor({ isMcpMode: true }));
|
|
241
|
+
expect(out.llm_skipped).toBeUndefined();
|
|
242
|
+
expect(out.llm_payload).toBeDefined();
|
|
243
|
+
const payload = out.llm_payload;
|
|
244
|
+
expect(payload.files[0].path).toBe('src/index.ts');
|
|
245
|
+
expect(payload.files[0].recorded_outgoing_edges.length).toBe(1);
|
|
246
|
+
});
|
|
247
|
+
});
|
|
248
|
+
//# sourceMappingURL=audit-verifiers.test.js.map
|