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
package/dist/tls.js
ADDED
|
@@ -0,0 +1,122 @@
|
|
|
1
|
+
import tls from 'node:tls';
|
|
2
|
+
import https from 'node:https';
|
|
3
|
+
function buildDN(obj) {
|
|
4
|
+
if (!obj)
|
|
5
|
+
return '';
|
|
6
|
+
return ['CN', 'O', 'C'].flatMap(k => (obj[k] ? [`${k}=${obj[k]}`] : [])).join(', ');
|
|
7
|
+
}
|
|
8
|
+
function checkChain(cert, selfSigned) {
|
|
9
|
+
if (selfSigned)
|
|
10
|
+
return true;
|
|
11
|
+
const visited = new Set();
|
|
12
|
+
let cur = cert;
|
|
13
|
+
let depth = 0;
|
|
14
|
+
while (cur) {
|
|
15
|
+
const fp = cur.fingerprint256 ?? cur.fingerprint;
|
|
16
|
+
if (!fp || visited.has(fp))
|
|
17
|
+
break;
|
|
18
|
+
visited.add(fp);
|
|
19
|
+
depth++;
|
|
20
|
+
const next = cur.issuerCertificate;
|
|
21
|
+
if (!next || next === cur)
|
|
22
|
+
break;
|
|
23
|
+
cur = next;
|
|
24
|
+
}
|
|
25
|
+
return depth >= 2;
|
|
26
|
+
}
|
|
27
|
+
async function fetchHsts(host, port, timeoutMs) {
|
|
28
|
+
return new Promise(resolve => {
|
|
29
|
+
const req = https.request({ host, port, path: '/', method: 'GET', rejectUnauthorized: false, timeout: timeoutMs }, res => {
|
|
30
|
+
resolve(res.headers['strict-transport-security']);
|
|
31
|
+
res.destroy();
|
|
32
|
+
});
|
|
33
|
+
req.on('timeout', () => { req.destroy(); resolve(undefined); });
|
|
34
|
+
req.on('error', () => resolve(undefined));
|
|
35
|
+
req.end();
|
|
36
|
+
});
|
|
37
|
+
}
|
|
38
|
+
export async function auditTls(hostOrUrl, opts) {
|
|
39
|
+
const timeoutMs = opts?.timeoutMs ?? 5000;
|
|
40
|
+
const port = opts?.port ?? 443;
|
|
41
|
+
let host;
|
|
42
|
+
if (/^https?:\/\//i.test(hostOrUrl)) {
|
|
43
|
+
host = new URL(hostOrUrl).hostname;
|
|
44
|
+
}
|
|
45
|
+
else {
|
|
46
|
+
host = hostOrUrl;
|
|
47
|
+
}
|
|
48
|
+
const issues = [];
|
|
49
|
+
let socket = null;
|
|
50
|
+
const tlsResult = await new Promise((resolve, reject) => {
|
|
51
|
+
socket = tls.connect({ host, port, servername: host, rejectUnauthorized: false, ALPNProtocols: ['h2', 'http/1.1'], timeout: timeoutMs }, () => {
|
|
52
|
+
const s = socket;
|
|
53
|
+
resolve({
|
|
54
|
+
protocol: s.getProtocol() ?? undefined,
|
|
55
|
+
cipher: s.getCipher(),
|
|
56
|
+
cert: s.getPeerCertificate(true),
|
|
57
|
+
});
|
|
58
|
+
});
|
|
59
|
+
socket.on('timeout', () => reject(new Error('TLS connection timed out')));
|
|
60
|
+
socket.on('error', reject);
|
|
61
|
+
}).finally(() => {
|
|
62
|
+
try {
|
|
63
|
+
socket?.end();
|
|
64
|
+
}
|
|
65
|
+
catch { /* ignore */ }
|
|
66
|
+
});
|
|
67
|
+
const { protocol, cipher, cert: rawCert } = tlsResult;
|
|
68
|
+
if (protocol && !['TLSv1.2', 'TLSv1.3'].includes(protocol)) {
|
|
69
|
+
issues.push({ level: 'error', message: `Weak TLS protocol: ${protocol}` });
|
|
70
|
+
}
|
|
71
|
+
if (cipher && /(CBC|RC4|3DES|MD5|NULL|EXPORT)/i.test(cipher.name)) {
|
|
72
|
+
issues.push({ level: 'warn', message: `Weak cipher suite: ${cipher.name}` });
|
|
73
|
+
}
|
|
74
|
+
let certInfo;
|
|
75
|
+
let chainComplete = false;
|
|
76
|
+
if (rawCert && rawCert.subject) {
|
|
77
|
+
const subject = buildDN(rawCert.subject);
|
|
78
|
+
const issuer = buildDN(rawCert.issuer);
|
|
79
|
+
const validFrom = rawCert.valid_from ?? '';
|
|
80
|
+
const validTo = rawCert.valid_to ?? '';
|
|
81
|
+
const daysUntilExpiry = Math.floor((new Date(validTo).getTime() - Date.now()) / 86_400_000);
|
|
82
|
+
const selfSigned = rawCert.subject?.CN !== undefined && rawCert.subject.CN === rawCert.issuer?.CN;
|
|
83
|
+
const keyLength = rawCert.bits;
|
|
84
|
+
chainComplete = checkChain(rawCert, selfSigned);
|
|
85
|
+
certInfo = { subject, issuer, validFrom, validTo, daysUntilExpiry, selfSigned, keyLength };
|
|
86
|
+
if (daysUntilExpiry < 7) {
|
|
87
|
+
issues.push({ level: 'error', message: `Certificate expires in ${daysUntilExpiry} day(s)` });
|
|
88
|
+
}
|
|
89
|
+
else if (daysUntilExpiry < 30) {
|
|
90
|
+
issues.push({ level: 'warn', message: `Certificate expires in ${daysUntilExpiry} day(s)` });
|
|
91
|
+
}
|
|
92
|
+
if (selfSigned && port === 443) {
|
|
93
|
+
issues.push({ level: 'error', message: 'Self-signed certificate on port 443' });
|
|
94
|
+
}
|
|
95
|
+
if (keyLength !== undefined && keyLength < 2048) {
|
|
96
|
+
issues.push({ level: 'warn', message: `Weak key length: ${keyLength} bits` });
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
const hstsHeader = await fetchHsts(host, port, timeoutMs);
|
|
100
|
+
let hstsPreloadEligible = false;
|
|
101
|
+
if (hstsHeader) {
|
|
102
|
+
const maxAgeMatch = /max-age=(\d+)/i.exec(hstsHeader);
|
|
103
|
+
const maxAge = maxAgeMatch ? parseInt(maxAgeMatch[1], 10) : 0;
|
|
104
|
+
const includeSubDomains = /includeSubDomains/i.test(hstsHeader);
|
|
105
|
+
const preload = /\bpreload\b/i.test(hstsHeader);
|
|
106
|
+
hstsPreloadEligible = maxAge >= 31_536_000 && includeSubDomains && preload;
|
|
107
|
+
}
|
|
108
|
+
const passed = issues.every(i => i.level !== 'error');
|
|
109
|
+
return {
|
|
110
|
+
host,
|
|
111
|
+
port,
|
|
112
|
+
protocol,
|
|
113
|
+
cipher,
|
|
114
|
+
cert: certInfo,
|
|
115
|
+
chainComplete,
|
|
116
|
+
hstsHeader,
|
|
117
|
+
hstsPreloadEligible,
|
|
118
|
+
issues,
|
|
119
|
+
passed,
|
|
120
|
+
};
|
|
121
|
+
}
|
|
122
|
+
//# sourceMappingURL=tls.js.map
|
package/dist/tls.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tls.js","sourceRoot":"","sources":["../src/tls.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import type { Page } from 'playwright';
|
|
2
|
+
export interface TouchTargetOptions {
|
|
3
|
+
minSize?: number;
|
|
4
|
+
onlyViewport?: boolean;
|
|
5
|
+
ignoreHidden?: boolean;
|
|
6
|
+
}
|
|
7
|
+
export interface TouchTargetFinding {
|
|
8
|
+
selector: string;
|
|
9
|
+
html: string;
|
|
10
|
+
width: number;
|
|
11
|
+
height: number;
|
|
12
|
+
overlapsWith?: string;
|
|
13
|
+
}
|
|
14
|
+
export interface TouchTargetResult {
|
|
15
|
+
page: string;
|
|
16
|
+
scanned: number;
|
|
17
|
+
tooSmall: TouchTargetFinding[];
|
|
18
|
+
overlapping: TouchTargetFinding[];
|
|
19
|
+
passed: boolean;
|
|
20
|
+
}
|
|
21
|
+
export declare function auditTouchTargets(page: Page, opts?: TouchTargetOptions): Promise<TouchTargetResult>;
|
|
22
|
+
//# sourceMappingURL=touchtargets.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"touchtargets.d.ts","sourceRoot":"","sources":["../src/touchtargets.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,YAAY,CAAC;AAEvC,MAAM,WAAW,kBAAkB;IACjC,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,YAAY,CAAC,EAAE,OAAO,CAAC;IACvB,YAAY,CAAC,EAAE,OAAO,CAAC;CACxB;AAED,MAAM,WAAW,kBAAkB;IACjC,QAAQ,EAAE,MAAM,CAAC;IACjB,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,EAAE,MAAM,CAAC;IACf,YAAY,CAAC,EAAE,MAAM,CAAC;CACvB;AAED,MAAM,WAAW,iBAAiB;IAChC,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,MAAM,CAAC;IAChB,QAAQ,EAAE,kBAAkB,EAAE,CAAC;IAC/B,WAAW,EAAE,kBAAkB,EAAE,CAAC;IAClC,MAAM,EAAE,OAAO,CAAC;CACjB;AAED,wBAAsB,iBAAiB,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,CAAC,EAAE,kBAAkB,GAAG,OAAO,CAAC,iBAAiB,CAAC,CA+FzG"}
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
export async function auditTouchTargets(page, opts) {
|
|
2
|
+
const minSize = opts?.minSize ?? 44;
|
|
3
|
+
const onlyViewport = opts?.onlyViewport ?? false;
|
|
4
|
+
const ignoreHidden = opts?.ignoreHidden ?? true;
|
|
5
|
+
const url = page.url();
|
|
6
|
+
const { scanned, tooSmall, overlapping } = await page.evaluate(({ minSize, onlyViewport, ignoreHidden }) => {
|
|
7
|
+
const SELECTOR = [
|
|
8
|
+
'a[href]', 'button',
|
|
9
|
+
'input:not([type=hidden])', 'select', 'textarea',
|
|
10
|
+
'[role=button]', '[role=link]', '[role=menuitem]',
|
|
11
|
+
'[role=tab]', '[role=checkbox]', '[role=radio]',
|
|
12
|
+
'[onclick]', '[tabindex]:not([tabindex="-1"])',
|
|
13
|
+
].join(',');
|
|
14
|
+
function buildSelector(el) {
|
|
15
|
+
const id = el.id;
|
|
16
|
+
if (id)
|
|
17
|
+
return `#${CSS.escape(id)}`;
|
|
18
|
+
const testid = el.getAttribute('data-testid');
|
|
19
|
+
if (testid)
|
|
20
|
+
return `[data-testid="${testid}"]`;
|
|
21
|
+
const tag = el.tagName.toLowerCase();
|
|
22
|
+
const classes = Array.from(el.classList).slice(0, 3);
|
|
23
|
+
return classes.length ? `${tag}.${classes.map(c => CSS.escape(c)).join('.')}` : tag;
|
|
24
|
+
}
|
|
25
|
+
function rectsOverlap(a, b) {
|
|
26
|
+
return !(a.right <= b.left || b.right <= a.left || a.bottom <= b.top || b.bottom <= a.top);
|
|
27
|
+
}
|
|
28
|
+
const nodes = Array.from(document.querySelectorAll(SELECTOR));
|
|
29
|
+
const items = [];
|
|
30
|
+
for (const el of nodes) {
|
|
31
|
+
const rect = el.getBoundingClientRect();
|
|
32
|
+
if (ignoreHidden && rect.width === 0 && rect.height === 0)
|
|
33
|
+
continue;
|
|
34
|
+
if (onlyViewport && (rect.bottom < 0 || rect.top > window.innerHeight))
|
|
35
|
+
continue;
|
|
36
|
+
items.push({
|
|
37
|
+
selector: buildSelector(el),
|
|
38
|
+
html: el.outerHTML.slice(0, 120),
|
|
39
|
+
rect,
|
|
40
|
+
});
|
|
41
|
+
}
|
|
42
|
+
const tooSmall = [];
|
|
43
|
+
const overlapping = [];
|
|
44
|
+
for (let i = 0; i < items.length; i++) {
|
|
45
|
+
const item = items[i];
|
|
46
|
+
const { rect } = item;
|
|
47
|
+
if (rect.width < minSize || rect.height < minSize) {
|
|
48
|
+
tooSmall.push({
|
|
49
|
+
selector: item.selector,
|
|
50
|
+
html: item.html,
|
|
51
|
+
width: rect.width,
|
|
52
|
+
height: rect.height,
|
|
53
|
+
});
|
|
54
|
+
for (let j = 0; j < items.length; j++) {
|
|
55
|
+
if (i === j)
|
|
56
|
+
continue;
|
|
57
|
+
if (rectsOverlap(rect, items[j].rect)) {
|
|
58
|
+
overlapping.push({
|
|
59
|
+
selector: item.selector,
|
|
60
|
+
html: item.html,
|
|
61
|
+
width: rect.width,
|
|
62
|
+
height: rect.height,
|
|
63
|
+
overlapsWith: items[j].selector,
|
|
64
|
+
});
|
|
65
|
+
break;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
return { scanned: items.length, tooSmall, overlapping };
|
|
71
|
+
}, { minSize, onlyViewport, ignoreHidden });
|
|
72
|
+
return {
|
|
73
|
+
page: url,
|
|
74
|
+
scanned,
|
|
75
|
+
tooSmall,
|
|
76
|
+
overlapping,
|
|
77
|
+
passed: tooSmall.length === 0 && overlapping.length === 0,
|
|
78
|
+
};
|
|
79
|
+
}
|
|
80
|
+
//# sourceMappingURL=touchtargets.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"touchtargets.js","sourceRoot":"","sources":["../src/touchtargets.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import type { Page } from 'playwright';
|
|
2
|
+
export type TrackerCategory = 'analytics' | 'advertising' | 'tag-manager' | 'session-replay' | 'consent' | 'fingerprint' | 'chat-widget' | 'a-b-testing' | 'heatmap' | 'other';
|
|
3
|
+
export interface TrackerEntry {
|
|
4
|
+
name: string;
|
|
5
|
+
category: TrackerCategory;
|
|
6
|
+
domains: string[];
|
|
7
|
+
requestCount: number;
|
|
8
|
+
bytesTransferred: number;
|
|
9
|
+
scriptSourceUrls: string[];
|
|
10
|
+
detectedVia: 'request-pattern' | 'global-variable' | 'cookie-name';
|
|
11
|
+
}
|
|
12
|
+
export interface TrackerSniffResult {
|
|
13
|
+
page: string;
|
|
14
|
+
trackers: TrackerEntry[];
|
|
15
|
+
stats: {
|
|
16
|
+
total: number;
|
|
17
|
+
byCategory: Record<TrackerCategory, number>;
|
|
18
|
+
totalBytes: number;
|
|
19
|
+
totalRequests: number;
|
|
20
|
+
};
|
|
21
|
+
consentRespected: boolean;
|
|
22
|
+
passed: boolean;
|
|
23
|
+
}
|
|
24
|
+
export declare function sniffTrackers(page: Page): Promise<TrackerSniffResult>;
|
|
25
|
+
//# sourceMappingURL=tracker-sniff.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tracker-sniff.d.ts","sourceRoot":"","sources":["../src/tracker-sniff.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,IAAI,EAA6B,MAAM,YAAY,CAAC;AAElE,MAAM,MAAM,eAAe,GACvB,WAAW,GACX,aAAa,GACb,aAAa,GACb,gBAAgB,GAChB,SAAS,GACT,aAAa,GACb,aAAa,GACb,aAAa,GACb,SAAS,GACT,OAAO,CAAC;AAEZ,MAAM,WAAW,YAAY;IAC3B,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,eAAe,CAAC;IAC1B,OAAO,EAAE,MAAM,EAAE,CAAC;IAClB,YAAY,EAAE,MAAM,CAAC;IACrB,gBAAgB,EAAE,MAAM,CAAC;IACzB,gBAAgB,EAAE,MAAM,EAAE,CAAC;IAC3B,WAAW,EAAE,iBAAiB,GAAG,iBAAiB,GAAG,aAAa,CAAC;CACpE;AAED,MAAM,WAAW,kBAAkB;IACjC,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,YAAY,EAAE,CAAC;IACzB,KAAK,EAAE;QACL,KAAK,EAAE,MAAM,CAAC;QACd,UAAU,EAAE,MAAM,CAAC,eAAe,EAAE,MAAM,CAAC,CAAC;QAC5C,UAAU,EAAE,MAAM,CAAC;QACnB,aAAa,EAAE,MAAM,CAAC;KACvB,CAAC;IACF,gBAAgB,EAAE,OAAO,CAAC;IAC1B,MAAM,EAAE,OAAO,CAAC;CACjB;AA8RD,wBAAsB,aAAa,CAAC,IAAI,EAAE,IAAI,GAAG,OAAO,CAAC,kBAAkB,CAAC,CAsH3E"}
|
|
@@ -0,0 +1,355 @@
|
|
|
1
|
+
const PATTERN_RULES = [
|
|
2
|
+
{ name: 'ga', category: 'analytics', hostPatterns: ['google-analytics.com', 'analytics.google.com', 'ssl.google-analytics.com'] },
|
|
3
|
+
{ name: 'gtm', category: 'tag-manager', hostPatterns: ['googletagmanager.com'] },
|
|
4
|
+
{ name: 'mixpanel', category: 'analytics', hostPatterns: ['mixpanel.com', 'api.mixpanel.com', 'cdn.mxpnl.com'] },
|
|
5
|
+
{ name: 'amplitude', category: 'analytics', hostPatterns: ['amplitude.com', 'api.amplitude.com', 'api2.amplitude.com', 'cdn.amplitude.com'] },
|
|
6
|
+
{ name: 'segment', category: 'analytics', hostPatterns: ['segment.io', 'segment.com', 'cdn.segment.io', 'api.segment.io'] },
|
|
7
|
+
{ name: 'plausible', category: 'analytics', hostPatterns: ['plausible.io'] },
|
|
8
|
+
{ name: 'fathom', category: 'analytics', hostPatterns: ['usefathom.com', 'cdn.usefathom.com'] },
|
|
9
|
+
{ name: 'cloudflare-insights', category: 'analytics', hostPatterns: ['cloudflareinsights.com', 'static.cloudflareinsights.com'] },
|
|
10
|
+
{ name: 'simple-analytics', category: 'analytics', hostPatterns: ['simpleanalytics.com', 'scripts.simpleanalyticscdn.com', 'queue.simpleanalyticscdn.com'] },
|
|
11
|
+
{ name: 'umami', category: 'analytics', hostPatterns: ['umami.is', 'cloud.umami.is'] },
|
|
12
|
+
{ name: 'matomo', category: 'analytics', hostPatterns: ['matomo.cloud', 'matomo.org'], pathPatterns: ['/matomo.js', '/piwik.js'] },
|
|
13
|
+
{ name: 'heap', category: 'analytics', hostPatterns: ['heap.io', 'heapanalytics.com'] },
|
|
14
|
+
{ name: 'posthog', category: 'analytics', hostPatterns: ['posthog.com', 'app.posthog.com', 'eu.posthog.com'] },
|
|
15
|
+
{ name: 'statcounter', category: 'analytics', hostPatterns: ['statcounter.com'] },
|
|
16
|
+
{ name: 'pendo', category: 'analytics', hostPatterns: ['pendo.io', 'cdn.pendo.io'] },
|
|
17
|
+
{ name: 'meta-pixel', category: 'advertising', hostPatterns: ['facebook.net', 'connect.facebook.net'], pathPatterns: ['/fbevents.js', '/fbds.js', '/tr'] },
|
|
18
|
+
{ name: 'ad-network-doubleclick', category: 'advertising', hostPatterns: ['doubleclick.net'] },
|
|
19
|
+
{ name: 'ad-network-googleads', category: 'advertising', hostPatterns: ['googleadservices.com', 'googlesyndication.com'] },
|
|
20
|
+
{ name: 'ad-pagead', category: 'advertising', hostPatterns: ['google.com'], pathPatterns: ['/pagead', '/ads/'] },
|
|
21
|
+
{ name: 'criteo', category: 'advertising', hostPatterns: ['criteo.com', 'criteo.net', 'static.criteo.net'] },
|
|
22
|
+
{ name: 'taboola', category: 'advertising', hostPatterns: ['taboola.com', 'cdn.taboola.com', 'trc.taboola.com'] },
|
|
23
|
+
{ name: 'outbrain', category: 'advertising', hostPatterns: ['outbrain.com', 'outbrainimg.com'] },
|
|
24
|
+
{ name: 'pubmatic', category: 'advertising', hostPatterns: ['pubmatic.com'] },
|
|
25
|
+
{ name: 'rubicon', category: 'advertising', hostPatterns: ['rubiconproject.com'] },
|
|
26
|
+
{ name: 'amazon-ads', category: 'advertising', hostPatterns: ['amazon-adsystem.com'] },
|
|
27
|
+
{ name: 'linkedin-insight', category: 'advertising', hostPatterns: ['ads.linkedin.com', 'snap.licdn.com', 'px.ads.linkedin.com'] },
|
|
28
|
+
{ name: 'twitter-ads', category: 'advertising', hostPatterns: ['ads-twitter.com', 'analytics.twitter.com', 't.co'], pathPatterns: ['/i/adsct'] },
|
|
29
|
+
{ name: 'bing-ads', category: 'advertising', hostPatterns: ['bat.bing.com'] },
|
|
30
|
+
{ name: 'reddit-ads', category: 'advertising', hostPatterns: ['redditstatic.com'], pathPatterns: ['/ads/'] },
|
|
31
|
+
{ name: 'snapchat-ads', category: 'advertising', hostPatterns: ['sc-static.net'], pathPatterns: ['/scevent.min.js'] },
|
|
32
|
+
{ name: 'tiktok-pixel', category: 'advertising', hostPatterns: ['analytics.tiktok.com'] },
|
|
33
|
+
{ name: 'pinterest-ads', category: 'advertising', hostPatterns: ['ct.pinterest.com', 's.pinimg.com'] },
|
|
34
|
+
{ name: 'tealium', category: 'tag-manager', hostPatterns: ['tealiumiq.com', 'tealium.com'] },
|
|
35
|
+
{ name: 'ensighten', category: 'tag-manager', hostPatterns: ['ensighten.com', 'nexus.ensighten.com'] },
|
|
36
|
+
{ name: 'adobe-dtm', category: 'tag-manager', hostPatterns: ['adobedtm.com'] },
|
|
37
|
+
{ name: 'hotjar', category: 'session-replay', hostPatterns: ['hotjar.com', 'static.hotjar.com', 'script.hotjar.com', 'insights.hotjar.com'] },
|
|
38
|
+
{ name: 'fullstory', category: 'session-replay', hostPatterns: ['fullstory.com', 'rs.fullstory.com', 'edge.fullstory.com'] },
|
|
39
|
+
{ name: 'logrocket', category: 'session-replay', hostPatterns: ['logrocket.com', 'cdn.logrocket.io', 'r.logrocket.io', 'logrocket.io'] },
|
|
40
|
+
{ name: 'mouseflow', category: 'session-replay', hostPatterns: ['mouseflow.com', 'cdn.mouseflow.com', 'n1.mouseflow.com'] },
|
|
41
|
+
{ name: 'smartlook', category: 'session-replay', hostPatterns: ['smartlook.com', 'rec.smartlook.com'] },
|
|
42
|
+
{ name: 'clarity', category: 'session-replay', hostPatterns: ['clarity.ms', 'c.clarity.ms'] },
|
|
43
|
+
{ name: 'inspectlet', category: 'session-replay', hostPatterns: ['inspectlet.com'] },
|
|
44
|
+
{ name: 'consent-cookiebot', category: 'consent', hostPatterns: ['cookiebot.com', 'consent.cookiebot.com'] },
|
|
45
|
+
{ name: 'consent-onetrust', category: 'consent', hostPatterns: ['onetrust.com', 'cdn.cookielaw.org', 'geolocation.onetrust.com'] },
|
|
46
|
+
{ name: 'consent-quantcast', category: 'consent', hostPatterns: ['quantcast.com'], pathPatterns: ['/cmp'] },
|
|
47
|
+
{ name: 'consent-osano', category: 'consent', hostPatterns: ['osano.com', 'cmp.osano.com'] },
|
|
48
|
+
{ name: 'consent-iubenda', category: 'consent', hostPatterns: ['iubenda.com', 'cdn.iubenda.com'] },
|
|
49
|
+
{ name: 'consent-trustarc', category: 'consent', hostPatterns: ['trustarc.com', 'consent.trustarc.com'] },
|
|
50
|
+
{ name: 'consent-usercentrics', category: 'consent', hostPatterns: ['usercentrics.eu', 'app.usercentrics.eu'] },
|
|
51
|
+
{ name: 'consent-didomi', category: 'consent', hostPatterns: ['didomi.io', 'sdk.privacy-center.org'] },
|
|
52
|
+
{ name: 'fingerprint', category: 'fingerprint', hostPatterns: ['fingerprintjs.com', 'api.fpjs.io', 'fpjscdn.net', 'metrics.fpjs.io'] },
|
|
53
|
+
{ name: 'chat-intercom', category: 'chat-widget', hostPatterns: ['intercom.io', 'intercomcdn.com', 'widget.intercom.io'] },
|
|
54
|
+
{ name: 'chat-drift', category: 'chat-widget', hostPatterns: ['drift.com', 'js.driftt.com'] },
|
|
55
|
+
{ name: 'chat-zendesk', category: 'chat-widget', hostPatterns: ['zendesk.com', 'zdassets.com'], pathPatterns: ['/widgets', '/embeddable_framework'] },
|
|
56
|
+
{ name: 'chat-tawk', category: 'chat-widget', hostPatterns: ['tawk.to', 'embed.tawk.to'] },
|
|
57
|
+
{ name: 'chat-livechat', category: 'chat-widget', hostPatterns: ['livechatinc.com', 'cdn.livechatinc.com'] },
|
|
58
|
+
{ name: 'chat-crisp', category: 'chat-widget', hostPatterns: ['crisp.chat', 'client.crisp.chat'] },
|
|
59
|
+
{ name: 'chat-helpscout', category: 'chat-widget', hostPatterns: ['helpscout.com', 'helpscout.net', 'beacon-v2.helpscout.net'] },
|
|
60
|
+
{ name: 'chat-freshchat', category: 'chat-widget', hostPatterns: ['freshchat.com', 'wchat.freshchat.com'] },
|
|
61
|
+
{ name: 'optimizely', category: 'a-b-testing', hostPatterns: ['optimizely.com', 'cdn.optimizely.com'] },
|
|
62
|
+
{ name: 'launchdarkly', category: 'a-b-testing', hostPatterns: ['launchdarkly.com', 'app.launchdarkly.com', 'clientsdk.launchdarkly.com', 'events.launchdarkly.com'] },
|
|
63
|
+
{ name: 'split-io', category: 'a-b-testing', hostPatterns: ['split.io', 'sdk.split.io', 'events.split.io'] },
|
|
64
|
+
{ name: 'vwo', category: 'a-b-testing', hostPatterns: ['visualwebsiteoptimizer.com', 'dev.visualwebsiteoptimizer.com'] },
|
|
65
|
+
{ name: 'ab-tasty', category: 'a-b-testing', hostPatterns: ['abtasty.com', 'try.abtasty.com'] },
|
|
66
|
+
{ name: 'crazyegg', category: 'heatmap', hostPatterns: ['crazyegg.com', 'script.crazyegg.com'] },
|
|
67
|
+
{ name: 'lucky-orange', category: 'heatmap', hostPatterns: ['luckyorange.com', 'luckyorange.net'] },
|
|
68
|
+
];
|
|
69
|
+
const GLOBAL_RULES = [
|
|
70
|
+
{ name: 'gtm', category: 'tag-manager', globals: ['dataLayer', 'google_tag_manager'] },
|
|
71
|
+
{ name: 'ga', category: 'analytics', globals: ['gtag', 'ga', 'GoogleAnalyticsObject'] },
|
|
72
|
+
{ name: 'meta-pixel', category: 'advertising', globals: ['fbq', '_fbq'] },
|
|
73
|
+
{ name: 'matomo', category: 'analytics', globals: ['_paq', 'Matomo'] },
|
|
74
|
+
{ name: 'mixpanel', category: 'analytics', globals: ['mixpanel'] },
|
|
75
|
+
{ name: 'amplitude', category: 'analytics', globals: ['amplitude'] },
|
|
76
|
+
{ name: 'segment', category: 'analytics', globals: ['analytics'] },
|
|
77
|
+
{ name: 'heap', category: 'analytics', globals: ['heap'] },
|
|
78
|
+
{ name: 'posthog', category: 'analytics', globals: ['posthog'] },
|
|
79
|
+
{ name: 'pendo', category: 'analytics', globals: ['pendo'] },
|
|
80
|
+
{ name: 'plausible', category: 'analytics', globals: ['plausible'] },
|
|
81
|
+
{ name: 'fathom', category: 'analytics', globals: ['fathom'] },
|
|
82
|
+
{ name: 'chat-intercom', category: 'chat-widget', globals: ['Intercom'] },
|
|
83
|
+
{ name: 'chat-drift', category: 'chat-widget', globals: ['drift'] },
|
|
84
|
+
{ name: 'chat-zendesk', category: 'chat-widget', globals: ['zE', 'zESettings'] },
|
|
85
|
+
{ name: 'chat-tawk', category: 'chat-widget', globals: ['Tawk_API'] },
|
|
86
|
+
{ name: 'chat-livechat', category: 'chat-widget', globals: ['LiveChatWidget', 'LC_API'] },
|
|
87
|
+
{ name: 'chat-crisp', category: 'chat-widget', globals: ['$crisp', 'CRISP_WEBSITE_ID'] },
|
|
88
|
+
{ name: 'hotjar', category: 'session-replay', globals: ['hj', '_hjSettings'] },
|
|
89
|
+
{ name: 'fullstory', category: 'session-replay', globals: ['FS', '_fs_host'] },
|
|
90
|
+
{ name: 'logrocket', category: 'session-replay', globals: ['LogRocket', '_LRLogger'] },
|
|
91
|
+
{ name: 'clarity', category: 'session-replay', globals: ['clarity'] },
|
|
92
|
+
{ name: 'smartlook', category: 'session-replay', globals: ['smartlook'] },
|
|
93
|
+
{ name: 'mouseflow', category: 'session-replay', globals: ['_mfq'] },
|
|
94
|
+
{ name: 'optimizely', category: 'a-b-testing', globals: ['optimizely'] },
|
|
95
|
+
{ name: 'launchdarkly', category: 'a-b-testing', globals: ['LDClient'] },
|
|
96
|
+
{ name: 'split-io', category: 'a-b-testing', globals: ['splitio'] },
|
|
97
|
+
{ name: 'vwo', category: 'a-b-testing', globals: ['_vwo_code', 'VWO'] },
|
|
98
|
+
{ name: 'fingerprint', category: 'fingerprint', globals: ['FingerprintJS', 'Fingerprint2'] },
|
|
99
|
+
{ name: 'consent-cookiebot', category: 'consent', globals: ['Cookiebot'] },
|
|
100
|
+
{ name: 'consent-onetrust', category: 'consent', globals: ['OneTrust', 'OnetrustActiveGroups'] },
|
|
101
|
+
{ name: 'consent-osano', category: 'consent', globals: ['Osano'] },
|
|
102
|
+
{ name: 'consent-iubenda', category: 'consent', globals: ['_iub'] },
|
|
103
|
+
{ name: 'consent-usercentrics', category: 'consent', globals: ['UC_UI', 'usercentrics'] },
|
|
104
|
+
];
|
|
105
|
+
const COOKIE_RULES = [
|
|
106
|
+
{ name: 'ga', category: 'analytics', cookiePatterns: [/^_ga(_|$)/, /^_gid$/, /^_gat/] },
|
|
107
|
+
{ name: 'meta-pixel', category: 'advertising', cookiePatterns: [/^_fbp$/, /^_fbc$/] },
|
|
108
|
+
{ name: 'mixpanel', category: 'analytics', cookiePatterns: [/^mp_/] },
|
|
109
|
+
{ name: 'amplitude', category: 'analytics', cookiePatterns: [/^amplitude_/, /^amp_/] },
|
|
110
|
+
{ name: 'hotjar', category: 'session-replay', cookiePatterns: [/^_hjid$/, /^_hjSession/, /^_hjIncludedIn/] },
|
|
111
|
+
{ name: 'chat-intercom', category: 'chat-widget', cookiePatterns: [/^intercom-/] },
|
|
112
|
+
{ name: 'segment', category: 'analytics', cookiePatterns: [/^ajs_/] },
|
|
113
|
+
{ name: 'bing-ads', category: 'advertising', cookiePatterns: [/^_uetsid$/, /^_uetvid$/, /^MUID$/] },
|
|
114
|
+
{ name: 'ad-network-doubleclick', category: 'advertising', cookiePatterns: [/^IDE$/, /^test_cookie$/] },
|
|
115
|
+
{ name: 'matomo', category: 'analytics', cookiePatterns: [/^_pk_/] },
|
|
116
|
+
{ name: 'heap', category: 'analytics', cookiePatterns: [/^_hp2_/] },
|
|
117
|
+
{ name: 'clarity', category: 'session-replay', cookiePatterns: [/^_clck$/, /^_clsk$/] },
|
|
118
|
+
{ name: 'optimizely', category: 'a-b-testing', cookiePatterns: [/^optimizelyEndUserId$/] },
|
|
119
|
+
{ name: 'crazyegg', category: 'heatmap', cookiePatterns: [/^_ceg\./] },
|
|
120
|
+
];
|
|
121
|
+
const CONSENT_COOKIE_NAME_RX = /(consent|gdpr|cookie_consent|cookiepolicy|cookielaw|cc_cookie|euconsent|cookiepref)/i;
|
|
122
|
+
const CONSENT_ACCEPTED_VALUE_RX = /(true|accepted|yes|granted|allow|opt[_-]?in|1\b)/i;
|
|
123
|
+
function parseHost(url) {
|
|
124
|
+
try {
|
|
125
|
+
const u = new URL(url);
|
|
126
|
+
if (u.protocol !== 'http:' && u.protocol !== 'https:')
|
|
127
|
+
return null;
|
|
128
|
+
return u.hostname.toLowerCase();
|
|
129
|
+
}
|
|
130
|
+
catch {
|
|
131
|
+
return null;
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
function matchPattern(host, url, rule) {
|
|
135
|
+
const hostHit = rule.hostPatterns.some((h) => host === h || host.endsWith(`.${h}`) || host.includes(h));
|
|
136
|
+
if (!hostHit)
|
|
137
|
+
return false;
|
|
138
|
+
if (!rule.pathPatterns || rule.pathPatterns.length === 0)
|
|
139
|
+
return true;
|
|
140
|
+
return rule.pathPatterns.some((p) => url.includes(p));
|
|
141
|
+
}
|
|
142
|
+
function classifyUrl(url) {
|
|
143
|
+
const host = parseHost(url);
|
|
144
|
+
if (!host)
|
|
145
|
+
return null;
|
|
146
|
+
for (const rule of PATTERN_RULES) {
|
|
147
|
+
if (matchPattern(host, url, rule)) {
|
|
148
|
+
return { name: rule.name, category: rule.category };
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
return null;
|
|
152
|
+
}
|
|
153
|
+
async function detectContentLength(request) {
|
|
154
|
+
let response = null;
|
|
155
|
+
try {
|
|
156
|
+
response = await request.response();
|
|
157
|
+
}
|
|
158
|
+
catch {
|
|
159
|
+
return 0;
|
|
160
|
+
}
|
|
161
|
+
if (!response)
|
|
162
|
+
return 0;
|
|
163
|
+
try {
|
|
164
|
+
const headers = await response.allHeaders();
|
|
165
|
+
const lenHeader = headers['content-length'];
|
|
166
|
+
if (typeof lenHeader === 'string') {
|
|
167
|
+
const parsed = parseInt(lenHeader, 10);
|
|
168
|
+
if (!Number.isNaN(parsed) && parsed >= 0)
|
|
169
|
+
return parsed;
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
catch {
|
|
173
|
+
return 0;
|
|
174
|
+
}
|
|
175
|
+
return 0;
|
|
176
|
+
}
|
|
177
|
+
function getOrCreateEntry(map, name, category, detectedVia) {
|
|
178
|
+
const existing = map.get(name);
|
|
179
|
+
if (existing)
|
|
180
|
+
return existing;
|
|
181
|
+
const created = {
|
|
182
|
+
name,
|
|
183
|
+
category,
|
|
184
|
+
domains: new Set(),
|
|
185
|
+
requestCount: 0,
|
|
186
|
+
bytesTransferred: 0,
|
|
187
|
+
scriptSourceUrls: new Set(),
|
|
188
|
+
detectedVia,
|
|
189
|
+
firstRequestTime: Number.POSITIVE_INFINITY,
|
|
190
|
+
};
|
|
191
|
+
map.set(name, created);
|
|
192
|
+
return created;
|
|
193
|
+
}
|
|
194
|
+
function findConsentTimestamp(cookies) {
|
|
195
|
+
let earliest = null;
|
|
196
|
+
for (const cookie of cookies) {
|
|
197
|
+
if (!CONSENT_COOKIE_NAME_RX.test(cookie.name))
|
|
198
|
+
continue;
|
|
199
|
+
if (!CONSENT_ACCEPTED_VALUE_RX.test(cookie.value))
|
|
200
|
+
continue;
|
|
201
|
+
const expires = typeof cookie.expires === 'number' && cookie.expires > 0 ? cookie.expires * 1000 : Date.now();
|
|
202
|
+
if (earliest === null || expires < earliest)
|
|
203
|
+
earliest = expires;
|
|
204
|
+
}
|
|
205
|
+
return earliest;
|
|
206
|
+
}
|
|
207
|
+
async function detectGlobals(page) {
|
|
208
|
+
try {
|
|
209
|
+
return await page.evaluate((rules) => {
|
|
210
|
+
const hits = [];
|
|
211
|
+
const win = window;
|
|
212
|
+
for (const rule of rules) {
|
|
213
|
+
for (const key of rule.globals) {
|
|
214
|
+
if (typeof win[key] !== 'undefined' && win[key] !== null) {
|
|
215
|
+
hits.push({ name: rule.name, category: rule.category });
|
|
216
|
+
break;
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
}
|
|
220
|
+
return hits;
|
|
221
|
+
}, GLOBAL_RULES);
|
|
222
|
+
}
|
|
223
|
+
catch {
|
|
224
|
+
return [];
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
function toRecord(byCategory) {
|
|
228
|
+
const base = {
|
|
229
|
+
'analytics': 0,
|
|
230
|
+
'advertising': 0,
|
|
231
|
+
'tag-manager': 0,
|
|
232
|
+
'session-replay': 0,
|
|
233
|
+
'consent': 0,
|
|
234
|
+
'fingerprint': 0,
|
|
235
|
+
'chat-widget': 0,
|
|
236
|
+
'a-b-testing': 0,
|
|
237
|
+
'heatmap': 0,
|
|
238
|
+
'other': 0,
|
|
239
|
+
};
|
|
240
|
+
for (const [cat, count] of byCategory)
|
|
241
|
+
base[cat] = count;
|
|
242
|
+
return base;
|
|
243
|
+
}
|
|
244
|
+
export async function sniffTrackers(page) {
|
|
245
|
+
const pageUrl = page.url();
|
|
246
|
+
const records = [];
|
|
247
|
+
const pendingSizePromises = [];
|
|
248
|
+
const onRequestFinished = (request) => {
|
|
249
|
+
const url = request.url();
|
|
250
|
+
const host = parseHost(url);
|
|
251
|
+
if (!host)
|
|
252
|
+
return;
|
|
253
|
+
const sizePromise = detectContentLength(request).then((bytes) => {
|
|
254
|
+
const record = {
|
|
255
|
+
url,
|
|
256
|
+
host,
|
|
257
|
+
timestamp: Date.now(),
|
|
258
|
+
bytes,
|
|
259
|
+
resourceType: request.resourceType(),
|
|
260
|
+
initiatorScriptUrl: url,
|
|
261
|
+
};
|
|
262
|
+
records.push(record);
|
|
263
|
+
}).catch(() => {
|
|
264
|
+
/* swallow */
|
|
265
|
+
});
|
|
266
|
+
pendingSizePromises.push(sizePromise);
|
|
267
|
+
};
|
|
268
|
+
page.on('requestfinished', onRequestFinished);
|
|
269
|
+
try {
|
|
270
|
+
await page.waitForLoadState('networkidle', { timeout: 5000 });
|
|
271
|
+
}
|
|
272
|
+
catch {
|
|
273
|
+
/* ignore timeouts; still capture what we have */
|
|
274
|
+
}
|
|
275
|
+
page.off('requestfinished', onRequestFinished);
|
|
276
|
+
await Promise.all(pendingSizePromises).catch(() => {
|
|
277
|
+
/* ignore */
|
|
278
|
+
});
|
|
279
|
+
const aggregates = new Map();
|
|
280
|
+
for (const rec of records) {
|
|
281
|
+
const hit = classifyUrl(rec.url);
|
|
282
|
+
if (!hit)
|
|
283
|
+
continue;
|
|
284
|
+
const entry = getOrCreateEntry(aggregates, hit.name, hit.category, 'request-pattern');
|
|
285
|
+
entry.domains.add(rec.host);
|
|
286
|
+
entry.requestCount += 1;
|
|
287
|
+
entry.bytesTransferred += rec.bytes;
|
|
288
|
+
if (rec.resourceType === 'script' && rec.initiatorScriptUrl) {
|
|
289
|
+
entry.scriptSourceUrls.add(rec.initiatorScriptUrl);
|
|
290
|
+
}
|
|
291
|
+
if (rec.timestamp < entry.firstRequestTime)
|
|
292
|
+
entry.firstRequestTime = rec.timestamp;
|
|
293
|
+
}
|
|
294
|
+
const globalHits = await detectGlobals(page);
|
|
295
|
+
for (const hit of globalHits) {
|
|
296
|
+
if (!aggregates.has(hit.name)) {
|
|
297
|
+
getOrCreateEntry(aggregates, hit.name, hit.category, 'global-variable');
|
|
298
|
+
}
|
|
299
|
+
}
|
|
300
|
+
const cookies = await page.context().cookies();
|
|
301
|
+
for (const rule of COOKIE_RULES) {
|
|
302
|
+
const matchedCookies = cookies.filter((c) => rule.cookiePatterns.some((rx) => rx.test(c.name)));
|
|
303
|
+
if (matchedCookies.length === 0)
|
|
304
|
+
continue;
|
|
305
|
+
if (!aggregates.has(rule.name)) {
|
|
306
|
+
const entry = getOrCreateEntry(aggregates, rule.name, rule.category, 'cookie-name');
|
|
307
|
+
for (const c of matchedCookies)
|
|
308
|
+
entry.domains.add(c.domain.replace(/^\./, ''));
|
|
309
|
+
}
|
|
310
|
+
}
|
|
311
|
+
const consentTime = findConsentTimestamp(cookies);
|
|
312
|
+
let consentRespected = true;
|
|
313
|
+
if (aggregates.size > 0) {
|
|
314
|
+
if (consentTime === null) {
|
|
315
|
+
const anyFired = Array.from(aggregates.values()).some((e) => e.requestCount > 0);
|
|
316
|
+
consentRespected = !anyFired;
|
|
317
|
+
}
|
|
318
|
+
else {
|
|
319
|
+
consentRespected = Array.from(aggregates.values()).every((e) => e.requestCount === 0 || e.firstRequestTime >= consentTime);
|
|
320
|
+
}
|
|
321
|
+
}
|
|
322
|
+
const byCategory = new Map();
|
|
323
|
+
let totalBytes = 0;
|
|
324
|
+
let totalRequests = 0;
|
|
325
|
+
const trackers = [];
|
|
326
|
+
for (const entry of aggregates.values()) {
|
|
327
|
+
byCategory.set(entry.category, (byCategory.get(entry.category) || 0) + 1);
|
|
328
|
+
totalBytes += entry.bytesTransferred;
|
|
329
|
+
totalRequests += entry.requestCount;
|
|
330
|
+
trackers.push({
|
|
331
|
+
name: entry.name,
|
|
332
|
+
category: entry.category,
|
|
333
|
+
domains: Array.from(entry.domains).sort(),
|
|
334
|
+
requestCount: entry.requestCount,
|
|
335
|
+
bytesTransferred: entry.bytesTransferred,
|
|
336
|
+
scriptSourceUrls: Array.from(entry.scriptSourceUrls).sort(),
|
|
337
|
+
detectedVia: entry.detectedVia,
|
|
338
|
+
});
|
|
339
|
+
}
|
|
340
|
+
trackers.sort((a, b) => b.bytesTransferred - a.bytesTransferred || b.requestCount - a.requestCount || a.name.localeCompare(b.name));
|
|
341
|
+
const stats = {
|
|
342
|
+
total: trackers.length,
|
|
343
|
+
byCategory: toRecord(byCategory),
|
|
344
|
+
totalBytes,
|
|
345
|
+
totalRequests,
|
|
346
|
+
};
|
|
347
|
+
return {
|
|
348
|
+
page: pageUrl,
|
|
349
|
+
trackers,
|
|
350
|
+
stats,
|
|
351
|
+
consentRespected,
|
|
352
|
+
passed: stats.total === 0 || consentRespected,
|
|
353
|
+
};
|
|
354
|
+
}
|
|
355
|
+
//# sourceMappingURL=tracker-sniff.js.map
|