@lyse-labs/lyse 0.1.0-alpha.2
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/LICENSE +45 -0
- package/PRIVACY.md +181 -0
- package/README.md +34 -0
- package/SECURITY.md +43 -0
- package/dist/bench/evidence-pack/anti-dummy.d.ts +11 -0
- package/dist/bench/evidence-pack/anti-dummy.js +20 -0
- package/dist/bench/evidence-pack/anti-dummy.js.map +1 -0
- package/dist/bench/evidence-pack/builder.d.ts +14 -0
- package/dist/bench/evidence-pack/builder.js +77 -0
- package/dist/bench/evidence-pack/builder.js.map +1 -0
- package/dist/bench/evidence-pack/histograms.d.ts +2 -0
- package/dist/bench/evidence-pack/histograms.js +131 -0
- package/dist/bench/evidence-pack/histograms.js.map +1 -0
- package/dist/bench/evidence-pack/manifest-detector.d.ts +2 -0
- package/dist/bench/evidence-pack/manifest-detector.js +65 -0
- package/dist/bench/evidence-pack/manifest-detector.js.map +1 -0
- package/dist/bench/evidence-pack/package-json-digest.d.ts +2 -0
- package/dist/bench/evidence-pack/package-json-digest.js +51 -0
- package/dist/bench/evidence-pack/package-json-digest.js.map +1 -0
- package/dist/bench/evidence-pack/sampler.d.ts +2 -0
- package/dist/bench/evidence-pack/sampler.js +140 -0
- package/dist/bench/evidence-pack/sampler.js.map +1 -0
- package/dist/bench/evidence-pack/types.d.ts +127 -0
- package/dist/bench/evidence-pack/types.js +2 -0
- package/dist/bench/evidence-pack/types.js.map +1 -0
- package/dist/bench/evidence-pack/verifier-corpus.d.ts +5 -0
- package/dist/bench/evidence-pack/verifier-corpus.js +13 -0
- package/dist/bench/evidence-pack/verifier-corpus.js.map +1 -0
- package/dist/bench/taxonomy-loader.d.ts +20 -0
- package/dist/bench/taxonomy-loader.js +28 -0
- package/dist/bench/taxonomy-loader.js.map +1 -0
- package/dist/bench/taxonomy.v3.json +20 -0
- package/dist/cli/__tests__/version-migration.test.d.ts +1 -0
- package/dist/cli/__tests__/version-migration.test.js +69 -0
- package/dist/cli/__tests__/version-migration.test.js.map +1 -0
- package/dist/cli/output/__tests__/eslint-style.test.d.ts +1 -0
- package/dist/cli/output/__tests__/eslint-style.test.js +205 -0
- package/dist/cli/output/__tests__/eslint-style.test.js.map +1 -0
- package/dist/cli/output/__tests__/limit.test.d.ts +1 -0
- package/dist/cli/output/__tests__/limit.test.js +65 -0
- package/dist/cli/output/__tests__/limit.test.js.map +1 -0
- package/dist/cli/output/eslint-style.d.ts +18 -0
- package/dist/cli/output/eslint-style.js +42 -0
- package/dist/cli/output/eslint-style.js.map +1 -0
- package/dist/cli/output/limit.d.ts +5 -0
- package/dist/cli/output/limit.js +32 -0
- package/dist/cli/output/limit.js.map +1 -0
- package/dist/cli/output/score-gauge.d.ts +4 -0
- package/dist/cli/output/score-gauge.js +15 -0
- package/dist/cli/output/score-gauge.js.map +1 -0
- package/dist/cli/version-migration.d.ts +18 -0
- package/dist/cli/version-migration.js +49 -0
- package/dist/cli/version-migration.js.map +1 -0
- package/dist/cli.d.ts +2 -0
- package/dist/cli.js +856 -0
- package/dist/cli.js.map +1 -0
- package/dist/codemods/diff.d.ts +17 -0
- package/dist/codemods/diff.js +77 -0
- package/dist/codemods/diff.js.map +1 -0
- package/dist/codemods/git-helpers.d.ts +10 -0
- package/dist/codemods/git-helpers.js +75 -0
- package/dist/codemods/git-helpers.js.map +1 -0
- package/dist/codemods/index.d.ts +27 -0
- package/dist/codemods/index.js +40 -0
- package/dist/codemods/index.js.map +1 -0
- package/dist/codemods/naming-component-pascalcase.d.ts +13 -0
- package/dist/codemods/naming-component-pascalcase.js +106 -0
- package/dist/codemods/naming-component-pascalcase.js.map +1 -0
- package/dist/codemods/naming-hook-prefix.d.ts +13 -0
- package/dist/codemods/naming-hook-prefix.js +86 -0
- package/dist/codemods/naming-hook-prefix.js.map +1 -0
- package/dist/codemods/safety.d.ts +50 -0
- package/dist/codemods/safety.js +109 -0
- package/dist/codemods/safety.js.map +1 -0
- package/dist/codemods/safety.test.d.ts +1 -0
- package/dist/codemods/safety.test.js +154 -0
- package/dist/codemods/safety.test.js.map +1 -0
- package/dist/codemods/shadow-native.d.ts +2 -0
- package/dist/codemods/shadow-native.js +103 -0
- package/dist/codemods/shadow-native.js.map +1 -0
- package/dist/codemods/tokens-color.d.ts +2 -0
- package/dist/codemods/tokens-color.js +69 -0
- package/dist/codemods/tokens-color.js.map +1 -0
- package/dist/codemods/tokens-spacing.d.ts +2 -0
- package/dist/codemods/tokens-spacing.js +66 -0
- package/dist/codemods/tokens-spacing.js.map +1 -0
- package/dist/commands/__tests__/email-prompt.test.d.ts +1 -0
- package/dist/commands/__tests__/email-prompt.test.js +126 -0
- package/dist/commands/__tests__/email-prompt.test.js.map +1 -0
- package/dist/commands/__tests__/explain-score.test.d.ts +1 -0
- package/dist/commands/__tests__/explain-score.test.js +88 -0
- package/dist/commands/__tests__/explain-score.test.js.map +1 -0
- package/dist/commands/__tests__/feedback.test.d.ts +1 -0
- package/dist/commands/__tests__/feedback.test.js +186 -0
- package/dist/commands/__tests__/feedback.test.js.map +1 -0
- package/dist/commands/audit-flags.d.ts +49 -0
- package/dist/commands/audit-flags.js +17 -0
- package/dist/commands/audit-flags.js.map +1 -0
- package/dist/commands/audit-pipeline.d.ts +31 -0
- package/dist/commands/audit-pipeline.js +342 -0
- package/dist/commands/audit-pipeline.js.map +1 -0
- package/dist/commands/bench-pack.d.ts +5 -0
- package/dist/commands/bench-pack.js +61 -0
- package/dist/commands/bench-pack.js.map +1 -0
- package/dist/commands/ci-setup.d.ts +9 -0
- package/dist/commands/ci-setup.js +42 -0
- package/dist/commands/ci-setup.js.map +1 -0
- package/dist/commands/email-prompt.d.ts +36 -0
- package/dist/commands/email-prompt.js +160 -0
- package/dist/commands/email-prompt.js.map +1 -0
- package/dist/commands/explain-score.d.ts +35 -0
- package/dist/commands/explain-score.js +137 -0
- package/dist/commands/explain-score.js.map +1 -0
- package/dist/commands/explain.d.ts +6 -0
- package/dist/commands/explain.js +93 -0
- package/dist/commands/explain.js.map +1 -0
- package/dist/commands/feedback.d.ts +31 -0
- package/dist/commands/feedback.js +155 -0
- package/dist/commands/feedback.js.map +1 -0
- package/dist/commands/fix.d.ts +48 -0
- package/dist/commands/fix.js +191 -0
- package/dist/commands/fix.js.map +1 -0
- package/dist/commands/init-detect.d.ts +20 -0
- package/dist/commands/init-detect.js +124 -0
- package/dist/commands/init-detect.js.map +1 -0
- package/dist/commands/init-write-agents-md.d.ts +14 -0
- package/dist/commands/init-write-agents-md.js +71 -0
- package/dist/commands/init-write-agents-md.js.map +1 -0
- package/dist/commands/init-write-lyse-md.d.ts +14 -0
- package/dist/commands/init-write-lyse-md.js +253 -0
- package/dist/commands/init-write-lyse-md.js.map +1 -0
- package/dist/commands/init.d.ts +8 -0
- package/dist/commands/init.js +147 -0
- package/dist/commands/init.js.map +1 -0
- package/dist/commands/mcp-entry.d.ts +33 -0
- package/dist/commands/mcp-entry.js +47 -0
- package/dist/commands/mcp-entry.js.map +1 -0
- package/dist/commands/mcp-setup.d.ts +10 -0
- package/dist/commands/mcp-setup.js +74 -0
- package/dist/commands/mcp-setup.js.map +1 -0
- package/dist/commands/share.d.ts +4 -0
- package/dist/commands/share.js +60 -0
- package/dist/commands/share.js.map +1 -0
- package/dist/commands/telemetry.d.ts +4 -0
- package/dist/commands/telemetry.js +27 -0
- package/dist/commands/telemetry.js.map +1 -0
- package/dist/commands/templates/lyse-workflow.yml.template +30 -0
- package/dist/config/schema.d.ts +158 -0
- package/dist/config/schema.js +136 -0
- package/dist/config/schema.js.map +1 -0
- package/dist/credentials/keychain.d.ts +20 -0
- package/dist/credentials/keychain.js +35 -0
- package/dist/credentials/keychain.js.map +1 -0
- package/dist/credentials/keychain.test.d.ts +1 -0
- package/dist/credentials/keychain.test.js +32 -0
- package/dist/credentials/keychain.test.js.map +1 -0
- package/dist/credentials/paths.d.ts +1 -0
- package/dist/credentials/paths.js +6 -0
- package/dist/credentials/paths.js.map +1 -0
- package/dist/credentials/store.d.ts +17 -0
- package/dist/credentials/store.js +60 -0
- package/dist/credentials/store.js.map +1 -0
- package/dist/credentials/store.test.d.ts +1 -0
- package/dist/credentials/store.test.js +48 -0
- package/dist/credentials/store.test.js.map +1 -0
- package/dist/detection/from-filesystem.d.ts +2 -0
- package/dist/detection/from-filesystem.js +26 -0
- package/dist/detection/from-filesystem.js.map +1 -0
- package/dist/detection/from-git.d.ts +2 -0
- package/dist/detection/from-git.js +53 -0
- package/dist/detection/from-git.js.map +1 -0
- package/dist/detection/from-package-json.d.ts +2 -0
- package/dist/detection/from-package-json.js +194 -0
- package/dist/detection/from-package-json.js.map +1 -0
- package/dist/detection/pre-flight.d.ts +5 -0
- package/dist/detection/pre-flight.js +47 -0
- package/dist/detection/pre-flight.js.map +1 -0
- package/dist/detection/types.d.ts +27 -0
- package/dist/detection/types.js +2 -0
- package/dist/detection/types.js.map +1 -0
- package/dist/entitlement/check.d.ts +7 -0
- package/dist/entitlement/check.js +37 -0
- package/dist/entitlement/check.js.map +1 -0
- package/dist/entitlement/index.d.ts +1 -0
- package/dist/entitlement/index.js +2 -0
- package/dist/entitlement/index.js.map +1 -0
- package/dist/entitlement/keys.d.ts +10 -0
- package/dist/entitlement/keys.js +14 -0
- package/dist/entitlement/keys.js.map +1 -0
- package/dist/history/ndjson-store.d.ts +70 -0
- package/dist/history/ndjson-store.js +102 -0
- package/dist/history/ndjson-store.js.map +1 -0
- package/dist/identity/index.d.ts +1 -0
- package/dist/identity/index.js +2 -0
- package/dist/identity/index.js.map +1 -0
- package/dist/identity/repo-bucket.d.ts +22 -0
- package/dist/identity/repo-bucket.js +78 -0
- package/dist/identity/repo-bucket.js.map +1 -0
- package/dist/index.d.ts +1 -0
- package/dist/index.js +15 -0
- package/dist/index.js.map +1 -0
- package/dist/llm/layer4-stage.d.ts +18 -0
- package/dist/llm/layer4-stage.js +12 -0
- package/dist/llm/layer4-stage.js.map +1 -0
- package/dist/loaders/components.d.ts +26 -0
- package/dist/loaders/components.js +285 -0
- package/dist/loaders/components.js.map +1 -0
- package/dist/loaders/stories.d.ts +2 -0
- package/dist/loaders/stories.js +231 -0
- package/dist/loaders/stories.js.map +1 -0
- package/dist/loaders/token-graph.d.ts +39 -0
- package/dist/loaders/token-graph.js +146 -0
- package/dist/loaders/token-graph.js.map +1 -0
- package/dist/loaders/tokens.d.ts +2 -0
- package/dist/loaders/tokens.js +521 -0
- package/dist/loaders/tokens.js.map +1 -0
- package/dist/mcp/_find-root.d.ts +12 -0
- package/dist/mcp/_find-root.js +28 -0
- package/dist/mcp/_find-root.js.map +1 -0
- package/dist/mcp/server.d.ts +1 -0
- package/dist/mcp/server.js +42 -0
- package/dist/mcp/server.js.map +1 -0
- package/dist/mcp/tools/audit-file.d.ts +25 -0
- package/dist/mcp/tools/audit-file.js +147 -0
- package/dist/mcp/tools/audit-file.js.map +1 -0
- package/dist/mcp/tools/suggest-fix.d.ts +13 -0
- package/dist/mcp/tools/suggest-fix.js +100 -0
- package/dist/mcp/tools/suggest-fix.js.map +1 -0
- package/dist/menu/action-menu.d.ts +6 -0
- package/dist/menu/action-menu.js +15 -0
- package/dist/menu/action-menu.js.map +1 -0
- package/dist/menu/prompts.d.ts +7 -0
- package/dist/menu/prompts.js +30 -0
- package/dist/menu/prompts.js.map +1 -0
- package/dist/menu/repl.d.ts +17 -0
- package/dist/menu/repl.js +77 -0
- package/dist/menu/repl.js.map +1 -0
- package/dist/parsers/css-in-js.d.ts +2 -0
- package/dist/parsers/css-in-js.js +69 -0
- package/dist/parsers/css-in-js.js.map +1 -0
- package/dist/parsers/css.d.ts +2 -0
- package/dist/parsers/css.js +32 -0
- package/dist/parsers/css.js.map +1 -0
- package/dist/parsers/scss-transform.d.ts +25 -0
- package/dist/parsers/scss-transform.js +55 -0
- package/dist/parsers/scss-transform.js.map +1 -0
- package/dist/parsers/tailwind-v4.d.ts +8 -0
- package/dist/parsers/tailwind-v4.js +90 -0
- package/dist/parsers/tailwind-v4.js.map +1 -0
- package/dist/parsers/ts-morph-project.d.ts +16 -0
- package/dist/parsers/ts-morph-project.js +77 -0
- package/dist/parsers/ts-morph-project.js.map +1 -0
- package/dist/parsers/ts.d.ts +2 -0
- package/dist/parsers/ts.js +61 -0
- package/dist/parsers/ts.js.map +1 -0
- package/dist/reliability/__tests__/types.test.d.ts +1 -0
- package/dist/reliability/__tests__/types.test.js +14 -0
- package/dist/reliability/__tests__/types.test.js.map +1 -0
- package/dist/reliability/catalogue/__tests__/promotion.test.d.ts +1 -0
- package/dist/reliability/catalogue/__tests__/promotion.test.js +23 -0
- package/dist/reliability/catalogue/__tests__/promotion.test.js.map +1 -0
- package/dist/reliability/catalogue/__tests__/sub-axes.test.d.ts +1 -0
- package/dist/reliability/catalogue/__tests__/sub-axes.test.js +27 -0
- package/dist/reliability/catalogue/__tests__/sub-axes.test.js.map +1 -0
- package/dist/reliability/catalogue/promotion.d.ts +8 -0
- package/dist/reliability/catalogue/promotion.js +25 -0
- package/dist/reliability/catalogue/promotion.js.map +1 -0
- package/dist/reliability/catalogue/sub-axes.d.ts +3 -0
- package/dist/reliability/catalogue/sub-axes.js +18 -0
- package/dist/reliability/catalogue/sub-axes.js.map +1 -0
- package/dist/reliability/confidence/__tests__/manifest-loader.test.d.ts +1 -0
- package/dist/reliability/confidence/__tests__/manifest-loader.test.js +103 -0
- package/dist/reliability/confidence/__tests__/manifest-loader.test.js.map +1 -0
- package/dist/reliability/confidence/bundled-manifest.d.ts +2 -0
- package/dist/reliability/confidence/bundled-manifest.js +8 -0
- package/dist/reliability/confidence/bundled-manifest.js.map +1 -0
- package/dist/reliability/confidence/index.d.ts +3 -0
- package/dist/reliability/confidence/index.js +3 -0
- package/dist/reliability/confidence/index.js.map +1 -0
- package/dist/reliability/confidence/manifest-loader.d.ts +8 -0
- package/dist/reliability/confidence/manifest-loader.js +53 -0
- package/dist/reliability/confidence/manifest-loader.js.map +1 -0
- package/dist/reliability/feedback/__tests__/interactive.test.d.ts +1 -0
- package/dist/reliability/feedback/__tests__/interactive.test.js +85 -0
- package/dist/reliability/feedback/__tests__/interactive.test.js.map +1 -0
- package/dist/reliability/feedback/__tests__/repo-bucket.test.d.ts +1 -0
- package/dist/reliability/feedback/__tests__/repo-bucket.test.js +18 -0
- package/dist/reliability/feedback/__tests__/repo-bucket.test.js.map +1 -0
- package/dist/reliability/feedback/__tests__/sender.test.d.ts +1 -0
- package/dist/reliability/feedback/__tests__/sender.test.js +101 -0
- package/dist/reliability/feedback/__tests__/sender.test.js.map +1 -0
- package/dist/reliability/feedback/interactive.d.ts +20 -0
- package/dist/reliability/feedback/interactive.js +150 -0
- package/dist/reliability/feedback/interactive.js.map +1 -0
- package/dist/reliability/feedback/repo-bucket.d.ts +2 -0
- package/dist/reliability/feedback/repo-bucket.js +9 -0
- package/dist/reliability/feedback/repo-bucket.js.map +1 -0
- package/dist/reliability/feedback/sender.d.ts +16 -0
- package/dist/reliability/feedback/sender.js +28 -0
- package/dist/reliability/feedback/sender.js.map +1 -0
- package/dist/reliability/index.d.ts +1 -0
- package/dist/reliability/index.js +2 -0
- package/dist/reliability/index.js.map +1 -0
- package/dist/reliability/llm-eval/__tests__/budget.test.d.ts +1 -0
- package/dist/reliability/llm-eval/__tests__/budget.test.js +39 -0
- package/dist/reliability/llm-eval/__tests__/budget.test.js.map +1 -0
- package/dist/reliability/llm-eval/budget.d.ts +14 -0
- package/dist/reliability/llm-eval/budget.js +43 -0
- package/dist/reliability/llm-eval/budget.js.map +1 -0
- package/dist/reliability/score/__tests__/formula-v1.test.d.ts +1 -0
- package/dist/reliability/score/__tests__/formula-v1.test.js +29 -0
- package/dist/reliability/score/__tests__/formula-v1.test.js.map +1 -0
- package/dist/reliability/score/formula-v1.d.ts +13 -0
- package/dist/reliability/score/formula-v1.js +19 -0
- package/dist/reliability/score/formula-v1.js.map +1 -0
- package/dist/reliability/score/version-pin.d.ts +1 -0
- package/dist/reliability/score/version-pin.js +2 -0
- package/dist/reliability/score/version-pin.js.map +1 -0
- package/dist/reliability/score/weight.d.ts +1 -0
- package/dist/reliability/score/weight.js +5 -0
- package/dist/reliability/score/weight.js.map +1 -0
- package/dist/reliability/types.d.ts +39 -0
- package/dist/reliability/types.js +2 -0
- package/dist/reliability/types.js.map +1 -0
- package/dist/reporters/coverage-footer.d.ts +2 -0
- package/dist/reporters/coverage-footer.js +7 -0
- package/dist/reporters/coverage-footer.js.map +1 -0
- package/dist/reporters/json.d.ts +5 -0
- package/dist/reporters/json.js +51 -0
- package/dist/reporters/json.js.map +1 -0
- package/dist/reporters/markdown.d.ts +7 -0
- package/dist/reporters/markdown.js +35 -0
- package/dist/reporters/markdown.js.map +1 -0
- package/dist/reporters/sarif.d.ts +5 -0
- package/dist/reporters/sarif.js +109 -0
- package/dist/reporters/sarif.js.map +1 -0
- package/dist/reporters/terminal-format.d.ts +34 -0
- package/dist/reporters/terminal-format.js +97 -0
- package/dist/reporters/terminal-format.js.map +1 -0
- package/dist/reporters/terminal.d.ts +4 -0
- package/dist/reporters/terminal.js +201 -0
- package/dist/reporters/terminal.js.map +1 -0
- package/dist/rule-runner.d.ts +8 -0
- package/dist/rule-runner.js +22 -0
- package/dist/rule-runner.js.map +1 -0
- package/dist/rules/_codemod-adapter.d.ts +16 -0
- package/dist/rules/_codemod-adapter.js +49 -0
- package/dist/rules/_codemod-adapter.js.map +1 -0
- package/dist/rules/_exclude.d.ts +11 -0
- package/dist/rules/_exclude.js +17 -0
- package/dist/rules/_exclude.js.map +1 -0
- package/dist/rules/_function-body-analysis.d.ts +82 -0
- package/dist/rules/_function-body-analysis.js +379 -0
- package/dist/rules/_function-body-analysis.js.map +1 -0
- package/dist/rules/_rule-module.d.ts +31 -0
- package/dist/rules/_rule-module.js +32 -0
- package/dist/rules/_rule-module.js.map +1 -0
- package/dist/rules/_skip-context.d.ts +36 -0
- package/dist/rules/_skip-context.js +128 -0
- package/dist/rules/_skip-context.js.map +1 -0
- package/dist/rules/a11y-essentials.d.ts +1 -0
- package/dist/rules/a11y-essentials.js +140 -0
- package/dist/rules/a11y-essentials.js.map +1 -0
- package/dist/rules/ai-surface-agents-md-quality.d.ts +22 -0
- package/dist/rules/ai-surface-agents-md-quality.js +233 -0
- package/dist/rules/ai-surface-agents-md-quality.js.map +1 -0
- package/dist/rules/ai-surface-component-manifest-json.d.ts +19 -0
- package/dist/rules/ai-surface-component-manifest-json.js +232 -0
- package/dist/rules/ai-surface-component-manifest-json.js.map +1 -0
- package/dist/rules/ai-surface-ds-index-exported.d.ts +19 -0
- package/dist/rules/ai-surface-ds-index-exported.js +239 -0
- package/dist/rules/ai-surface-ds-index-exported.js.map +1 -0
- package/dist/rules/components-shadow-native.d.ts +2 -0
- package/dist/rules/components-shadow-native.js +118 -0
- package/dist/rules/components-shadow-native.js.map +1 -0
- package/dist/rules/manifest.d.ts +5 -0
- package/dist/rules/manifest.js +20 -0
- package/dist/rules/manifest.js.map +1 -0
- package/dist/rules/naming-component-pascalcase.d.ts +7 -0
- package/dist/rules/naming-component-pascalcase.js +197 -0
- package/dist/rules/naming-component-pascalcase.js.map +1 -0
- package/dist/rules/naming-hook-prefix.d.ts +6 -0
- package/dist/rules/naming-hook-prefix.js +185 -0
- package/dist/rules/naming-hook-prefix.js.map +1 -0
- package/dist/rules/pack-loader.d.ts +30 -0
- package/dist/rules/pack-loader.js +60 -0
- package/dist/rules/pack-loader.js.map +1 -0
- package/dist/rules/pack-validator.d.ts +8 -0
- package/dist/rules/pack-validator.js +97 -0
- package/dist/rules/pack-validator.js.map +1 -0
- package/dist/rules/registry.d.ts +3 -0
- package/dist/rules/registry.js +28 -0
- package/dist/rules/registry.js.map +1 -0
- package/dist/rules/storybook-coverage.d.ts +1 -0
- package/dist/rules/storybook-coverage.js +49 -0
- package/dist/rules/storybook-coverage.js.map +1 -0
- package/dist/rules/templates/_regex-utils.d.ts +5 -0
- package/dist/rules/templates/_regex-utils.js +8 -0
- package/dist/rules/templates/_regex-utils.js.map +1 -0
- package/dist/rules/templates/a11y-jsx-template.d.ts +15 -0
- package/dist/rules/templates/a11y-jsx-template.js +69 -0
- package/dist/rules/templates/a11y-jsx-template.js.map +1 -0
- package/dist/rules/templates/css-property-token-compliance.d.ts +18 -0
- package/dist/rules/templates/css-property-token-compliance.js +56 -0
- package/dist/rules/templates/css-property-token-compliance.js.map +1 -0
- package/dist/rules/templates/import-source-restriction.d.ts +18 -0
- package/dist/rules/templates/import-source-restriction.js +60 -0
- package/dist/rules/templates/import-source-restriction.js.map +1 -0
- package/dist/rules/templates/js-call-token-compliance.d.ts +18 -0
- package/dist/rules/templates/js-call-token-compliance.js +61 -0
- package/dist/rules/templates/js-call-token-compliance.js.map +1 -0
- package/dist/rules/templates/js-prop-token-compliance.d.ts +21 -0
- package/dist/rules/templates/js-prop-token-compliance.js +56 -0
- package/dist/rules/templates/js-prop-token-compliance.js.map +1 -0
- package/dist/rules/templates/naming-convention.d.ts +18 -0
- package/dist/rules/templates/naming-convention.js +76 -0
- package/dist/rules/templates/naming-convention.js.map +1 -0
- package/dist/rules/templates/registry.d.ts +5 -0
- package/dist/rules/templates/registry.js +30 -0
- package/dist/rules/templates/registry.js.map +1 -0
- package/dist/rules/templates/storybook-coverage-template.d.ts +15 -0
- package/dist/rules/templates/storybook-coverage-template.js +58 -0
- package/dist/rules/templates/storybook-coverage-template.js.map +1 -0
- package/dist/rules/templates/tailwind-utility-class-compliance.d.ts +15 -0
- package/dist/rules/templates/tailwind-utility-class-compliance.js +70 -0
- package/dist/rules/templates/tailwind-utility-class-compliance.js.map +1 -0
- package/dist/rules/templates/types.d.ts +16 -0
- package/dist/rules/templates/types.js +2 -0
- package/dist/rules/templates/types.js.map +1 -0
- package/dist/rules/tokens-description-coverage.d.ts +15 -0
- package/dist/rules/tokens-description-coverage.js +193 -0
- package/dist/rules/tokens-description-coverage.js.map +1 -0
- package/dist/rules/tokens-dtcg-conformance.d.ts +47 -0
- package/dist/rules/tokens-dtcg-conformance.js +363 -0
- package/dist/rules/tokens-dtcg-conformance.js.map +1 -0
- package/dist/rules/tokens-no-hardcoded-color.d.ts +18 -0
- package/dist/rules/tokens-no-hardcoded-color.js +436 -0
- package/dist/rules/tokens-no-hardcoded-color.js.map +1 -0
- package/dist/rules/tokens-no-hardcoded-spacing.d.ts +8 -0
- package/dist/rules/tokens-no-hardcoded-spacing.js +193 -0
- package/dist/rules/tokens-no-hardcoded-spacing.js.map +1 -0
- package/dist/scorer.d.ts +38 -0
- package/dist/scorer.js +126 -0
- package/dist/scorer.js.map +1 -0
- package/dist/share/clipboard.d.ts +26 -0
- package/dist/share/clipboard.js +76 -0
- package/dist/share/clipboard.js.map +1 -0
- package/dist/suppression/__tests__/inline.test.d.ts +1 -0
- package/dist/suppression/__tests__/inline.test.js +25 -0
- package/dist/suppression/__tests__/inline.test.js.map +1 -0
- package/dist/suppression/__tests__/lyseignore.test.d.ts +1 -0
- package/dist/suppression/__tests__/lyseignore.test.js +32 -0
- package/dist/suppression/__tests__/lyseignore.test.js.map +1 -0
- package/dist/suppression/defaults.d.ts +1 -0
- package/dist/suppression/defaults.js +14 -0
- package/dist/suppression/defaults.js.map +1 -0
- package/dist/suppression/inline.d.ts +7 -0
- package/dist/suppression/inline.js +32 -0
- package/dist/suppression/inline.js.map +1 -0
- package/dist/suppression/lyseignore.d.ts +2 -0
- package/dist/suppression/lyseignore.js +22 -0
- package/dist/suppression/lyseignore.js.map +1 -0
- package/dist/telemetry/consent.d.ts +48 -0
- package/dist/telemetry/consent.js +139 -0
- package/dist/telemetry/consent.js.map +1 -0
- package/dist/telemetry/index.d.ts +2 -0
- package/dist/telemetry/index.js +3 -0
- package/dist/telemetry/index.js.map +1 -0
- package/dist/telemetry/local-log.d.ts +64 -0
- package/dist/telemetry/local-log.js +123 -0
- package/dist/telemetry/local-log.js.map +1 -0
- package/dist/tokens/dtcg-model.d.ts +53 -0
- package/dist/tokens/dtcg-model.js +18 -0
- package/dist/tokens/dtcg-model.js.map +1 -0
- package/dist/tokens/normalizer.d.ts +17 -0
- package/dist/tokens/normalizer.js +360 -0
- package/dist/tokens/normalizer.js.map +1 -0
- package/dist/types.d.ts +300 -0
- package/dist/types.js +4 -0
- package/dist/types.js.map +1 -0
- package/dist/util/git.d.ts +6 -0
- package/dist/util/git.js +40 -0
- package/dist/util/git.js.map +1 -0
- package/dist/util/git.test.d.ts +5 -0
- package/dist/util/git.test.js +42 -0
- package/dist/util/git.test.js.map +1 -0
- package/dist/util/gitignore.d.ts +9 -0
- package/dist/util/gitignore.js +39 -0
- package/dist/util/gitignore.js.map +1 -0
- package/dist/util/hash-deps.d.ts +6 -0
- package/dist/util/hash-deps.js +24 -0
- package/dist/util/hash-deps.js.map +1 -0
- package/dist/util/spinner.d.ts +41 -0
- package/dist/util/spinner.js +126 -0
- package/dist/util/spinner.js.map +1 -0
- package/dist/util/with-spinner.d.ts +11 -0
- package/dist/util/with-spinner.js +19 -0
- package/dist/util/with-spinner.js.map +1 -0
- package/dist/walker.d.ts +14 -0
- package/dist/walker.js +91 -0
- package/dist/walker.js.map +1 -0
- package/package.json +83 -0
- package/rules-manifest.json +289 -0
- package/schemas/v1/lyse-config.json +82 -0
- package/schemas/v1/lyse-event.json +68 -0
- package/schemas/v1/lyse-license.json +19 -0
- package/schemas/v1/lyse-llm-payload.json +53 -0
- package/schemas/v1/lyse-llm-response.json +45 -0
- package/schemas/v1/lyse-result.json +98 -0
- package/schemas/v1/lyse-rules.json +42 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"bundled-manifest.js","sourceRoot":"","sources":["../../../src/reliability/confidence/bundled-manifest.ts"],"names":[],"mappings":"AAEA,MAAM,CAAC,MAAM,gBAAgB,GAAuB;IAClD,OAAO,EAAE,YAAY;IACrB,WAAW,EAAE,sBAAsB;IACnC,SAAS,EAAE,sBAAsB;IACjC,UAAU,EAAE,sBAAsB;IAClC,OAAO,EAAE,EAAE;CACZ,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/reliability/confidence/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,sBAAsB,CAAC;AAEpD,OAAO,EAAE,gBAAgB,EAAE,MAAM,uBAAuB,CAAC"}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import type { ConfidenceManifest } from "../types.js";
|
|
2
|
+
export interface LoadManifestOpts {
|
|
3
|
+
url: string;
|
|
4
|
+
fallback: ConfidenceManifest;
|
|
5
|
+
pinnedDate?: string;
|
|
6
|
+
cacheDir?: string;
|
|
7
|
+
}
|
|
8
|
+
export declare function loadManifest(opts: LoadManifestOpts): Promise<ConfidenceManifest>;
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import { mkdir, readFile, stat, writeFile } from "node:fs/promises";
|
|
2
|
+
import { existsSync } from "node:fs";
|
|
3
|
+
import { homedir } from "node:os";
|
|
4
|
+
import { dirname, resolve } from "node:path";
|
|
5
|
+
const TTL_MS = 7 * 24 * 60 * 60 * 1000;
|
|
6
|
+
const FETCH_TIMEOUT_MS = 3000;
|
|
7
|
+
// Honors LYSE_CACHE_DIR env override for CI / testing; defaults to ~/.cache/lyse.
|
|
8
|
+
function resolveCacheFile(cacheDir) {
|
|
9
|
+
const dir = cacheDir ?? process.env["LYSE_CACHE_DIR"] ?? resolve(homedir(), ".cache", "lyse");
|
|
10
|
+
return resolve(dir, "manifest.json");
|
|
11
|
+
}
|
|
12
|
+
async function readFreshCache(cacheFile) {
|
|
13
|
+
if (!existsSync(cacheFile))
|
|
14
|
+
return null;
|
|
15
|
+
try {
|
|
16
|
+
const s = await stat(cacheFile);
|
|
17
|
+
if (Date.now() - s.mtimeMs >= TTL_MS)
|
|
18
|
+
return null;
|
|
19
|
+
const raw = await readFile(cacheFile, "utf8");
|
|
20
|
+
return JSON.parse(raw);
|
|
21
|
+
}
|
|
22
|
+
catch {
|
|
23
|
+
return null;
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
async function writeCache(cacheFile, manifest) {
|
|
27
|
+
try {
|
|
28
|
+
await mkdir(dirname(cacheFile), { recursive: true });
|
|
29
|
+
await writeFile(cacheFile, JSON.stringify(manifest), "utf8");
|
|
30
|
+
}
|
|
31
|
+
catch {
|
|
32
|
+
// Read-only filesystem must not break the loader.
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
export async function loadManifest(opts) {
|
|
36
|
+
const cacheFile = resolveCacheFile(opts.cacheDir);
|
|
37
|
+
const cached = await readFreshCache(cacheFile);
|
|
38
|
+
if (cached)
|
|
39
|
+
return cached;
|
|
40
|
+
try {
|
|
41
|
+
const url = opts.pinnedDate ? `${opts.url}?pinned=${opts.pinnedDate}` : opts.url;
|
|
42
|
+
const res = await fetch(url, { signal: AbortSignal.timeout(FETCH_TIMEOUT_MS) });
|
|
43
|
+
if (!res.ok)
|
|
44
|
+
throw new Error(`HTTP ${res.status}`);
|
|
45
|
+
const manifest = (await res.json());
|
|
46
|
+
await writeCache(cacheFile, manifest);
|
|
47
|
+
return manifest;
|
|
48
|
+
}
|
|
49
|
+
catch {
|
|
50
|
+
return opts.fallback;
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
//# sourceMappingURL=manifest-loader.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"manifest-loader.js","sourceRoot":"","sources":["../../../src/reliability/confidence/manifest-loader.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,kBAAkB,CAAC;AACpE,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACrC,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAClC,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAG7C,MAAM,MAAM,GAAG,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC;AACvC,MAAM,gBAAgB,GAAG,IAAI,CAAC;AAS9B,kFAAkF;AAClF,SAAS,gBAAgB,CAAC,QAAiB;IACzC,MAAM,GAAG,GAAG,QAAQ,IAAI,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC,IAAI,OAAO,CAAC,OAAO,EAAE,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC;IAC9F,OAAO,OAAO,CAAC,GAAG,EAAE,eAAe,CAAC,CAAC;AACvC,CAAC;AAED,KAAK,UAAU,cAAc,CAAC,SAAiB;IAC7C,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC;QAAE,OAAO,IAAI,CAAC;IACxC,IAAI,CAAC;QACH,MAAM,CAAC,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,CAAC;QAChC,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC,OAAO,IAAI,MAAM;YAAE,OAAO,IAAI,CAAC;QAClD,MAAM,GAAG,GAAG,MAAM,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAC,CAAC;QAC9C,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAuB,CAAC;IAC/C,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC;AAED,KAAK,UAAU,UAAU,CAAC,SAAiB,EAAE,QAA4B;IACvE,IAAI,CAAC;QACH,MAAM,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QACrD,MAAM,SAAS,CAAC,SAAS,EAAE,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,EAAE,MAAM,CAAC,CAAC;IAC/D,CAAC;IAAC,MAAM,CAAC;QACP,kDAAkD;IACpD,CAAC;AACH,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,YAAY,CAAC,IAAsB;IACvD,MAAM,SAAS,GAAG,gBAAgB,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IAElD,MAAM,MAAM,GAAG,MAAM,cAAc,CAAC,SAAS,CAAC,CAAC;IAC/C,IAAI,MAAM;QAAE,OAAO,MAAM,CAAC;IAE1B,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,GAAG,WAAW,IAAI,CAAC,UAAU,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC;QACjF,MAAM,GAAG,GAAG,MAAM,KAAK,CAAC,GAAG,EAAE,EAAE,MAAM,EAAE,WAAW,CAAC,OAAO,CAAC,gBAAgB,CAAC,EAAE,CAAC,CAAC;QAChF,IAAI,CAAC,GAAG,CAAC,EAAE;YAAE,MAAM,IAAI,KAAK,CAAC,QAAQ,GAAG,CAAC,MAAM,EAAE,CAAC,CAAC;QACnD,MAAM,QAAQ,GAAG,CAAC,MAAM,GAAG,CAAC,IAAI,EAAE,CAAuB,CAAC;QAC1D,MAAM,UAAU,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;QACtC,OAAO,QAAQ,CAAC;IAClB,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAC,QAAQ,CAAC;IACvB,CAAC;AACH,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
import { describe, expect, it, vi, beforeEach, afterEach } from "vitest";
|
|
2
|
+
import { fetchBucketSalt, getFeedbackEndpoint, getSaltUrl, getSubAxisForRule, runInteractiveFeedback, } from "../interactive.js";
|
|
3
|
+
describe("getSubAxisForRule", () => {
|
|
4
|
+
it("maps known rule IDs to seeded sub-axes", () => {
|
|
5
|
+
expect(getSubAxisForRule("tokens/no-hardcoded-color")).toBe("tokens.color");
|
|
6
|
+
expect(getSubAxisForRule("tokens/no-hardcoded-spacing")).toBe("tokens.spacing");
|
|
7
|
+
expect(getSubAxisForRule("a11y/essentials")).toBe("a11y.essentials");
|
|
8
|
+
expect(getSubAxisForRule("components/no-native-shadows")).toBe("components.native-shadows");
|
|
9
|
+
});
|
|
10
|
+
it("returns 'unmapped' for unknown rule IDs", () => {
|
|
11
|
+
expect(getSubAxisForRule("definitely/not-a-rule")).toBe("unmapped");
|
|
12
|
+
});
|
|
13
|
+
});
|
|
14
|
+
describe("endpoint / salt URL env overrides", () => {
|
|
15
|
+
const origEndpoint = process.env["LYSE_FEEDBACK_ENDPOINT"];
|
|
16
|
+
const origSalt = process.env["LYSE_FEEDBACK_SALT_URL"];
|
|
17
|
+
afterEach(() => {
|
|
18
|
+
if (origEndpoint === undefined)
|
|
19
|
+
delete process.env["LYSE_FEEDBACK_ENDPOINT"];
|
|
20
|
+
else
|
|
21
|
+
process.env["LYSE_FEEDBACK_ENDPOINT"] = origEndpoint;
|
|
22
|
+
if (origSalt === undefined)
|
|
23
|
+
delete process.env["LYSE_FEEDBACK_SALT_URL"];
|
|
24
|
+
else
|
|
25
|
+
process.env["LYSE_FEEDBACK_SALT_URL"] = origSalt;
|
|
26
|
+
});
|
|
27
|
+
it("returns production endpoint by default", () => {
|
|
28
|
+
delete process.env["LYSE_FEEDBACK_ENDPOINT"];
|
|
29
|
+
expect(getFeedbackEndpoint()).toBe("https://api.getlyse.com/v1/feedback");
|
|
30
|
+
});
|
|
31
|
+
it("honors LYSE_FEEDBACK_ENDPOINT override", () => {
|
|
32
|
+
process.env["LYSE_FEEDBACK_ENDPOINT"] = "http://staging.test/v1/feedback";
|
|
33
|
+
expect(getFeedbackEndpoint()).toBe("http://staging.test/v1/feedback");
|
|
34
|
+
});
|
|
35
|
+
it("returns production salt URL by default", () => {
|
|
36
|
+
delete process.env["LYSE_FEEDBACK_SALT_URL"];
|
|
37
|
+
expect(getSaltUrl()).toBe("https://api.getlyse.com/v1/bucket-salt");
|
|
38
|
+
});
|
|
39
|
+
it("honors LYSE_FEEDBACK_SALT_URL override", () => {
|
|
40
|
+
process.env["LYSE_FEEDBACK_SALT_URL"] = "http://staging.test/v1/bucket-salt";
|
|
41
|
+
expect(getSaltUrl()).toBe("http://staging.test/v1/bucket-salt");
|
|
42
|
+
});
|
|
43
|
+
});
|
|
44
|
+
describe("fetchBucketSalt", () => {
|
|
45
|
+
let fetchMock;
|
|
46
|
+
beforeEach(() => {
|
|
47
|
+
fetchMock = vi.fn();
|
|
48
|
+
vi.stubGlobal("fetch", fetchMock);
|
|
49
|
+
});
|
|
50
|
+
afterEach(() => {
|
|
51
|
+
vi.unstubAllGlobals();
|
|
52
|
+
});
|
|
53
|
+
it("returns raw string body when not JSON", async () => {
|
|
54
|
+
fetchMock.mockResolvedValueOnce(new Response("salt-2026-05-22"));
|
|
55
|
+
const s = await fetchBucketSalt("http://test/salt");
|
|
56
|
+
expect(s).toBe("salt-2026-05-22");
|
|
57
|
+
});
|
|
58
|
+
it("extracts .salt from JSON body", async () => {
|
|
59
|
+
fetchMock.mockResolvedValueOnce(new Response(JSON.stringify({ salt: "abc-123" })));
|
|
60
|
+
const s = await fetchBucketSalt("http://test/salt");
|
|
61
|
+
expect(s).toBe("abc-123");
|
|
62
|
+
});
|
|
63
|
+
it("returns null on non-2xx", async () => {
|
|
64
|
+
fetchMock.mockResolvedValueOnce(new Response(null, { status: 500 }));
|
|
65
|
+
const s = await fetchBucketSalt("http://test/salt");
|
|
66
|
+
expect(s).toBeNull();
|
|
67
|
+
});
|
|
68
|
+
it("returns null on network error", async () => {
|
|
69
|
+
fetchMock.mockRejectedValueOnce(new Error("ENETDOWN"));
|
|
70
|
+
const s = await fetchBucketSalt("http://test/salt");
|
|
71
|
+
expect(s).toBeNull();
|
|
72
|
+
});
|
|
73
|
+
it("returns null on empty body", async () => {
|
|
74
|
+
fetchMock.mockResolvedValueOnce(new Response(""));
|
|
75
|
+
const s = await fetchBucketSalt("http://test/salt");
|
|
76
|
+
expect(s).toBeNull();
|
|
77
|
+
});
|
|
78
|
+
});
|
|
79
|
+
describe("runInteractiveFeedback (empty short-circuit)", () => {
|
|
80
|
+
it("returns immediately when there are no findings", async () => {
|
|
81
|
+
const r = await runInteractiveFeedback({ findings: [], repoRoot: process.cwd() });
|
|
82
|
+
expect(r).toEqual({ sent: 0, skipped: 0, aborted: false });
|
|
83
|
+
});
|
|
84
|
+
});
|
|
85
|
+
//# sourceMappingURL=interactive.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"interactive.test.js","sourceRoot":"","sources":["../../../../src/reliability/feedback/__tests__/interactive.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import { describe, expect, it } from "vitest";
|
|
2
|
+
import { computeRepoBucket } from "../repo-bucket.js";
|
|
3
|
+
describe("repo_bucket HMAC", () => {
|
|
4
|
+
it("normalizes URL (case, trailing .git)", () => {
|
|
5
|
+
const a = computeRepoBucket("https://github.com/FOO/bar.git", "salt-2026-05-30");
|
|
6
|
+
const b = computeRepoBucket("https://GITHUB.COM/foo/bar", "salt-2026-05-30");
|
|
7
|
+
expect(a).toBe(b);
|
|
8
|
+
});
|
|
9
|
+
it("changes when salt rotates", () => {
|
|
10
|
+
const a = computeRepoBucket("https://github.com/foo/bar.git", "salt-A");
|
|
11
|
+
const b = computeRepoBucket("https://github.com/foo/bar.git", "salt-B");
|
|
12
|
+
expect(a).not.toBe(b);
|
|
13
|
+
});
|
|
14
|
+
it("returns 16-hex prefix", () => {
|
|
15
|
+
expect(computeRepoBucket("https://github.com/foo/bar.git", "s")).toMatch(/^[0-9a-f]{16}$/);
|
|
16
|
+
});
|
|
17
|
+
});
|
|
18
|
+
//# sourceMappingURL=repo-bucket.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"repo-bucket.test.js","sourceRoot":"","sources":["../../../../src/reliability/feedback/__tests__/repo-bucket.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAC;AAC9C,OAAO,EAAE,iBAAiB,EAAE,MAAM,mBAAmB,CAAC;AAEtD,QAAQ,CAAC,kBAAkB,EAAE,GAAG,EAAE;IAChC,EAAE,CAAC,sCAAsC,EAAE,GAAG,EAAE;QAC9C,MAAM,CAAC,GAAG,iBAAiB,CAAC,gCAAgC,EAAE,iBAAiB,CAAC,CAAC;QACjF,MAAM,CAAC,GAAG,iBAAiB,CAAC,4BAA4B,EAAE,iBAAiB,CAAC,CAAC;QAC7E,MAAM,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACpB,CAAC,CAAC,CAAC;IACH,EAAE,CAAC,2BAA2B,EAAE,GAAG,EAAE;QACnC,MAAM,CAAC,GAAG,iBAAiB,CAAC,gCAAgC,EAAE,QAAQ,CAAC,CAAC;QACxE,MAAM,CAAC,GAAG,iBAAiB,CAAC,gCAAgC,EAAE,QAAQ,CAAC,CAAC;QACxE,MAAM,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACxB,CAAC,CAAC,CAAC;IACH,EAAE,CAAC,uBAAuB,EAAE,GAAG,EAAE;QAC/B,MAAM,CAAC,iBAAiB,CAAC,gCAAgC,EAAE,GAAG,CAAC,CAAC,CAAC,OAAO,CAAC,gBAAgB,CAAC,CAAC;IAC7F,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
import { describe, expect, it, vi, beforeEach, afterEach } from "vitest";
|
|
2
|
+
import { sendFeedback } from "../sender.js";
|
|
3
|
+
import { __setCacheForTest, resetConsentCache } from "../../../telemetry/consent.js";
|
|
4
|
+
function enableConsent() {
|
|
5
|
+
__setCacheForTest({ accepted: true, initialised: true });
|
|
6
|
+
}
|
|
7
|
+
describe("sendFeedback", () => {
|
|
8
|
+
let fetchMock;
|
|
9
|
+
beforeEach(() => {
|
|
10
|
+
__setCacheForTest({ accepted: false, initialised: true });
|
|
11
|
+
fetchMock = vi.fn(async () => new Response(null, { status: 204 }));
|
|
12
|
+
vi.stubGlobal("fetch", fetchMock);
|
|
13
|
+
});
|
|
14
|
+
afterEach(() => {
|
|
15
|
+
resetConsentCache();
|
|
16
|
+
vi.unstubAllGlobals();
|
|
17
|
+
});
|
|
18
|
+
it("does NOT fetch when consent is absent", async () => {
|
|
19
|
+
const r = await sendFeedback({
|
|
20
|
+
endpoint: "http://example.test/v1/feedback",
|
|
21
|
+
finding: { ruleId: "x", subAxisId: "tokens.color" },
|
|
22
|
+
verdict: "valid",
|
|
23
|
+
remoteUrl: "https://github.com/foo/bar.git",
|
|
24
|
+
salt: "s",
|
|
25
|
+
});
|
|
26
|
+
expect(fetchMock).not.toHaveBeenCalled();
|
|
27
|
+
expect(r.ok).toBe(false);
|
|
28
|
+
});
|
|
29
|
+
it("does NOT fetch when consent.accepted=false", async () => {
|
|
30
|
+
__setCacheForTest({ accepted: false, initialised: true });
|
|
31
|
+
const r = await sendFeedback({
|
|
32
|
+
endpoint: "http://example.test/v1/feedback",
|
|
33
|
+
finding: { ruleId: "x", subAxisId: "tokens.color" },
|
|
34
|
+
verdict: "valid",
|
|
35
|
+
remoteUrl: "https://github.com/foo/bar.git",
|
|
36
|
+
salt: "s",
|
|
37
|
+
});
|
|
38
|
+
expect(fetchMock).not.toHaveBeenCalled();
|
|
39
|
+
expect(r.ok).toBe(false);
|
|
40
|
+
});
|
|
41
|
+
it("posts JSON payload with HMACed repoBucket when consent is accepted", async () => {
|
|
42
|
+
enableConsent();
|
|
43
|
+
await sendFeedback({
|
|
44
|
+
endpoint: "http://example.test/v1/feedback",
|
|
45
|
+
finding: { ruleId: "tokens/no-hardcoded-color", subAxisId: "tokens.color" },
|
|
46
|
+
verdict: "valid",
|
|
47
|
+
remoteUrl: "https://github.com/foo/bar.git",
|
|
48
|
+
salt: "s-2026-05-22",
|
|
49
|
+
});
|
|
50
|
+
expect(fetchMock).toHaveBeenCalledTimes(1);
|
|
51
|
+
const [url, init] = fetchMock.mock.calls[0];
|
|
52
|
+
expect(url).toBe("http://example.test/v1/feedback");
|
|
53
|
+
expect(init.method).toBe("POST");
|
|
54
|
+
const body = JSON.parse(init.body);
|
|
55
|
+
expect(body.ruleId).toBe("tokens/no-hardcoded-color");
|
|
56
|
+
expect(body.subAxisId).toBe("tokens.color");
|
|
57
|
+
expect(body.verdict).toBe("valid");
|
|
58
|
+
expect(body.signed).toBe(false);
|
|
59
|
+
expect(body.repoBucket).toMatch(/^[0-9a-f]{16}$/);
|
|
60
|
+
});
|
|
61
|
+
it("payload contains ONLY the 5 allowed fields (no PII leak)", async () => {
|
|
62
|
+
enableConsent();
|
|
63
|
+
await sendFeedback({
|
|
64
|
+
endpoint: "http://example.test/v1/feedback",
|
|
65
|
+
finding: { ruleId: "r", subAxisId: "a.b" },
|
|
66
|
+
verdict: "invalid",
|
|
67
|
+
remoteUrl: "https://github.com/foo/secret-repo.git",
|
|
68
|
+
salt: "s",
|
|
69
|
+
});
|
|
70
|
+
const [, init] = fetchMock.mock.calls[0];
|
|
71
|
+
const body = JSON.parse(init.body);
|
|
72
|
+
expect(Object.keys(body).sort()).toEqual(["repoBucket", "ruleId", "signed", "subAxisId", "verdict"]);
|
|
73
|
+
expect(JSON.stringify(body)).not.toContain("secret-repo");
|
|
74
|
+
expect(JSON.stringify(body)).not.toContain("github.com");
|
|
75
|
+
});
|
|
76
|
+
it("returns ok=false on network failure (silent)", async () => {
|
|
77
|
+
enableConsent();
|
|
78
|
+
fetchMock.mockRejectedValueOnce(new Error("connection refused"));
|
|
79
|
+
const r = await sendFeedback({
|
|
80
|
+
endpoint: "http://invalid/v1/feedback",
|
|
81
|
+
finding: { ruleId: "x", subAxisId: "tokens.color" },
|
|
82
|
+
verdict: "valid",
|
|
83
|
+
remoteUrl: "https://github.com/foo/bar.git",
|
|
84
|
+
salt: "s",
|
|
85
|
+
});
|
|
86
|
+
expect(r.ok).toBe(false);
|
|
87
|
+
});
|
|
88
|
+
it("returns ok=false when server returns non-2xx", async () => {
|
|
89
|
+
enableConsent();
|
|
90
|
+
fetchMock.mockResolvedValueOnce(new Response(null, { status: 500 }));
|
|
91
|
+
const r = await sendFeedback({
|
|
92
|
+
endpoint: "http://example.test/v1/feedback",
|
|
93
|
+
finding: { ruleId: "x", subAxisId: "tokens.color" },
|
|
94
|
+
verdict: "valid",
|
|
95
|
+
remoteUrl: "https://github.com/foo/bar.git",
|
|
96
|
+
salt: "s",
|
|
97
|
+
});
|
|
98
|
+
expect(r.ok).toBe(false);
|
|
99
|
+
});
|
|
100
|
+
});
|
|
101
|
+
//# sourceMappingURL=sender.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sender.test.js","sourceRoot":"","sources":["../../../../src/reliability/feedback/__tests__/sender.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import type { Finding } from "../../types.js";
|
|
2
|
+
export interface InteractiveFeedbackResult {
|
|
3
|
+
sent: number;
|
|
4
|
+
skipped: number;
|
|
5
|
+
aborted: boolean;
|
|
6
|
+
}
|
|
7
|
+
export declare function getSubAxisForRule(ruleId: string): string;
|
|
8
|
+
export declare function getFeedbackEndpoint(): string;
|
|
9
|
+
export declare function getSaltUrl(): string;
|
|
10
|
+
export declare function fetchBucketSalt(url: string, timeoutMs?: number): Promise<string | null>;
|
|
11
|
+
export declare function getRemoteOriginUrl(repoRoot: string): string | null;
|
|
12
|
+
export interface RunInteractiveFeedbackOpts {
|
|
13
|
+
findings: Finding[];
|
|
14
|
+
repoRoot: string;
|
|
15
|
+
endpoint?: string;
|
|
16
|
+
saltUrl?: string;
|
|
17
|
+
saltOverride?: string;
|
|
18
|
+
remoteUrlOverride?: string;
|
|
19
|
+
}
|
|
20
|
+
export declare function runInteractiveFeedback(opts: RunInteractiveFeedbackOpts): Promise<InteractiveFeedbackResult>;
|
|
@@ -0,0 +1,150 @@
|
|
|
1
|
+
import { execSync } from "node:child_process";
|
|
2
|
+
import prompts from "prompts";
|
|
3
|
+
import { SUB_AXES } from "../catalogue/sub-axes.js";
|
|
4
|
+
import { sendFeedback } from "./sender.js";
|
|
5
|
+
import { telemetryEnabled } from "../../telemetry/index.js";
|
|
6
|
+
const DEFAULT_FEEDBACK_ENDPOINT = "https://api.getlyse.com/v1/feedback";
|
|
7
|
+
const DEFAULT_SALT_URL = "https://api.getlyse.com/v1/bucket-salt";
|
|
8
|
+
const SALT_FETCH_TIMEOUT_MS = 3000;
|
|
9
|
+
// Build a ruleId → subAxisId lookup from the seeded catalogue.
|
|
10
|
+
// Rules without a sub-axis mapping are reported with subAxisId="unmapped"
|
|
11
|
+
// so the worker can still aggregate them under a tracking bucket.
|
|
12
|
+
const RULE_TO_SUB_AXIS = (() => {
|
|
13
|
+
const m = new Map();
|
|
14
|
+
for (const sa of SUB_AXES) {
|
|
15
|
+
for (const r of sa.ruleIds)
|
|
16
|
+
m.set(r, sa.id);
|
|
17
|
+
}
|
|
18
|
+
return m;
|
|
19
|
+
})();
|
|
20
|
+
export function getSubAxisForRule(ruleId) {
|
|
21
|
+
return RULE_TO_SUB_AXIS.get(ruleId) ?? "unmapped";
|
|
22
|
+
}
|
|
23
|
+
export function getFeedbackEndpoint() {
|
|
24
|
+
const override = process.env["LYSE_FEEDBACK_ENDPOINT"];
|
|
25
|
+
return override && override.length > 0 ? override : DEFAULT_FEEDBACK_ENDPOINT;
|
|
26
|
+
}
|
|
27
|
+
export function getSaltUrl() {
|
|
28
|
+
const override = process.env["LYSE_FEEDBACK_SALT_URL"];
|
|
29
|
+
return override && override.length > 0 ? override : DEFAULT_SALT_URL;
|
|
30
|
+
}
|
|
31
|
+
export async function fetchBucketSalt(url, timeoutMs = SALT_FETCH_TIMEOUT_MS) {
|
|
32
|
+
try {
|
|
33
|
+
const res = await fetch(url, { signal: AbortSignal.timeout(timeoutMs) });
|
|
34
|
+
if (!res.ok)
|
|
35
|
+
return null;
|
|
36
|
+
const txt = (await res.text()).trim();
|
|
37
|
+
if (txt.length === 0)
|
|
38
|
+
return null;
|
|
39
|
+
try {
|
|
40
|
+
const parsed = JSON.parse(txt);
|
|
41
|
+
if (typeof parsed === "string")
|
|
42
|
+
return parsed;
|
|
43
|
+
if (parsed && typeof parsed.salt === "string")
|
|
44
|
+
return parsed.salt;
|
|
45
|
+
}
|
|
46
|
+
catch {
|
|
47
|
+
// Not JSON — treat as raw salt string.
|
|
48
|
+
}
|
|
49
|
+
return txt;
|
|
50
|
+
}
|
|
51
|
+
catch {
|
|
52
|
+
return null;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
export function getRemoteOriginUrl(repoRoot) {
|
|
56
|
+
try {
|
|
57
|
+
const out = execSync("git config --get remote.origin.url", {
|
|
58
|
+
cwd: repoRoot,
|
|
59
|
+
encoding: "utf8",
|
|
60
|
+
stdio: ["ignore", "pipe", "ignore"],
|
|
61
|
+
timeout: 5000,
|
|
62
|
+
}).trim();
|
|
63
|
+
return out.length > 0 ? out : null;
|
|
64
|
+
}
|
|
65
|
+
catch {
|
|
66
|
+
return null;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
// Runs an interactive y/n/?/s/q prompt for each finding. Sends a privacy-clean
|
|
70
|
+
// telemetry event per verdict when consent has been accepted; otherwise the
|
|
71
|
+
// prompts run but no network requests are made (still useful for local triage).
|
|
72
|
+
export async function runInteractiveFeedback(opts) {
|
|
73
|
+
const findings = opts.findings;
|
|
74
|
+
if (findings.length === 0)
|
|
75
|
+
return { sent: 0, skipped: 0, aborted: false };
|
|
76
|
+
const endpoint = opts.endpoint ?? getFeedbackEndpoint();
|
|
77
|
+
const telemetryOn = telemetryEnabled();
|
|
78
|
+
let salt = opts.saltOverride ?? null;
|
|
79
|
+
let remoteUrl = opts.remoteUrlOverride ?? null;
|
|
80
|
+
if (telemetryOn) {
|
|
81
|
+
if (!remoteUrl)
|
|
82
|
+
remoteUrl = getRemoteOriginUrl(opts.repoRoot);
|
|
83
|
+
if (!salt)
|
|
84
|
+
salt = await fetchBucketSalt(opts.saltUrl ?? getSaltUrl());
|
|
85
|
+
if (!salt || !remoteUrl) {
|
|
86
|
+
process.stdout.write("[telemetry: salt unavailable; feedback disabled this session]\n");
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
const canSend = telemetryOn && !!salt && !!remoteUrl;
|
|
90
|
+
process.stdout.write(`\nInteractive review: ${findings.length} finding${findings.length === 1 ? "" : "s"}.\n`);
|
|
91
|
+
process.stdout.write(" y=valid · n=invalid · ?=not sure · s=skip · q=quit\n");
|
|
92
|
+
let sent = 0;
|
|
93
|
+
let skipped = 0;
|
|
94
|
+
let aborted = false;
|
|
95
|
+
for (let i = 0; i < findings.length; i++) {
|
|
96
|
+
const f = findings[i];
|
|
97
|
+
process.stdout.write(`\n[${i + 1}/${findings.length}] ${f.location.file}:${f.location.line} ${f.severity} ${f.ruleId}\n`);
|
|
98
|
+
if (f.message)
|
|
99
|
+
process.stdout.write(` ${f.message}\n`);
|
|
100
|
+
const r = await prompts({
|
|
101
|
+
type: "select",
|
|
102
|
+
name: "v",
|
|
103
|
+
message: "Is this a real issue?",
|
|
104
|
+
choices: [
|
|
105
|
+
{ title: "y · valid (real issue)", value: "valid" },
|
|
106
|
+
{ title: "n · invalid (false positive)", value: "invalid" },
|
|
107
|
+
{ title: "? · not sure (skip)", value: "unsure" },
|
|
108
|
+
{ title: "s · skip silently", value: "skipped" },
|
|
109
|
+
{ title: "q · quit interactive mode", value: "quit" },
|
|
110
|
+
],
|
|
111
|
+
initial: 0,
|
|
112
|
+
});
|
|
113
|
+
const verdict = r.v;
|
|
114
|
+
if (verdict === undefined || verdict === "quit") {
|
|
115
|
+
aborted = true;
|
|
116
|
+
break;
|
|
117
|
+
}
|
|
118
|
+
if (verdict === "unsure" || verdict === "skipped") {
|
|
119
|
+
skipped++;
|
|
120
|
+
continue;
|
|
121
|
+
}
|
|
122
|
+
if (!canSend) {
|
|
123
|
+
skipped++;
|
|
124
|
+
continue;
|
|
125
|
+
}
|
|
126
|
+
const subAxisId = getSubAxisForRule(f.ruleId);
|
|
127
|
+
const res = await sendFeedback({
|
|
128
|
+
endpoint,
|
|
129
|
+
finding: { ruleId: f.ruleId, subAxisId },
|
|
130
|
+
verdict: verdict,
|
|
131
|
+
remoteUrl: remoteUrl,
|
|
132
|
+
salt: salt,
|
|
133
|
+
});
|
|
134
|
+
if (res.ok)
|
|
135
|
+
sent++;
|
|
136
|
+
else
|
|
137
|
+
skipped++;
|
|
138
|
+
}
|
|
139
|
+
if (canSend) {
|
|
140
|
+
process.stdout.write(`\nFeedback session: ${sent} sent, ${skipped} skipped.\n`);
|
|
141
|
+
}
|
|
142
|
+
else if (telemetryOn) {
|
|
143
|
+
process.stdout.write(`\nFeedback session: ${skipped} reviewed locally (telemetry disabled).\n`);
|
|
144
|
+
}
|
|
145
|
+
else {
|
|
146
|
+
process.stdout.write(`\nFeedback session: ${skipped} reviewed locally (run \`lyse telemetry on\` to share anonymized verdicts).\n`);
|
|
147
|
+
}
|
|
148
|
+
return { sent, skipped, aborted };
|
|
149
|
+
}
|
|
150
|
+
//# sourceMappingURL=interactive.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"interactive.js","sourceRoot":"","sources":["../../../src/reliability/feedback/interactive.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { createHmac } from "node:crypto";
|
|
2
|
+
export function normalizeRemoteUrl(url) {
|
|
3
|
+
return url.toLowerCase().replace(/\.git$/, "").replace(/\/+$/, "");
|
|
4
|
+
}
|
|
5
|
+
export function computeRepoBucket(remoteUrl, salt) {
|
|
6
|
+
const normalized = normalizeRemoteUrl(remoteUrl);
|
|
7
|
+
return createHmac("sha256", salt).update(normalized).digest("hex").slice(0, 16);
|
|
8
|
+
}
|
|
9
|
+
//# sourceMappingURL=repo-bucket.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"repo-bucket.js","sourceRoot":"","sources":["../../../src/reliability/feedback/repo-bucket.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAEzC,MAAM,UAAU,kBAAkB,CAAC,GAAW;IAC5C,OAAO,GAAG,CAAC,WAAW,EAAE,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;AACrE,CAAC;AAED,MAAM,UAAU,iBAAiB,CAAC,SAAiB,EAAE,IAAY;IAC/D,MAAM,UAAU,GAAG,kBAAkB,CAAC,SAAS,CAAC,CAAC;IACjD,OAAO,UAAU,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;AAClF,CAAC"}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
export type FeedbackVerdict = "valid" | "invalid" | "skipped";
|
|
2
|
+
export interface FeedbackFinding {
|
|
3
|
+
ruleId: string;
|
|
4
|
+
subAxisId: string;
|
|
5
|
+
}
|
|
6
|
+
export interface SendFeedbackArgs {
|
|
7
|
+
endpoint: string;
|
|
8
|
+
finding: FeedbackFinding;
|
|
9
|
+
verdict: FeedbackVerdict;
|
|
10
|
+
remoteUrl: string;
|
|
11
|
+
salt: string;
|
|
12
|
+
timeoutMs?: number;
|
|
13
|
+
}
|
|
14
|
+
export declare function sendFeedback(args: SendFeedbackArgs): Promise<{
|
|
15
|
+
ok: boolean;
|
|
16
|
+
}>;
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import { computeRepoBucket } from "./repo-bucket.js";
|
|
2
|
+
import { telemetryEnabled } from "../../telemetry/index.js";
|
|
3
|
+
// Opt-in only: returns ok=false (without fetching) unless the user has accepted
|
|
4
|
+
// telemetry consent (~/.lyse/consent.json). Anything else disables the sender.
|
|
5
|
+
export async function sendFeedback(args) {
|
|
6
|
+
if (!telemetryEnabled())
|
|
7
|
+
return { ok: false };
|
|
8
|
+
const repoBucket = computeRepoBucket(args.remoteUrl, args.salt);
|
|
9
|
+
try {
|
|
10
|
+
const res = await fetch(args.endpoint, {
|
|
11
|
+
method: "POST",
|
|
12
|
+
headers: { "content-type": "application/json" },
|
|
13
|
+
body: JSON.stringify({
|
|
14
|
+
ruleId: args.finding.ruleId,
|
|
15
|
+
subAxisId: args.finding.subAxisId,
|
|
16
|
+
repoBucket,
|
|
17
|
+
verdict: args.verdict,
|
|
18
|
+
signed: false,
|
|
19
|
+
}),
|
|
20
|
+
signal: AbortSignal.timeout(args.timeoutMs ?? 2000),
|
|
21
|
+
});
|
|
22
|
+
return { ok: res.ok };
|
|
23
|
+
}
|
|
24
|
+
catch {
|
|
25
|
+
return { ok: false };
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
//# sourceMappingURL=sender.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sender.js","sourceRoot":"","sources":["../../../src/reliability/feedback/sender.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,iBAAiB,EAAE,MAAM,kBAAkB,CAAC;AACrD,OAAO,EAAE,gBAAgB,EAAE,MAAM,0BAA0B,CAAC;AAkB5D,gFAAgF;AAChF,+EAA+E;AAC/E,MAAM,CAAC,KAAK,UAAU,YAAY,CAAC,IAAsB;IACvD,IAAI,CAAC,gBAAgB,EAAE;QAAE,OAAO,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC;IAC9C,MAAM,UAAU,GAAG,iBAAiB,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC;IAChE,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,MAAM,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE;YACrC,MAAM,EAAE,MAAM;YACd,OAAO,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE;YAC/C,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC;gBACnB,MAAM,EAAE,IAAI,CAAC,OAAO,CAAC,MAAM;gBAC3B,SAAS,EAAE,IAAI,CAAC,OAAO,CAAC,SAAS;gBACjC,UAAU;gBACV,OAAO,EAAE,IAAI,CAAC,OAAO;gBACrB,MAAM,EAAE,KAAK;aACd,CAAC;YACF,MAAM,EAAE,WAAW,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC;SACpD,CAAC,CAAC;QACH,OAAO,EAAE,EAAE,EAAE,GAAG,CAAC,EAAE,EAAE,CAAC;IACxB,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC;IACvB,CAAC;AACH,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export type { ScoringVersion, SubAxisStatus, SubAxisId, SubAxisRecord, ConfidenceManifest, Finding, } from "./types.js";
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/reliability/index.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|