uxinspect 0.2.0 → 0.11.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/README.md +332 -22
- package/dist/a11y-filter.d.ts +15 -0
- package/dist/a11y-filter.d.ts.map +1 -0
- package/dist/a11y-filter.js +107 -0
- package/dist/a11y-filter.js.map +1 -0
- package/dist/ab-compare.d.ts +23 -0
- package/dist/ab-compare.d.ts.map +1 -0
- package/dist/ab-compare.js +340 -0
- package/dist/ab-compare.js.map +1 -0
- package/dist/ai-codegen.d.ts +30 -0
- package/dist/ai-codegen.d.ts.map +1 -0
- package/dist/ai-codegen.js +296 -0
- package/dist/ai-codegen.js.map +1 -0
- package/dist/ai-triage.d.ts +26 -0
- package/dist/ai-triage.d.ts.map +1 -0
- package/dist/ai-triage.js +207 -0
- package/dist/ai-triage.js.map +1 -0
- package/dist/amp.d.ts +32 -0
- package/dist/amp.d.ts.map +1 -0
- package/dist/amp.js +179 -0
- package/dist/amp.js.map +1 -0
- package/dist/animation-audit.d.ts +25 -0
- package/dist/animation-audit.d.ts.map +1 -0
- package/dist/animation-audit.js +296 -0
- package/dist/animation-audit.js.map +1 -0
- package/dist/api.d.ts +3 -0
- package/dist/api.d.ts.map +1 -0
- package/dist/api.js +85 -0
- package/dist/api.js.map +1 -0
- package/dist/aria-audit.d.ts +20 -0
- package/dist/aria-audit.d.ts.map +1 -0
- package/dist/aria-audit.js +445 -0
- package/dist/aria-audit.js.map +1 -0
- package/dist/assertions.d.ts +30 -0
- package/dist/assertions.d.ts.map +1 -0
- package/dist/assertions.js +342 -0
- package/dist/assertions.js.map +1 -0
- package/dist/autofix.d.ts +40 -0
- package/dist/autofix.d.ts.map +1 -0
- package/dist/autofix.js +244 -0
- package/dist/autofix.js.map +1 -0
- package/dist/badge.d.ts +27 -0
- package/dist/badge.d.ts.map +1 -0
- package/dist/badge.js +183 -0
- package/dist/badge.js.map +1 -0
- package/dist/baseline-drift.d.ts +43 -0
- package/dist/baseline-drift.d.ts.map +1 -0
- package/dist/baseline-drift.js +208 -0
- package/dist/baseline-drift.js.map +1 -0
- package/dist/bdd.d.ts +31 -0
- package/dist/bdd.d.ts.map +1 -0
- package/dist/bdd.js +316 -0
- package/dist/bdd.js.map +1 -0
- package/dist/bisect.d.ts +32 -0
- package/dist/bisect.d.ts.map +1 -0
- package/dist/bisect.js +253 -0
- package/dist/bisect.js.map +1 -0
- package/dist/budget-diff.d.ts +37 -0
- package/dist/budget-diff.d.ts.map +1 -0
- package/dist/budget-diff.js +273 -0
- package/dist/budget-diff.js.map +1 -0
- package/dist/budget-file.d.ts +15 -0
- package/dist/budget-file.d.ts.map +1 -0
- package/dist/budget-file.js +185 -0
- package/dist/budget-file.js.map +1 -0
- package/dist/bundle-size.d.ts +36 -0
- package/dist/bundle-size.d.ts.map +1 -0
- package/dist/bundle-size.js +347 -0
- package/dist/bundle-size.js.map +1 -0
- package/dist/cache-headers.d.ts +33 -0
- package/dist/cache-headers.d.ts.map +1 -0
- package/dist/cache-headers.js +270 -0
- package/dist/cache-headers.js.map +1 -0
- package/dist/canonical-audit.d.ts +19 -0
- package/dist/canonical-audit.d.ts.map +1 -0
- package/dist/canonical-audit.js +196 -0
- package/dist/canonical-audit.js.map +1 -0
- package/dist/chaos.d.ts +38 -0
- package/dist/chaos.d.ts.map +1 -0
- package/dist/chaos.js +348 -0
- package/dist/chaos.js.map +1 -0
- package/dist/cli.js +201 -23
- package/dist/cli.js.map +1 -1
- package/dist/clickjacking-audit.d.ts +18 -0
- package/dist/clickjacking-audit.d.ts.map +1 -0
- package/dist/clickjacking-audit.js +231 -0
- package/dist/clickjacking-audit.js.map +1 -0
- package/dist/cls-culprit.d.ts +36 -0
- package/dist/cls-culprit.d.ts.map +1 -0
- package/dist/cls-culprit.js +203 -0
- package/dist/cls-culprit.js.map +1 -0
- package/dist/cls-timeline.d.ts +30 -0
- package/dist/cls-timeline.d.ts.map +1 -0
- package/dist/cls-timeline.js +61 -0
- package/dist/cls-timeline.js.map +1 -0
- package/dist/codegen-converter.d.ts +19 -0
- package/dist/codegen-converter.d.ts.map +1 -0
- package/dist/codegen-converter.js +464 -0
- package/dist/codegen-converter.js.map +1 -0
- package/dist/compression.d.ts +14 -0
- package/dist/compression.d.ts.map +1 -0
- package/dist/compression.js +150 -0
- package/dist/compression.js.map +1 -0
- package/dist/console-errors.d.ts +24 -0
- package/dist/console-errors.d.ts.map +1 -0
- package/dist/console-errors.js +96 -0
- package/dist/console-errors.js.map +1 -0
- package/dist/content-quality.d.ts +34 -0
- package/dist/content-quality.d.ts.map +1 -0
- package/dist/content-quality.js +124 -0
- package/dist/content-quality.js.map +1 -0
- package/dist/contract-openapi.d.ts +74 -0
- package/dist/contract-openapi.d.ts.map +1 -0
- package/dist/contract-openapi.js +305 -0
- package/dist/contract-openapi.js.map +1 -0
- package/dist/cookie-banner.d.ts +27 -0
- package/dist/cookie-banner.d.ts.map +1 -0
- package/dist/cookie-banner.js +285 -0
- package/dist/cookie-banner.js.map +1 -0
- package/dist/cookie-flags-audit.d.ts +35 -0
- package/dist/cookie-flags-audit.d.ts.map +1 -0
- package/dist/cookie-flags-audit.js +167 -0
- package/dist/cookie-flags-audit.js.map +1 -0
- package/dist/cpu-throttle.d.ts +34 -0
- package/dist/cpu-throttle.d.ts.map +1 -0
- package/dist/cpu-throttle.js +149 -0
- package/dist/cpu-throttle.js.map +1 -0
- package/dist/crawl.d.ts +29 -0
- package/dist/crawl.d.ts.map +1 -0
- package/dist/crawl.js +153 -0
- package/dist/crawl.js.map +1 -0
- package/dist/critical-css.d.ts +25 -0
- package/dist/critical-css.d.ts.map +1 -0
- package/dist/critical-css.js +353 -0
- package/dist/critical-css.js.map +1 -0
- package/dist/cross-browser.d.ts +44 -0
- package/dist/cross-browser.d.ts.map +1 -0
- package/dist/cross-browser.js +300 -0
- package/dist/cross-browser.js.map +1 -0
- package/dist/csrf-audit.d.ts +33 -0
- package/dist/csrf-audit.d.ts.map +1 -0
- package/dist/csrf-audit.js +276 -0
- package/dist/csrf-audit.js.map +1 -0
- package/dist/css-coverage.d.ts +20 -0
- package/dist/css-coverage.d.ts.map +1 -0
- package/dist/css-coverage.js +91 -0
- package/dist/css-coverage.js.map +1 -0
- package/dist/csv-exporter.d.ts +34 -0
- package/dist/csv-exporter.d.ts.map +1 -0
- package/dist/csv-exporter.js +241 -0
- package/dist/csv-exporter.js.map +1 -0
- package/dist/dark-mode-audit.d.ts +31 -0
- package/dist/dark-mode-audit.d.ts.map +1 -0
- package/dist/dark-mode-audit.js +236 -0
- package/dist/dark-mode-audit.js.map +1 -0
- package/dist/dead-images.d.ts +18 -0
- package/dist/dead-images.d.ts.map +1 -0
- package/dist/dead-images.js +236 -0
- package/dist/dead-images.js.map +1 -0
- package/dist/deadclicks.d.ts +19 -0
- package/dist/deadclicks.d.ts.map +1 -0
- package/dist/deadclicks.js +109 -0
- package/dist/deadclicks.js.map +1 -0
- package/dist/discord-formatter.d.ts +39 -0
- package/dist/discord-formatter.d.ts.map +1 -0
- package/dist/discord-formatter.js +191 -0
- package/dist/discord-formatter.js.map +1 -0
- package/dist/dom-audit.d.ts +23 -0
- package/dist/dom-audit.d.ts.map +1 -0
- package/dist/dom-audit.js +111 -0
- package/dist/dom-audit.js.map +1 -0
- package/dist/driver.d.ts.map +1 -1
- package/dist/driver.js +10 -0
- package/dist/driver.js.map +1 -1
- package/dist/error-page-audit.d.ts +26 -0
- package/dist/error-page-audit.d.ts.map +1 -0
- package/dist/error-page-audit.js +219 -0
- package/dist/error-page-audit.js.map +1 -0
- package/dist/event-listener-audit.d.ts +22 -0
- package/dist/event-listener-audit.d.ts.map +1 -0
- package/dist/event-listener-audit.js +156 -0
- package/dist/event-listener-audit.js.map +1 -0
- package/dist/exposed-paths.d.ts +21 -0
- package/dist/exposed-paths.d.ts.map +1 -0
- package/dist/exposed-paths.js +116 -0
- package/dist/exposed-paths.js.map +1 -0
- package/dist/favicon-audit.d.ts +28 -0
- package/dist/favicon-audit.d.ts.map +1 -0
- package/dist/favicon-audit.js +358 -0
- package/dist/favicon-audit.js.map +1 -0
- package/dist/flaky-detector.d.ts +32 -0
- package/dist/flaky-detector.d.ts.map +1 -0
- package/dist/flaky-detector.js +254 -0
- package/dist/flaky-detector.js.map +1 -0
- package/dist/flaky.d.ts +28 -0
- package/dist/flaky.d.ts.map +1 -0
- package/dist/flaky.js +106 -0
- package/dist/flaky.js.map +1 -0
- package/dist/focus-trap-audit.d.ts +29 -0
- package/dist/focus-trap-audit.d.ts.map +1 -0
- package/dist/focus-trap-audit.js +285 -0
- package/dist/focus-trap-audit.js.map +1 -0
- package/dist/font-loading.d.ts +29 -0
- package/dist/font-loading.d.ts.map +1 -0
- package/dist/font-loading.js +216 -0
- package/dist/font-loading.js.map +1 -0
- package/dist/forms-audit.d.ts +23 -0
- package/dist/forms-audit.d.ts.map +1 -0
- package/dist/forms-audit.js +147 -0
- package/dist/forms-audit.js.map +1 -0
- package/dist/github-annotations.d.ts +17 -0
- package/dist/github-annotations.d.ts.map +1 -0
- package/dist/github-annotations.js +264 -0
- package/dist/github-annotations.js.map +1 -0
- package/dist/graphql.d.ts +60 -0
- package/dist/graphql.d.ts.map +1 -0
- package/dist/graphql.js +188 -0
- package/dist/graphql.js.map +1 -0
- package/dist/har-waterfall.d.ts +37 -0
- package/dist/har-waterfall.d.ts.map +1 -0
- package/dist/har-waterfall.js +376 -0
- package/dist/har-waterfall.js.map +1 -0
- package/dist/heading-hierarchy.d.ts +20 -0
- package/dist/heading-hierarchy.d.ts.map +1 -0
- package/dist/heading-hierarchy.js +112 -0
- package/dist/heading-hierarchy.js.map +1 -0
- package/dist/headless-detect.d.ts +22 -0
- package/dist/headless-detect.d.ts.map +1 -0
- package/dist/headless-detect.js +167 -0
- package/dist/headless-detect.js.map +1 -0
- package/dist/history-timeline.d.ts +13 -0
- package/dist/history-timeline.d.ts.map +1 -0
- package/dist/history-timeline.js +327 -0
- package/dist/history-timeline.js.map +1 -0
- package/dist/hreflang-audit.d.ts +26 -0
- package/dist/hreflang-audit.d.ts.map +1 -0
- package/dist/hreflang-audit.js +273 -0
- package/dist/hreflang-audit.js.map +1 -0
- package/dist/hydration-audit.d.ts +21 -0
- package/dist/hydration-audit.d.ts.map +1 -0
- package/dist/hydration-audit.js +277 -0
- package/dist/hydration-audit.js.map +1 -0
- package/dist/image-audit.d.ts +41 -0
- package/dist/image-audit.d.ts.map +1 -0
- package/dist/image-audit.js +229 -0
- package/dist/image-audit.js.map +1 -0
- package/dist/index.d.ts +119 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +708 -2
- package/dist/index.js.map +1 -1
- package/dist/init-wizard.d.ts +33 -0
- package/dist/init-wizard.d.ts.map +1 -0
- package/dist/init-wizard.js +289 -0
- package/dist/init-wizard.js.map +1 -0
- package/dist/inp-audit.d.ts +26 -0
- package/dist/inp-audit.d.ts.map +1 -0
- package/dist/inp-audit.js +202 -0
- package/dist/inp-audit.js.map +1 -0
- package/dist/js-coverage.d.ts +20 -0
- package/dist/js-coverage.d.ts.map +1 -0
- package/dist/js-coverage.js +81 -0
- package/dist/js-coverage.js.map +1 -0
- package/dist/json-schema.d.ts +27 -0
- package/dist/json-schema.d.ts.map +1 -0
- package/dist/json-schema.js +284 -0
- package/dist/json-schema.js.map +1 -0
- package/dist/keyboard.d.ts +21 -0
- package/dist/keyboard.d.ts.map +1 -0
- package/dist/keyboard.js +119 -0
- package/dist/keyboard.js.map +1 -0
- package/dist/lang-audit.d.ts +24 -0
- package/dist/lang-audit.d.ts.map +1 -0
- package/dist/lang-audit.js +141 -0
- package/dist/lang-audit.js.map +1 -0
- package/dist/lcp-element.d.ts +22 -0
- package/dist/lcp-element.d.ts.map +1 -0
- package/dist/lcp-element.js +240 -0
- package/dist/lcp-element.js.map +1 -0
- package/dist/longtasks.d.ts +38 -0
- package/dist/longtasks.d.ts.map +1 -0
- package/dist/longtasks.js +97 -0
- package/dist/longtasks.js.map +1 -0
- package/dist/mailbox.d.ts +35 -0
- package/dist/mailbox.d.ts.map +1 -0
- package/dist/mailbox.js +207 -0
- package/dist/mailbox.js.map +1 -0
- package/dist/media-audit.d.ts +20 -0
- package/dist/media-audit.d.ts.map +1 -0
- package/dist/media-audit.js +182 -0
- package/dist/media-audit.js.map +1 -0
- package/dist/metrics-exporter.d.ts +23 -0
- package/dist/metrics-exporter.d.ts.map +1 -0
- package/dist/metrics-exporter.js +297 -0
- package/dist/metrics-exporter.js.map +1 -0
- package/dist/mixed-content.d.ts +19 -0
- package/dist/mixed-content.d.ts.map +1 -0
- package/dist/mixed-content.js +86 -0
- package/dist/mixed-content.js.map +1 -0
- package/dist/motion-prefs.d.ts +21 -0
- package/dist/motion-prefs.d.ts.map +1 -0
- package/dist/motion-prefs.js +170 -0
- package/dist/motion-prefs.js.map +1 -0
- package/dist/open-graph.d.ts +40 -0
- package/dist/open-graph.d.ts.map +1 -0
- package/dist/open-graph.js +200 -0
- package/dist/open-graph.js.map +1 -0
- package/dist/orphan-assets.d.ts +17 -0
- package/dist/orphan-assets.d.ts.map +1 -0
- package/dist/orphan-assets.js +174 -0
- package/dist/orphan-assets.js.map +1 -0
- package/dist/page-object.d.ts +18 -0
- package/dist/page-object.d.ts.map +1 -0
- package/dist/page-object.js +346 -0
- package/dist/page-object.js.map +1 -0
- package/dist/pagination-audit.d.ts +24 -0
- package/dist/pagination-audit.d.ts.map +1 -0
- package/dist/pagination-audit.js +285 -0
- package/dist/pagination-audit.js.map +1 -0
- package/dist/passive-security.d.ts +19 -0
- package/dist/passive-security.d.ts.map +1 -0
- package/dist/passive-security.js +149 -0
- package/dist/passive-security.js.map +1 -0
- package/dist/pr-comment.d.ts +13 -0
- package/dist/pr-comment.d.ts.map +1 -0
- package/dist/pr-comment.js +316 -0
- package/dist/pr-comment.js.map +1 -0
- package/dist/precommit.d.ts +24 -0
- package/dist/precommit.d.ts.map +1 -0
- package/dist/precommit.js +239 -0
- package/dist/precommit.js.map +1 -0
- package/dist/prerender-audit.d.ts +22 -0
- package/dist/prerender-audit.d.ts.map +1 -0
- package/dist/prerender-audit.js +158 -0
- package/dist/prerender-audit.js.map +1 -0
- package/dist/print-audit.d.ts +21 -0
- package/dist/print-audit.d.ts.map +1 -0
- package/dist/print-audit.js +281 -0
- package/dist/print-audit.js.map +1 -0
- package/dist/protocol-audit.d.ts +17 -0
- package/dist/protocol-audit.d.ts.map +1 -0
- package/dist/protocol-audit.js +128 -0
- package/dist/protocol-audit.js.map +1 -0
- package/dist/reading-level.d.ts +37 -0
- package/dist/reading-level.d.ts.map +1 -0
- package/dist/reading-level.js +220 -0
- package/dist/reading-level.js.map +1 -0
- package/dist/redirects.d.ts +24 -0
- package/dist/redirects.d.ts.map +1 -0
- package/dist/redirects.js +119 -0
- package/dist/redirects.js.map +1 -0
- package/dist/report.d.ts +1 -1
- package/dist/report.d.ts.map +1 -1
- package/dist/report.js +736 -1
- package/dist/report.js.map +1 -1
- package/dist/reporter-plugin.d.ts +32 -0
- package/dist/reporter-plugin.d.ts.map +1 -0
- package/dist/reporter-plugin.js +120 -0
- package/dist/reporter-plugin.js.map +1 -0
- package/dist/resource-hints.d.ts +23 -0
- package/dist/resource-hints.d.ts.map +1 -0
- package/dist/resource-hints.js +225 -0
- package/dist/resource-hints.js.map +1 -0
- package/dist/retire.d.ts +22 -0
- package/dist/retire.d.ts.map +1 -0
- package/dist/retire.js +140 -0
- package/dist/retire.js.map +1 -0
- package/dist/retry.d.ts +20 -0
- package/dist/retry.d.ts.map +1 -0
- package/dist/retry.js +120 -0
- package/dist/retry.js.map +1 -0
- package/dist/robots-audit.d.ts +24 -0
- package/dist/robots-audit.d.ts.map +1 -0
- package/dist/robots-audit.js +206 -0
- package/dist/robots-audit.js.map +1 -0
- package/dist/rum.d.ts +35 -0
- package/dist/rum.d.ts.map +1 -0
- package/dist/rum.js +219 -0
- package/dist/rum.js.map +1 -0
- package/dist/schedule.d.ts +30 -0
- package/dist/schedule.d.ts.map +1 -0
- package/dist/schedule.js +238 -0
- package/dist/schedule.js.map +1 -0
- package/dist/secret-scan.d.ts +24 -0
- package/dist/secret-scan.d.ts.map +1 -0
- package/dist/secret-scan.js +202 -0
- package/dist/secret-scan.js.map +1 -0
- package/dist/service-worker.d.ts +26 -0
- package/dist/service-worker.d.ts.map +1 -0
- package/dist/service-worker.js +179 -0
- package/dist/service-worker.js.map +1 -0
- package/dist/shard.d.ts +14 -0
- package/dist/shard.d.ts.map +1 -0
- package/dist/shard.js +72 -0
- package/dist/shard.js.map +1 -0
- package/dist/sitemap-flows.d.ts +13 -0
- package/dist/sitemap-flows.d.ts.map +1 -0
- package/dist/sitemap-flows.js +157 -0
- package/dist/sitemap-flows.js.map +1 -0
- package/dist/sitemap.d.ts +27 -0
- package/dist/sitemap.d.ts.map +1 -0
- package/dist/sitemap.js +137 -0
- package/dist/sitemap.js.map +1 -0
- package/dist/slack-formatter.d.ts +35 -0
- package/dist/slack-formatter.d.ts.map +1 -0
- package/dist/slack-formatter.js +193 -0
- package/dist/slack-formatter.js.map +1 -0
- package/dist/sourcemap-scan.d.ts +24 -0
- package/dist/sourcemap-scan.d.ts.map +1 -0
- package/dist/sourcemap-scan.js +232 -0
- package/dist/sourcemap-scan.js.map +1 -0
- package/dist/sri-audit.d.ts +23 -0
- package/dist/sri-audit.d.ts.map +1 -0
- package/dist/sri-audit.js +180 -0
- package/dist/sri-audit.js.map +1 -0
- package/dist/storage-audit.d.ts +28 -0
- package/dist/storage-audit.d.ts.map +1 -0
- package/dist/storage-audit.js +263 -0
- package/dist/storage-audit.js.map +1 -0
- package/dist/storybook.d.ts +48 -0
- package/dist/storybook.d.ts.map +1 -0
- package/dist/storybook.js +191 -0
- package/dist/storybook.js.map +1 -0
- package/dist/structured-data.d.ts +25 -0
- package/dist/structured-data.d.ts.map +1 -0
- package/dist/structured-data.js +164 -0
- package/dist/structured-data.js.map +1 -0
- package/dist/svg-audit.d.ts +20 -0
- package/dist/svg-audit.d.ts.map +1 -0
- package/dist/svg-audit.js +213 -0
- package/dist/svg-audit.js.map +1 -0
- package/dist/table-audit.d.ts +18 -0
- package/dist/table-audit.d.ts.map +1 -0
- package/dist/table-audit.js +188 -0
- package/dist/table-audit.js.map +1 -0
- package/dist/teams-formatter.d.ts +66 -0
- package/dist/teams-formatter.d.ts.map +1 -0
- package/dist/teams-formatter.js +194 -0
- package/dist/teams-formatter.js.map +1 -0
- package/dist/third-party.d.ts +35 -0
- package/dist/third-party.d.ts.map +1 -0
- package/dist/third-party.js +175 -0
- package/dist/third-party.js.map +1 -0
- package/dist/tls.d.ts +33 -0
- package/dist/tls.d.ts.map +1 -0
- package/dist/tls.js +122 -0
- package/dist/tls.js.map +1 -0
- package/dist/touchtargets.d.ts +22 -0
- package/dist/touchtargets.d.ts.map +1 -0
- package/dist/touchtargets.js +80 -0
- package/dist/touchtargets.js.map +1 -0
- package/dist/tracker-sniff.d.ts +25 -0
- package/dist/tracker-sniff.d.ts.map +1 -0
- package/dist/tracker-sniff.js +355 -0
- package/dist/tracker-sniff.js.map +1 -0
- package/dist/types.d.ts +265 -1
- package/dist/types.d.ts.map +1 -1
- package/dist/visual-mask.d.ts +33 -0
- package/dist/visual-mask.d.ts.map +1 -0
- package/dist/visual-mask.js +102 -0
- package/dist/visual-mask.js.map +1 -0
- package/dist/visual-ssim.d.ts +26 -0
- package/dist/visual-ssim.d.ts.map +1 -0
- package/dist/visual-ssim.js +153 -0
- package/dist/visual-ssim.js.map +1 -0
- package/dist/watch-mode.d.ts +10 -0
- package/dist/watch-mode.d.ts.map +1 -0
- package/dist/watch-mode.js +156 -0
- package/dist/watch-mode.js.map +1 -0
- package/dist/web-worker-audit.d.ts +27 -0
- package/dist/web-worker-audit.d.ts.map +1 -0
- package/dist/web-worker-audit.js +324 -0
- package/dist/web-worker-audit.js.map +1 -0
- package/dist/webfonts.d.ts +26 -0
- package/dist/webfonts.d.ts.map +1 -0
- package/dist/webfonts.js +244 -0
- package/dist/webfonts.js.map +1 -0
- package/dist/webhook-reporter.d.ts +20 -0
- package/dist/webhook-reporter.d.ts.map +1 -0
- package/dist/webhook-reporter.js +124 -0
- package/dist/webhook-reporter.js.map +1 -0
- package/dist/websocket.d.ts +39 -0
- package/dist/websocket.d.ts.map +1 -0
- package/dist/websocket.js +233 -0
- package/dist/websocket.js.map +1 -0
- package/dist/worker-runtime.d.ts +129 -0
- package/dist/worker-runtime.d.ts.map +1 -0
- package/dist/worker-runtime.js +414 -0
- package/dist/worker-runtime.js.map +1 -0
- package/dist/zindex-audit.d.ts +28 -0
- package/dist/zindex-audit.d.ts.map +1 -0
- package/dist/zindex-audit.js +291 -0
- package/dist/zindex-audit.js.map +1 -0
- package/package.json +10 -2
|
@@ -0,0 +1,300 @@
|
|
|
1
|
+
import { promises as fs } from 'node:fs';
|
|
2
|
+
import path from 'node:path';
|
|
3
|
+
import { PNG } from 'pngjs';
|
|
4
|
+
import pixelmatch from 'pixelmatch';
|
|
5
|
+
const DEFAULT_ENGINES = ['chromium', 'firefox', 'webkit'];
|
|
6
|
+
const DEFAULT_PIXEL_THRESHOLD = 0.02;
|
|
7
|
+
const PIXELMATCH_THRESHOLD = 0.1;
|
|
8
|
+
const METRIC_LABELS = {
|
|
9
|
+
perfLcp: 'perfLcp',
|
|
10
|
+
perfCls: 'perfCls',
|
|
11
|
+
a11yCriticals: 'a11yCriticals',
|
|
12
|
+
visualDiffs: 'visualDiffs',
|
|
13
|
+
consoleErrorCount: 'consoleErrorCount',
|
|
14
|
+
};
|
|
15
|
+
async function fileExists(p) {
|
|
16
|
+
try {
|
|
17
|
+
await fs.access(p);
|
|
18
|
+
return true;
|
|
19
|
+
}
|
|
20
|
+
catch {
|
|
21
|
+
return false;
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
async function readPng(p) {
|
|
25
|
+
try {
|
|
26
|
+
const bytes = await fs.readFile(p);
|
|
27
|
+
return PNG.sync.read(bytes);
|
|
28
|
+
}
|
|
29
|
+
catch {
|
|
30
|
+
return null;
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
function cloneConfigForEngine(config, engine, engineOutDir) {
|
|
34
|
+
return {
|
|
35
|
+
...config,
|
|
36
|
+
browser: engine,
|
|
37
|
+
output: {
|
|
38
|
+
...(config.output ?? {}),
|
|
39
|
+
dir: path.join(engineOutDir, 'report'),
|
|
40
|
+
baselineDir: path.join(engineOutDir, 'baselines'),
|
|
41
|
+
},
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
function countA11yCriticals(result) {
|
|
45
|
+
if (!result.a11y)
|
|
46
|
+
return 0;
|
|
47
|
+
let n = 0;
|
|
48
|
+
for (const page of result.a11y) {
|
|
49
|
+
for (const v of page.violations) {
|
|
50
|
+
if (v.impact === 'critical')
|
|
51
|
+
n++;
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
return n;
|
|
55
|
+
}
|
|
56
|
+
function countVisualDiffs(result) {
|
|
57
|
+
if (!result.visual)
|
|
58
|
+
return 0;
|
|
59
|
+
let n = 0;
|
|
60
|
+
for (const v of result.visual) {
|
|
61
|
+
if (!v.passed)
|
|
62
|
+
n++;
|
|
63
|
+
}
|
|
64
|
+
return n;
|
|
65
|
+
}
|
|
66
|
+
function countConsoleErrors(result) {
|
|
67
|
+
if (!result.consoleErrors)
|
|
68
|
+
return 0;
|
|
69
|
+
let n = 0;
|
|
70
|
+
for (const c of result.consoleErrors)
|
|
71
|
+
n += c.errorCount;
|
|
72
|
+
return n;
|
|
73
|
+
}
|
|
74
|
+
function avgLcp(result) {
|
|
75
|
+
if (!result.perf || result.perf.length === 0)
|
|
76
|
+
return undefined;
|
|
77
|
+
let sum = 0;
|
|
78
|
+
for (const p of result.perf)
|
|
79
|
+
sum += p.metrics.lcp;
|
|
80
|
+
return Math.round(sum / result.perf.length);
|
|
81
|
+
}
|
|
82
|
+
function avgCls(result) {
|
|
83
|
+
if (!result.perf || result.perf.length === 0)
|
|
84
|
+
return undefined;
|
|
85
|
+
let sum = 0;
|
|
86
|
+
for (const p of result.perf)
|
|
87
|
+
sum += p.metrics.cls;
|
|
88
|
+
return sum / result.perf.length;
|
|
89
|
+
}
|
|
90
|
+
function summarizeResult(engine, result, durationMs) {
|
|
91
|
+
return {
|
|
92
|
+
engine,
|
|
93
|
+
passed: result.passed,
|
|
94
|
+
durationMs,
|
|
95
|
+
perfLcp: avgLcp(result),
|
|
96
|
+
perfCls: avgCls(result),
|
|
97
|
+
a11yCriticals: countA11yCriticals(result),
|
|
98
|
+
visualDiffs: countVisualDiffs(result),
|
|
99
|
+
consoleErrorCount: countConsoleErrors(result),
|
|
100
|
+
};
|
|
101
|
+
}
|
|
102
|
+
function enumerateFlowViewports(config) {
|
|
103
|
+
const flows = config.flows ?? [{ name: 'load', steps: [{ goto: config.url }] }];
|
|
104
|
+
const viewports = config.viewports ?? [{ name: 'desktop', width: 1280, height: 800 }];
|
|
105
|
+
const pairs = [];
|
|
106
|
+
for (const f of flows) {
|
|
107
|
+
for (const v of viewports) {
|
|
108
|
+
pairs.push({ flow: f.name, viewport: v.name });
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
return pairs;
|
|
112
|
+
}
|
|
113
|
+
function screenshotPath(outDir, engine, flow, viewport) {
|
|
114
|
+
return path.join(outDir, engine, 'report', 'current', `${flow}-${viewport}.png`);
|
|
115
|
+
}
|
|
116
|
+
async function diffPair(outDir, engineA, engineB, flow, viewport) {
|
|
117
|
+
const pathA = screenshotPath(outDir, engineA, flow, viewport);
|
|
118
|
+
const pathB = screenshotPath(outDir, engineB, flow, viewport);
|
|
119
|
+
const existsA = await fileExists(pathA);
|
|
120
|
+
const existsB = await fileExists(pathB);
|
|
121
|
+
if (!existsA || !existsB)
|
|
122
|
+
return null;
|
|
123
|
+
const pngA = await readPng(pathA);
|
|
124
|
+
const pngB = await readPng(pathB);
|
|
125
|
+
if (!pngA || !pngB)
|
|
126
|
+
return null;
|
|
127
|
+
const diffDir = path.join(outDir, 'diffs', `${engineA}-vs-${engineB}`);
|
|
128
|
+
await fs.mkdir(diffDir, { recursive: true });
|
|
129
|
+
const diffPath = path.join(diffDir, `${flow}-${viewport}.png`);
|
|
130
|
+
if (pngA.width !== pngB.width || pngA.height !== pngB.height) {
|
|
131
|
+
const w = Math.max(pngA.width, pngB.width);
|
|
132
|
+
const h = Math.max(pngA.height, pngB.height);
|
|
133
|
+
const marker = new PNG({ width: w, height: h });
|
|
134
|
+
await fs.writeFile(diffPath, PNG.sync.write(marker));
|
|
135
|
+
return {
|
|
136
|
+
engineA,
|
|
137
|
+
engineB,
|
|
138
|
+
flow,
|
|
139
|
+
viewport,
|
|
140
|
+
diffRatio: 1,
|
|
141
|
+
diffPixels: w * h,
|
|
142
|
+
diffPath,
|
|
143
|
+
};
|
|
144
|
+
}
|
|
145
|
+
const { width, height } = pngA;
|
|
146
|
+
const diff = new PNG({ width, height });
|
|
147
|
+
const diffPixels = pixelmatch(pngA.data, pngB.data, diff.data, width, height, {
|
|
148
|
+
threshold: PIXELMATCH_THRESHOLD,
|
|
149
|
+
});
|
|
150
|
+
await fs.writeFile(diffPath, PNG.sync.write(diff));
|
|
151
|
+
const diffRatio = width * height > 0 ? diffPixels / (width * height) : 0;
|
|
152
|
+
return { engineA, engineB, flow, viewport, diffRatio, diffPixels, diffPath };
|
|
153
|
+
}
|
|
154
|
+
function numericOutcomeValue(outcome, key) {
|
|
155
|
+
const v = outcome[key];
|
|
156
|
+
return typeof v === 'number' ? v : 0;
|
|
157
|
+
}
|
|
158
|
+
function buildMetricDeltas(outcomes) {
|
|
159
|
+
const deltas = [];
|
|
160
|
+
const keys = Object.keys(METRIC_LABELS);
|
|
161
|
+
for (let i = 0; i < outcomes.length; i++) {
|
|
162
|
+
for (let j = i + 1; j < outcomes.length; j++) {
|
|
163
|
+
const a = outcomes[i];
|
|
164
|
+
const b = outcomes[j];
|
|
165
|
+
if (!a || !b)
|
|
166
|
+
continue;
|
|
167
|
+
for (const key of keys) {
|
|
168
|
+
const aVal = numericOutcomeValue(a, key);
|
|
169
|
+
const bVal = numericOutcomeValue(b, key);
|
|
170
|
+
if (a[key] === undefined && b[key] === undefined)
|
|
171
|
+
continue;
|
|
172
|
+
const delta = Math.abs(aVal - bVal);
|
|
173
|
+
const denom = Math.max(aVal, bVal, 1);
|
|
174
|
+
deltas.push({
|
|
175
|
+
metric: METRIC_LABELS[key],
|
|
176
|
+
engineA: a.engine,
|
|
177
|
+
engineB: b.engine,
|
|
178
|
+
delta,
|
|
179
|
+
ratio: delta / denom,
|
|
180
|
+
});
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
return deltas;
|
|
185
|
+
}
|
|
186
|
+
function buildDivergenceLines(config, perEngineResults, outcomes) {
|
|
187
|
+
const lines = [];
|
|
188
|
+
const outcomeByEngine = new Map();
|
|
189
|
+
for (const o of outcomes)
|
|
190
|
+
outcomeByEngine.set(o.engine, o);
|
|
191
|
+
const statuses = outcomes.map((o) => `${o.engine}:${o.passed ? 'pass' : 'fail'}`);
|
|
192
|
+
const uniqueStatuses = new Set(outcomes.map((o) => o.passed));
|
|
193
|
+
if (uniqueStatuses.size > 1) {
|
|
194
|
+
lines.push(`overall: engines diverge (${statuses.join(', ')})`);
|
|
195
|
+
}
|
|
196
|
+
const pairs = enumerateFlowViewports(config);
|
|
197
|
+
for (const { flow, viewport } of pairs) {
|
|
198
|
+
const perEngine = [];
|
|
199
|
+
for (const [engine, result] of perEngineResults.entries()) {
|
|
200
|
+
if (!result) {
|
|
201
|
+
perEngine.push({ engine, passed: null });
|
|
202
|
+
continue;
|
|
203
|
+
}
|
|
204
|
+
const flowResult = result.flows.find((f) => f.name === flow);
|
|
205
|
+
const visualResult = result.visual?.find((v) => v.viewport === viewport && (v.current.includes(`${flow}-${viewport}`) || v.baseline.includes(`${flow}-${viewport}`)));
|
|
206
|
+
const passed = flowResult === undefined
|
|
207
|
+
? null
|
|
208
|
+
: flowResult.passed && (visualResult ? visualResult.passed : true);
|
|
209
|
+
perEngine.push({ engine, passed });
|
|
210
|
+
}
|
|
211
|
+
const known = perEngine.filter((p) => p.passed !== null);
|
|
212
|
+
if (known.length < 2)
|
|
213
|
+
continue;
|
|
214
|
+
const distinct = new Set(known.map((p) => p.passed));
|
|
215
|
+
if (distinct.size > 1) {
|
|
216
|
+
const passes = known.filter((p) => p.passed === true).map((p) => p.engine);
|
|
217
|
+
const fails = known.filter((p) => p.passed === false).map((p) => p.engine);
|
|
218
|
+
lines.push(`flow ${flow} @ ${viewport}: passed on ${passes.join('+') || 'none'}, failed on ${fails.join('+') || 'none'}`);
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
return lines;
|
|
222
|
+
}
|
|
223
|
+
async function loadInspect() {
|
|
224
|
+
const mod = (await import('./index.js'));
|
|
225
|
+
return mod.inspect;
|
|
226
|
+
}
|
|
227
|
+
export async function runCrossBrowser(config, opts) {
|
|
228
|
+
const engines = opts.engines ?? DEFAULT_ENGINES;
|
|
229
|
+
const threshold = opts.pixelDiffThreshold ?? DEFAULT_PIXEL_THRESHOLD;
|
|
230
|
+
const outDir = path.resolve(opts.outDir);
|
|
231
|
+
await fs.mkdir(outDir, { recursive: true });
|
|
232
|
+
const inspect = await loadInspect();
|
|
233
|
+
const outcomes = [];
|
|
234
|
+
const perEngineResults = new Map();
|
|
235
|
+
for (const engine of engines) {
|
|
236
|
+
const engineOutDir = path.join(outDir, engine);
|
|
237
|
+
await fs.mkdir(engineOutDir, { recursive: true });
|
|
238
|
+
const engineConfig = cloneConfigForEngine(config, engine, engineOutDir);
|
|
239
|
+
const started = Date.now();
|
|
240
|
+
try {
|
|
241
|
+
const result = await inspect(engineConfig);
|
|
242
|
+
const durationMs = Date.now() - started;
|
|
243
|
+
perEngineResults.set(engine, result);
|
|
244
|
+
outcomes.push(summarizeResult(engine, result, durationMs));
|
|
245
|
+
}
|
|
246
|
+
catch (err) {
|
|
247
|
+
const durationMs = Date.now() - started;
|
|
248
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
249
|
+
perEngineResults.set(engine, null);
|
|
250
|
+
outcomes.push({
|
|
251
|
+
engine,
|
|
252
|
+
passed: false,
|
|
253
|
+
durationMs,
|
|
254
|
+
a11yCriticals: 0,
|
|
255
|
+
visualDiffs: 0,
|
|
256
|
+
consoleErrorCount: 0,
|
|
257
|
+
error: message,
|
|
258
|
+
});
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
const flowVpPairs = enumerateFlowViewports(config);
|
|
262
|
+
const pairJobs = [];
|
|
263
|
+
for (let i = 0; i < engines.length; i++) {
|
|
264
|
+
for (let j = i + 1; j < engines.length; j++) {
|
|
265
|
+
const a = engines[i];
|
|
266
|
+
const b = engines[j];
|
|
267
|
+
if (!a || !b)
|
|
268
|
+
continue;
|
|
269
|
+
const resA = perEngineResults.get(a);
|
|
270
|
+
const resB = perEngineResults.get(b);
|
|
271
|
+
if (!resA || !resB)
|
|
272
|
+
continue;
|
|
273
|
+
for (const { flow, viewport } of flowVpPairs) {
|
|
274
|
+
pairJobs.push(diffPair(outDir, a, b, flow, viewport));
|
|
275
|
+
}
|
|
276
|
+
}
|
|
277
|
+
}
|
|
278
|
+
const diffResults = await Promise.all(pairJobs);
|
|
279
|
+
const screenshotDiffs = [];
|
|
280
|
+
for (const d of diffResults) {
|
|
281
|
+
if (d)
|
|
282
|
+
screenshotDiffs.push(d);
|
|
283
|
+
}
|
|
284
|
+
const metricDeltas = buildMetricDeltas(outcomes);
|
|
285
|
+
const divergent = buildDivergenceLines(config, perEngineResults, outcomes);
|
|
286
|
+
const allEnginesPassed = outcomes.every((o) => o.passed);
|
|
287
|
+
const allDiffsUnderThreshold = screenshotDiffs.every((d) => d.diffRatio <= threshold);
|
|
288
|
+
const passed = allEnginesPassed && allDiffsUnderThreshold;
|
|
289
|
+
return {
|
|
290
|
+
url: config.url,
|
|
291
|
+
engines,
|
|
292
|
+
outcomes,
|
|
293
|
+
screenshotDiffs,
|
|
294
|
+
metricDeltas,
|
|
295
|
+
divergent,
|
|
296
|
+
passed,
|
|
297
|
+
outDir,
|
|
298
|
+
};
|
|
299
|
+
}
|
|
300
|
+
//# sourceMappingURL=cross-browser.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cross-browser.js","sourceRoot":"","sources":["../src/cross-browser.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import type { Page, BrowserContext } from 'playwright';
|
|
2
|
+
export interface FormCsrfState {
|
|
3
|
+
selector: string;
|
|
4
|
+
action: string;
|
|
5
|
+
method: string;
|
|
6
|
+
hasCsrfField: boolean;
|
|
7
|
+
csrfFieldName?: string;
|
|
8
|
+
csrfFieldValue?: string;
|
|
9
|
+
metaCsrfPresent: boolean;
|
|
10
|
+
cookieSameSite: 'Strict' | 'Lax' | 'None' | 'missing';
|
|
11
|
+
isIdempotent: boolean;
|
|
12
|
+
isCrossOrigin: boolean;
|
|
13
|
+
passed: boolean;
|
|
14
|
+
}
|
|
15
|
+
export interface CsrfIssue {
|
|
16
|
+
kind: 'form-missing-csrf' | 'csrf-header-missing' | 'samesite-none-no-csrf' | 'weak-csrf-entropy' | 'csrf-in-url';
|
|
17
|
+
selector?: string;
|
|
18
|
+
detail: string;
|
|
19
|
+
}
|
|
20
|
+
export interface CsrfAuditResult {
|
|
21
|
+
page: string;
|
|
22
|
+
forms: FormCsrfState[];
|
|
23
|
+
metaCsrfToken: string | null;
|
|
24
|
+
cookieDefenses: {
|
|
25
|
+
sameSiteStrict: number;
|
|
26
|
+
sameSiteLax: number;
|
|
27
|
+
sameSiteNone: number;
|
|
28
|
+
};
|
|
29
|
+
issues: CsrfIssue[];
|
|
30
|
+
passed: boolean;
|
|
31
|
+
}
|
|
32
|
+
export declare function auditCsrf(page: Page, ctx: BrowserContext): Promise<CsrfAuditResult>;
|
|
33
|
+
//# sourceMappingURL=csrf-audit.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"csrf-audit.d.ts","sourceRoot":"","sources":["../src/csrf-audit.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,IAAI,EAAE,cAAc,EAAU,MAAM,YAAY,CAAC;AAE/D,MAAM,WAAW,aAAa;IAC5B,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,MAAM,CAAC;IACf,MAAM,EAAE,MAAM,CAAC;IACf,YAAY,EAAE,OAAO,CAAC;IACtB,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,eAAe,EAAE,OAAO,CAAC;IACzB,cAAc,EAAE,QAAQ,GAAG,KAAK,GAAG,MAAM,GAAG,SAAS,CAAC;IACtD,YAAY,EAAE,OAAO,CAAC;IACtB,aAAa,EAAE,OAAO,CAAC;IACvB,MAAM,EAAE,OAAO,CAAC;CACjB;AAED,MAAM,WAAW,SAAS;IACxB,IAAI,EACA,mBAAmB,GACnB,qBAAqB,GACrB,uBAAuB,GACvB,mBAAmB,GACnB,aAAa,CAAC;IAClB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,MAAM,EAAE,MAAM,CAAC;CAChB;AAED,MAAM,WAAW,eAAe;IAC9B,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,aAAa,EAAE,CAAC;IACvB,aAAa,EAAE,MAAM,GAAG,IAAI,CAAC;IAC7B,cAAc,EAAE;QAAE,cAAc,EAAE,MAAM,CAAC;QAAC,WAAW,EAAE,MAAM,CAAC;QAAC,YAAY,EAAE,MAAM,CAAA;KAAE,CAAC;IACtF,MAAM,EAAE,SAAS,EAAE,CAAC;IACpB,MAAM,EAAE,OAAO,CAAC;CACjB;AAyPD,wBAAsB,SAAS,CAC7B,IAAI,EAAE,IAAI,EACV,GAAG,EAAE,cAAc,GAClB,OAAO,CAAC,eAAe,CAAC,CA0D1B"}
|
|
@@ -0,0 +1,276 @@
|
|
|
1
|
+
const CSRF_COOKIE_NAMES = new Set(['csrftoken', '_csrf', 'XSRF-TOKEN']);
|
|
2
|
+
function redactValue(raw) {
|
|
3
|
+
if (raw.length <= 8)
|
|
4
|
+
return '***';
|
|
5
|
+
return `${raw.slice(0, 4)}...${raw.slice(-4)}`;
|
|
6
|
+
}
|
|
7
|
+
function shannonEntropy(value) {
|
|
8
|
+
if (value.length === 0)
|
|
9
|
+
return 0;
|
|
10
|
+
const counts = new Map();
|
|
11
|
+
for (const ch of value)
|
|
12
|
+
counts.set(ch, (counts.get(ch) ?? 0) + 1);
|
|
13
|
+
const total = value.length;
|
|
14
|
+
let entropy = 0;
|
|
15
|
+
for (const count of counts.values()) {
|
|
16
|
+
const p = count / total;
|
|
17
|
+
entropy -= p * Math.log2(p);
|
|
18
|
+
}
|
|
19
|
+
return entropy;
|
|
20
|
+
}
|
|
21
|
+
function normalizeSameSite(v) {
|
|
22
|
+
return v === 'Strict' || v === 'Lax' || v === 'None' ? v : undefined;
|
|
23
|
+
}
|
|
24
|
+
function originOf(url) {
|
|
25
|
+
try {
|
|
26
|
+
return new URL(url).origin;
|
|
27
|
+
}
|
|
28
|
+
catch {
|
|
29
|
+
return null;
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
function urlHasCsrfParam(url) {
|
|
33
|
+
try {
|
|
34
|
+
const parsed = new URL(url);
|
|
35
|
+
for (const key of parsed.searchParams.keys()) {
|
|
36
|
+
if (/csrf|_token|authenticity_token|xsrf/i.test(key))
|
|
37
|
+
return true;
|
|
38
|
+
}
|
|
39
|
+
return false;
|
|
40
|
+
}
|
|
41
|
+
catch {
|
|
42
|
+
return false;
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
async function probePage(page) {
|
|
46
|
+
return page.evaluate(() => {
|
|
47
|
+
const namePattern = /csrf|_token|authenticity_token|xsrf/i;
|
|
48
|
+
const headerPattern = /['"]x-csrf-token['"]|['"]x-xsrf-token['"]|['"]csrf-token['"]/i;
|
|
49
|
+
const fetchUsagePattern = /\bfetch\s*\(|XMLHttpRequest|axios/;
|
|
50
|
+
function buildSelector(form, index) {
|
|
51
|
+
if (form.id)
|
|
52
|
+
return `form#${form.id}`;
|
|
53
|
+
const name = form.getAttribute('name');
|
|
54
|
+
if (name)
|
|
55
|
+
return `form[name="${name}"]`;
|
|
56
|
+
const action = form.getAttribute('action');
|
|
57
|
+
if (action)
|
|
58
|
+
return `form[action="${action}"]`;
|
|
59
|
+
return `form:nth-of-type(${index + 1})`;
|
|
60
|
+
}
|
|
61
|
+
function paramHasCsrf(raw) {
|
|
62
|
+
try {
|
|
63
|
+
const parsed = new URL(raw, document.baseURI);
|
|
64
|
+
for (const k of parsed.searchParams.keys()) {
|
|
65
|
+
if (namePattern.test(k))
|
|
66
|
+
return true;
|
|
67
|
+
}
|
|
68
|
+
return false;
|
|
69
|
+
}
|
|
70
|
+
catch {
|
|
71
|
+
return false;
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
const formEls = Array.from(document.querySelectorAll('form'));
|
|
75
|
+
const forms = formEls.map((form, index) => {
|
|
76
|
+
const fe = form;
|
|
77
|
+
const actionRaw = fe.getAttribute('action') ?? '';
|
|
78
|
+
let action = actionRaw;
|
|
79
|
+
try {
|
|
80
|
+
action = new URL(actionRaw || document.URL, document.baseURI).href;
|
|
81
|
+
}
|
|
82
|
+
catch {
|
|
83
|
+
action = actionRaw;
|
|
84
|
+
}
|
|
85
|
+
const method = (fe.getAttribute('method') ?? 'get').toLowerCase();
|
|
86
|
+
const inputs = Array.from(fe.querySelectorAll('input[name]'));
|
|
87
|
+
let csrfFieldName = null;
|
|
88
|
+
let csrfFieldValue = null;
|
|
89
|
+
for (const input of inputs) {
|
|
90
|
+
const n = input.getAttribute('name') ?? '';
|
|
91
|
+
if (namePattern.test(n)) {
|
|
92
|
+
csrfFieldName = n;
|
|
93
|
+
csrfFieldValue = input.value ?? '';
|
|
94
|
+
break;
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
return {
|
|
98
|
+
selector: buildSelector(fe, index),
|
|
99
|
+
action,
|
|
100
|
+
method,
|
|
101
|
+
csrfFieldName,
|
|
102
|
+
csrfFieldValue,
|
|
103
|
+
actionQueryCsrf: paramHasCsrf(actionRaw || document.URL),
|
|
104
|
+
};
|
|
105
|
+
});
|
|
106
|
+
const metaEl = document.querySelector('meta[name="csrf-token"], meta[name="_csrf"], meta[name="x-csrf-token"]');
|
|
107
|
+
const metaCsrfToken = metaEl?.getAttribute('content') ?? null;
|
|
108
|
+
const scriptEls = Array.from(document.querySelectorAll('script:not([src])'));
|
|
109
|
+
let inlineScriptHasCsrfHeader = false;
|
|
110
|
+
let inlineScriptUsesFetch = false;
|
|
111
|
+
for (const script of scriptEls) {
|
|
112
|
+
const text = script.textContent ?? '';
|
|
113
|
+
if (headerPattern.test(text))
|
|
114
|
+
inlineScriptHasCsrfHeader = true;
|
|
115
|
+
if (fetchUsagePattern.test(text))
|
|
116
|
+
inlineScriptUsesFetch = true;
|
|
117
|
+
}
|
|
118
|
+
return {
|
|
119
|
+
forms,
|
|
120
|
+
metaCsrfToken,
|
|
121
|
+
inlineScriptHasCsrfHeader,
|
|
122
|
+
inlineScriptUsesFetch,
|
|
123
|
+
pageUrlHasCsrf: paramHasCsrf(document.URL),
|
|
124
|
+
};
|
|
125
|
+
});
|
|
126
|
+
}
|
|
127
|
+
function countSameSite(cookies) {
|
|
128
|
+
let sameSiteStrict = 0;
|
|
129
|
+
let sameSiteLax = 0;
|
|
130
|
+
let sameSiteNone = 0;
|
|
131
|
+
for (const c of cookies) {
|
|
132
|
+
const ss = normalizeSameSite(c.sameSite);
|
|
133
|
+
if (ss === 'Strict')
|
|
134
|
+
sameSiteStrict += 1;
|
|
135
|
+
else if (ss === 'Lax')
|
|
136
|
+
sameSiteLax += 1;
|
|
137
|
+
else if (ss === 'None')
|
|
138
|
+
sameSiteNone += 1;
|
|
139
|
+
}
|
|
140
|
+
return { sameSiteStrict, sameSiteLax, sameSiteNone };
|
|
141
|
+
}
|
|
142
|
+
function dominantSameSite(cookies) {
|
|
143
|
+
let strict = 0;
|
|
144
|
+
let lax = 0;
|
|
145
|
+
let none = 0;
|
|
146
|
+
let explicit = 0;
|
|
147
|
+
for (const c of cookies) {
|
|
148
|
+
const ss = normalizeSameSite(c.sameSite);
|
|
149
|
+
if (ss === undefined)
|
|
150
|
+
continue;
|
|
151
|
+
explicit += 1;
|
|
152
|
+
if (ss === 'Strict')
|
|
153
|
+
strict += 1;
|
|
154
|
+
else if (ss === 'Lax')
|
|
155
|
+
lax += 1;
|
|
156
|
+
else if (ss === 'None')
|
|
157
|
+
none += 1;
|
|
158
|
+
}
|
|
159
|
+
if (explicit === 0)
|
|
160
|
+
return 'missing';
|
|
161
|
+
if (strict >= lax && strict >= none)
|
|
162
|
+
return 'Strict';
|
|
163
|
+
if (lax >= none)
|
|
164
|
+
return 'Lax';
|
|
165
|
+
return 'None';
|
|
166
|
+
}
|
|
167
|
+
function hasCsrfCookie(cookies) {
|
|
168
|
+
for (const c of cookies)
|
|
169
|
+
if (CSRF_COOKIE_NAMES.has(c.name))
|
|
170
|
+
return true;
|
|
171
|
+
return false;
|
|
172
|
+
}
|
|
173
|
+
function buildFormState(raw, pageOrigin, metaPresent, cookieSameSite) {
|
|
174
|
+
const isIdempotent = raw.method === 'get' || raw.method === 'head';
|
|
175
|
+
const actionOrigin = originOf(raw.action);
|
|
176
|
+
const isCrossOrigin = actionOrigin !== null && pageOrigin !== null && actionOrigin !== pageOrigin;
|
|
177
|
+
const hasCsrfField = raw.csrfFieldName !== null;
|
|
178
|
+
const protectedForm = isIdempotent || hasCsrfField || metaPresent || cookieSameSite === 'Strict';
|
|
179
|
+
const state = {
|
|
180
|
+
selector: raw.selector,
|
|
181
|
+
action: raw.action,
|
|
182
|
+
method: raw.method,
|
|
183
|
+
hasCsrfField,
|
|
184
|
+
metaCsrfPresent: metaPresent,
|
|
185
|
+
cookieSameSite,
|
|
186
|
+
isIdempotent,
|
|
187
|
+
isCrossOrigin,
|
|
188
|
+
passed: protectedForm,
|
|
189
|
+
};
|
|
190
|
+
if (raw.csrfFieldName !== null)
|
|
191
|
+
state.csrfFieldName = raw.csrfFieldName;
|
|
192
|
+
if (raw.csrfFieldValue !== null && raw.csrfFieldValue.length > 0) {
|
|
193
|
+
state.csrfFieldValue = redactValue(raw.csrfFieldValue);
|
|
194
|
+
}
|
|
195
|
+
return state;
|
|
196
|
+
}
|
|
197
|
+
function collectFormIssues(raw, state) {
|
|
198
|
+
const issues = [];
|
|
199
|
+
if (!state.isIdempotent &&
|
|
200
|
+
!state.hasCsrfField &&
|
|
201
|
+
!state.metaCsrfPresent &&
|
|
202
|
+
state.cookieSameSite !== 'Strict') {
|
|
203
|
+
issues.push({
|
|
204
|
+
kind: 'form-missing-csrf',
|
|
205
|
+
selector: state.selector,
|
|
206
|
+
detail: `${state.method.toUpperCase()} form has no CSRF token, meta token, or SameSite=Strict cookie`,
|
|
207
|
+
});
|
|
208
|
+
}
|
|
209
|
+
if (raw.csrfFieldValue !== null &&
|
|
210
|
+
raw.csrfFieldValue.length > 0 &&
|
|
211
|
+
shannonEntropy(raw.csrfFieldValue) < 3) {
|
|
212
|
+
issues.push({
|
|
213
|
+
kind: 'weak-csrf-entropy',
|
|
214
|
+
selector: state.selector,
|
|
215
|
+
detail: `CSRF field "${raw.csrfFieldName ?? '?'}" has low Shannon entropy (<3 bits/char)`,
|
|
216
|
+
});
|
|
217
|
+
}
|
|
218
|
+
if (raw.actionQueryCsrf) {
|
|
219
|
+
issues.push({
|
|
220
|
+
kind: 'csrf-in-url',
|
|
221
|
+
selector: state.selector,
|
|
222
|
+
detail: 'CSRF-like parameter found in form action query string (leaks via Referer)',
|
|
223
|
+
});
|
|
224
|
+
}
|
|
225
|
+
return issues;
|
|
226
|
+
}
|
|
227
|
+
export async function auditCsrf(page, ctx) {
|
|
228
|
+
const pageUrl = page.url();
|
|
229
|
+
const pageOrigin = originOf(pageUrl);
|
|
230
|
+
const probe = await probePage(page);
|
|
231
|
+
const cookies = await ctx.cookies(pageUrl);
|
|
232
|
+
const cookieDefenses = countSameSite(cookies);
|
|
233
|
+
const cookieSameSite = dominantSameSite(cookies);
|
|
234
|
+
const csrfCookiePresent = hasCsrfCookie(cookies);
|
|
235
|
+
const metaPresent = probe.metaCsrfToken !== null;
|
|
236
|
+
const forms = probe.forms.map((raw) => buildFormState(raw, pageOrigin, metaPresent, cookieSameSite));
|
|
237
|
+
const issues = [];
|
|
238
|
+
probe.forms.forEach((raw, index) => {
|
|
239
|
+
const state = forms[index];
|
|
240
|
+
if (!state)
|
|
241
|
+
return;
|
|
242
|
+
for (const issue of collectFormIssues(raw, state))
|
|
243
|
+
issues.push(issue);
|
|
244
|
+
});
|
|
245
|
+
if (probe.inlineScriptUsesFetch &&
|
|
246
|
+
!probe.inlineScriptHasCsrfHeader &&
|
|
247
|
+
!metaPresent &&
|
|
248
|
+
!csrfCookiePresent) {
|
|
249
|
+
issues.push({
|
|
250
|
+
kind: 'csrf-header-missing',
|
|
251
|
+
detail: 'Inline scripts issue fetch/XHR without X-CSRF-Token header and no meta or cookie token is visible',
|
|
252
|
+
});
|
|
253
|
+
}
|
|
254
|
+
if (cookieDefenses.sameSiteNone > 0 && !csrfCookiePresent && !metaPresent) {
|
|
255
|
+
issues.push({
|
|
256
|
+
kind: 'samesite-none-no-csrf',
|
|
257
|
+
detail: `${cookieDefenses.sameSiteNone} cookie(s) use SameSite=None with no CSRF token visible`,
|
|
258
|
+
});
|
|
259
|
+
}
|
|
260
|
+
if (probe.pageUrlHasCsrf || urlHasCsrfParam(pageUrl)) {
|
|
261
|
+
issues.push({
|
|
262
|
+
kind: 'csrf-in-url',
|
|
263
|
+
detail: 'CSRF-like parameter visible in current page URL (leaks via Referer)',
|
|
264
|
+
});
|
|
265
|
+
}
|
|
266
|
+
const passed = issues.length === 0 && forms.every((f) => f.passed);
|
|
267
|
+
return {
|
|
268
|
+
page: pageUrl,
|
|
269
|
+
forms,
|
|
270
|
+
metaCsrfToken: probe.metaCsrfToken,
|
|
271
|
+
cookieDefenses,
|
|
272
|
+
issues,
|
|
273
|
+
passed,
|
|
274
|
+
};
|
|
275
|
+
}
|
|
276
|
+
//# sourceMappingURL=csrf-audit.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"csrf-audit.js","sourceRoot":"","sources":["../src/csrf-audit.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import type { Page } from 'playwright';
|
|
2
|
+
export interface CssCoverageFile {
|
|
3
|
+
url: string;
|
|
4
|
+
total: number;
|
|
5
|
+
used: number;
|
|
6
|
+
unusedRatio: number;
|
|
7
|
+
}
|
|
8
|
+
export interface CssCoverageResult {
|
|
9
|
+
page: string;
|
|
10
|
+
totalBytes: number;
|
|
11
|
+
usedBytes: number;
|
|
12
|
+
unusedBytes: number;
|
|
13
|
+
unusedRatio: number;
|
|
14
|
+
byFile: CssCoverageFile[];
|
|
15
|
+
passed: boolean;
|
|
16
|
+
}
|
|
17
|
+
export declare function auditCssCoverage(page: Page, opts?: {
|
|
18
|
+
threshold?: number;
|
|
19
|
+
}): Promise<CssCoverageResult>;
|
|
20
|
+
//# sourceMappingURL=css-coverage.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"css-coverage.d.ts","sourceRoot":"","sources":["../src/css-coverage.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,YAAY,CAAC;AAEvC,MAAM,WAAW,eAAe;IAC9B,GAAG,EAAE,MAAM,CAAC;IACZ,KAAK,EAAE,MAAM,CAAC;IACd,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;CACrB;AAED,MAAM,WAAW,iBAAiB;IAChC,IAAI,EAAE,MAAM,CAAC;IACb,UAAU,EAAE,MAAM,CAAC;IACnB,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,MAAM,CAAC;IACpB,MAAM,EAAE,eAAe,EAAE,CAAC;IAC1B,MAAM,EAAE,OAAO,CAAC;CACjB;AAmCD,wBAAsB,gBAAgB,CACpC,IAAI,EAAE,IAAI,EACV,IAAI,CAAC,EAAE;IAAE,SAAS,CAAC,EAAE,MAAM,CAAA;CAAE,GAC5B,OAAO,CAAC,iBAAiB,CAAC,CA8E5B"}
|