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.
Files changed (493) hide show
  1. package/README.md +332 -22
  2. package/dist/a11y-filter.d.ts +15 -0
  3. package/dist/a11y-filter.d.ts.map +1 -0
  4. package/dist/a11y-filter.js +107 -0
  5. package/dist/a11y-filter.js.map +1 -0
  6. package/dist/ab-compare.d.ts +23 -0
  7. package/dist/ab-compare.d.ts.map +1 -0
  8. package/dist/ab-compare.js +340 -0
  9. package/dist/ab-compare.js.map +1 -0
  10. package/dist/ai-codegen.d.ts +30 -0
  11. package/dist/ai-codegen.d.ts.map +1 -0
  12. package/dist/ai-codegen.js +296 -0
  13. package/dist/ai-codegen.js.map +1 -0
  14. package/dist/ai-triage.d.ts +26 -0
  15. package/dist/ai-triage.d.ts.map +1 -0
  16. package/dist/ai-triage.js +207 -0
  17. package/dist/ai-triage.js.map +1 -0
  18. package/dist/amp.d.ts +32 -0
  19. package/dist/amp.d.ts.map +1 -0
  20. package/dist/amp.js +179 -0
  21. package/dist/amp.js.map +1 -0
  22. package/dist/animation-audit.d.ts +25 -0
  23. package/dist/animation-audit.d.ts.map +1 -0
  24. package/dist/animation-audit.js +296 -0
  25. package/dist/animation-audit.js.map +1 -0
  26. package/dist/api.d.ts +3 -0
  27. package/dist/api.d.ts.map +1 -0
  28. package/dist/api.js +85 -0
  29. package/dist/api.js.map +1 -0
  30. package/dist/aria-audit.d.ts +20 -0
  31. package/dist/aria-audit.d.ts.map +1 -0
  32. package/dist/aria-audit.js +445 -0
  33. package/dist/aria-audit.js.map +1 -0
  34. package/dist/assertions.d.ts +30 -0
  35. package/dist/assertions.d.ts.map +1 -0
  36. package/dist/assertions.js +342 -0
  37. package/dist/assertions.js.map +1 -0
  38. package/dist/autofix.d.ts +40 -0
  39. package/dist/autofix.d.ts.map +1 -0
  40. package/dist/autofix.js +244 -0
  41. package/dist/autofix.js.map +1 -0
  42. package/dist/badge.d.ts +27 -0
  43. package/dist/badge.d.ts.map +1 -0
  44. package/dist/badge.js +183 -0
  45. package/dist/badge.js.map +1 -0
  46. package/dist/baseline-drift.d.ts +43 -0
  47. package/dist/baseline-drift.d.ts.map +1 -0
  48. package/dist/baseline-drift.js +208 -0
  49. package/dist/baseline-drift.js.map +1 -0
  50. package/dist/bdd.d.ts +31 -0
  51. package/dist/bdd.d.ts.map +1 -0
  52. package/dist/bdd.js +316 -0
  53. package/dist/bdd.js.map +1 -0
  54. package/dist/bisect.d.ts +32 -0
  55. package/dist/bisect.d.ts.map +1 -0
  56. package/dist/bisect.js +253 -0
  57. package/dist/bisect.js.map +1 -0
  58. package/dist/budget-diff.d.ts +37 -0
  59. package/dist/budget-diff.d.ts.map +1 -0
  60. package/dist/budget-diff.js +273 -0
  61. package/dist/budget-diff.js.map +1 -0
  62. package/dist/budget-file.d.ts +15 -0
  63. package/dist/budget-file.d.ts.map +1 -0
  64. package/dist/budget-file.js +185 -0
  65. package/dist/budget-file.js.map +1 -0
  66. package/dist/bundle-size.d.ts +36 -0
  67. package/dist/bundle-size.d.ts.map +1 -0
  68. package/dist/bundle-size.js +347 -0
  69. package/dist/bundle-size.js.map +1 -0
  70. package/dist/cache-headers.d.ts +33 -0
  71. package/dist/cache-headers.d.ts.map +1 -0
  72. package/dist/cache-headers.js +270 -0
  73. package/dist/cache-headers.js.map +1 -0
  74. package/dist/canonical-audit.d.ts +19 -0
  75. package/dist/canonical-audit.d.ts.map +1 -0
  76. package/dist/canonical-audit.js +196 -0
  77. package/dist/canonical-audit.js.map +1 -0
  78. package/dist/chaos.d.ts +38 -0
  79. package/dist/chaos.d.ts.map +1 -0
  80. package/dist/chaos.js +348 -0
  81. package/dist/chaos.js.map +1 -0
  82. package/dist/cli.js +201 -23
  83. package/dist/cli.js.map +1 -1
  84. package/dist/clickjacking-audit.d.ts +18 -0
  85. package/dist/clickjacking-audit.d.ts.map +1 -0
  86. package/dist/clickjacking-audit.js +231 -0
  87. package/dist/clickjacking-audit.js.map +1 -0
  88. package/dist/cls-culprit.d.ts +36 -0
  89. package/dist/cls-culprit.d.ts.map +1 -0
  90. package/dist/cls-culprit.js +203 -0
  91. package/dist/cls-culprit.js.map +1 -0
  92. package/dist/cls-timeline.d.ts +30 -0
  93. package/dist/cls-timeline.d.ts.map +1 -0
  94. package/dist/cls-timeline.js +61 -0
  95. package/dist/cls-timeline.js.map +1 -0
  96. package/dist/codegen-converter.d.ts +19 -0
  97. package/dist/codegen-converter.d.ts.map +1 -0
  98. package/dist/codegen-converter.js +464 -0
  99. package/dist/codegen-converter.js.map +1 -0
  100. package/dist/compression.d.ts +14 -0
  101. package/dist/compression.d.ts.map +1 -0
  102. package/dist/compression.js +150 -0
  103. package/dist/compression.js.map +1 -0
  104. package/dist/console-errors.d.ts +24 -0
  105. package/dist/console-errors.d.ts.map +1 -0
  106. package/dist/console-errors.js +96 -0
  107. package/dist/console-errors.js.map +1 -0
  108. package/dist/content-quality.d.ts +34 -0
  109. package/dist/content-quality.d.ts.map +1 -0
  110. package/dist/content-quality.js +124 -0
  111. package/dist/content-quality.js.map +1 -0
  112. package/dist/contract-openapi.d.ts +74 -0
  113. package/dist/contract-openapi.d.ts.map +1 -0
  114. package/dist/contract-openapi.js +305 -0
  115. package/dist/contract-openapi.js.map +1 -0
  116. package/dist/cookie-banner.d.ts +27 -0
  117. package/dist/cookie-banner.d.ts.map +1 -0
  118. package/dist/cookie-banner.js +285 -0
  119. package/dist/cookie-banner.js.map +1 -0
  120. package/dist/cookie-flags-audit.d.ts +35 -0
  121. package/dist/cookie-flags-audit.d.ts.map +1 -0
  122. package/dist/cookie-flags-audit.js +167 -0
  123. package/dist/cookie-flags-audit.js.map +1 -0
  124. package/dist/cpu-throttle.d.ts +34 -0
  125. package/dist/cpu-throttle.d.ts.map +1 -0
  126. package/dist/cpu-throttle.js +149 -0
  127. package/dist/cpu-throttle.js.map +1 -0
  128. package/dist/crawl.d.ts +29 -0
  129. package/dist/crawl.d.ts.map +1 -0
  130. package/dist/crawl.js +153 -0
  131. package/dist/crawl.js.map +1 -0
  132. package/dist/critical-css.d.ts +25 -0
  133. package/dist/critical-css.d.ts.map +1 -0
  134. package/dist/critical-css.js +353 -0
  135. package/dist/critical-css.js.map +1 -0
  136. package/dist/cross-browser.d.ts +44 -0
  137. package/dist/cross-browser.d.ts.map +1 -0
  138. package/dist/cross-browser.js +300 -0
  139. package/dist/cross-browser.js.map +1 -0
  140. package/dist/csrf-audit.d.ts +33 -0
  141. package/dist/csrf-audit.d.ts.map +1 -0
  142. package/dist/csrf-audit.js +276 -0
  143. package/dist/csrf-audit.js.map +1 -0
  144. package/dist/css-coverage.d.ts +20 -0
  145. package/dist/css-coverage.d.ts.map +1 -0
  146. package/dist/css-coverage.js +91 -0
  147. package/dist/css-coverage.js.map +1 -0
  148. package/dist/csv-exporter.d.ts +34 -0
  149. package/dist/csv-exporter.d.ts.map +1 -0
  150. package/dist/csv-exporter.js +241 -0
  151. package/dist/csv-exporter.js.map +1 -0
  152. package/dist/dark-mode-audit.d.ts +31 -0
  153. package/dist/dark-mode-audit.d.ts.map +1 -0
  154. package/dist/dark-mode-audit.js +236 -0
  155. package/dist/dark-mode-audit.js.map +1 -0
  156. package/dist/dead-images.d.ts +18 -0
  157. package/dist/dead-images.d.ts.map +1 -0
  158. package/dist/dead-images.js +236 -0
  159. package/dist/dead-images.js.map +1 -0
  160. package/dist/deadclicks.d.ts +19 -0
  161. package/dist/deadclicks.d.ts.map +1 -0
  162. package/dist/deadclicks.js +109 -0
  163. package/dist/deadclicks.js.map +1 -0
  164. package/dist/discord-formatter.d.ts +39 -0
  165. package/dist/discord-formatter.d.ts.map +1 -0
  166. package/dist/discord-formatter.js +191 -0
  167. package/dist/discord-formatter.js.map +1 -0
  168. package/dist/dom-audit.d.ts +23 -0
  169. package/dist/dom-audit.d.ts.map +1 -0
  170. package/dist/dom-audit.js +111 -0
  171. package/dist/dom-audit.js.map +1 -0
  172. package/dist/driver.d.ts.map +1 -1
  173. package/dist/driver.js +10 -0
  174. package/dist/driver.js.map +1 -1
  175. package/dist/error-page-audit.d.ts +26 -0
  176. package/dist/error-page-audit.d.ts.map +1 -0
  177. package/dist/error-page-audit.js +219 -0
  178. package/dist/error-page-audit.js.map +1 -0
  179. package/dist/event-listener-audit.d.ts +22 -0
  180. package/dist/event-listener-audit.d.ts.map +1 -0
  181. package/dist/event-listener-audit.js +156 -0
  182. package/dist/event-listener-audit.js.map +1 -0
  183. package/dist/exposed-paths.d.ts +21 -0
  184. package/dist/exposed-paths.d.ts.map +1 -0
  185. package/dist/exposed-paths.js +116 -0
  186. package/dist/exposed-paths.js.map +1 -0
  187. package/dist/favicon-audit.d.ts +28 -0
  188. package/dist/favicon-audit.d.ts.map +1 -0
  189. package/dist/favicon-audit.js +358 -0
  190. package/dist/favicon-audit.js.map +1 -0
  191. package/dist/flaky-detector.d.ts +32 -0
  192. package/dist/flaky-detector.d.ts.map +1 -0
  193. package/dist/flaky-detector.js +254 -0
  194. package/dist/flaky-detector.js.map +1 -0
  195. package/dist/flaky.d.ts +28 -0
  196. package/dist/flaky.d.ts.map +1 -0
  197. package/dist/flaky.js +106 -0
  198. package/dist/flaky.js.map +1 -0
  199. package/dist/focus-trap-audit.d.ts +29 -0
  200. package/dist/focus-trap-audit.d.ts.map +1 -0
  201. package/dist/focus-trap-audit.js +285 -0
  202. package/dist/focus-trap-audit.js.map +1 -0
  203. package/dist/font-loading.d.ts +29 -0
  204. package/dist/font-loading.d.ts.map +1 -0
  205. package/dist/font-loading.js +216 -0
  206. package/dist/font-loading.js.map +1 -0
  207. package/dist/forms-audit.d.ts +23 -0
  208. package/dist/forms-audit.d.ts.map +1 -0
  209. package/dist/forms-audit.js +147 -0
  210. package/dist/forms-audit.js.map +1 -0
  211. package/dist/github-annotations.d.ts +17 -0
  212. package/dist/github-annotations.d.ts.map +1 -0
  213. package/dist/github-annotations.js +264 -0
  214. package/dist/github-annotations.js.map +1 -0
  215. package/dist/graphql.d.ts +60 -0
  216. package/dist/graphql.d.ts.map +1 -0
  217. package/dist/graphql.js +188 -0
  218. package/dist/graphql.js.map +1 -0
  219. package/dist/har-waterfall.d.ts +37 -0
  220. package/dist/har-waterfall.d.ts.map +1 -0
  221. package/dist/har-waterfall.js +376 -0
  222. package/dist/har-waterfall.js.map +1 -0
  223. package/dist/heading-hierarchy.d.ts +20 -0
  224. package/dist/heading-hierarchy.d.ts.map +1 -0
  225. package/dist/heading-hierarchy.js +112 -0
  226. package/dist/heading-hierarchy.js.map +1 -0
  227. package/dist/headless-detect.d.ts +22 -0
  228. package/dist/headless-detect.d.ts.map +1 -0
  229. package/dist/headless-detect.js +167 -0
  230. package/dist/headless-detect.js.map +1 -0
  231. package/dist/history-timeline.d.ts +13 -0
  232. package/dist/history-timeline.d.ts.map +1 -0
  233. package/dist/history-timeline.js +327 -0
  234. package/dist/history-timeline.js.map +1 -0
  235. package/dist/hreflang-audit.d.ts +26 -0
  236. package/dist/hreflang-audit.d.ts.map +1 -0
  237. package/dist/hreflang-audit.js +273 -0
  238. package/dist/hreflang-audit.js.map +1 -0
  239. package/dist/hydration-audit.d.ts +21 -0
  240. package/dist/hydration-audit.d.ts.map +1 -0
  241. package/dist/hydration-audit.js +277 -0
  242. package/dist/hydration-audit.js.map +1 -0
  243. package/dist/image-audit.d.ts +41 -0
  244. package/dist/image-audit.d.ts.map +1 -0
  245. package/dist/image-audit.js +229 -0
  246. package/dist/image-audit.js.map +1 -0
  247. package/dist/index.d.ts +119 -0
  248. package/dist/index.d.ts.map +1 -1
  249. package/dist/index.js +708 -2
  250. package/dist/index.js.map +1 -1
  251. package/dist/init-wizard.d.ts +33 -0
  252. package/dist/init-wizard.d.ts.map +1 -0
  253. package/dist/init-wizard.js +289 -0
  254. package/dist/init-wizard.js.map +1 -0
  255. package/dist/inp-audit.d.ts +26 -0
  256. package/dist/inp-audit.d.ts.map +1 -0
  257. package/dist/inp-audit.js +202 -0
  258. package/dist/inp-audit.js.map +1 -0
  259. package/dist/js-coverage.d.ts +20 -0
  260. package/dist/js-coverage.d.ts.map +1 -0
  261. package/dist/js-coverage.js +81 -0
  262. package/dist/js-coverage.js.map +1 -0
  263. package/dist/json-schema.d.ts +27 -0
  264. package/dist/json-schema.d.ts.map +1 -0
  265. package/dist/json-schema.js +284 -0
  266. package/dist/json-schema.js.map +1 -0
  267. package/dist/keyboard.d.ts +21 -0
  268. package/dist/keyboard.d.ts.map +1 -0
  269. package/dist/keyboard.js +119 -0
  270. package/dist/keyboard.js.map +1 -0
  271. package/dist/lang-audit.d.ts +24 -0
  272. package/dist/lang-audit.d.ts.map +1 -0
  273. package/dist/lang-audit.js +141 -0
  274. package/dist/lang-audit.js.map +1 -0
  275. package/dist/lcp-element.d.ts +22 -0
  276. package/dist/lcp-element.d.ts.map +1 -0
  277. package/dist/lcp-element.js +240 -0
  278. package/dist/lcp-element.js.map +1 -0
  279. package/dist/longtasks.d.ts +38 -0
  280. package/dist/longtasks.d.ts.map +1 -0
  281. package/dist/longtasks.js +97 -0
  282. package/dist/longtasks.js.map +1 -0
  283. package/dist/mailbox.d.ts +35 -0
  284. package/dist/mailbox.d.ts.map +1 -0
  285. package/dist/mailbox.js +207 -0
  286. package/dist/mailbox.js.map +1 -0
  287. package/dist/media-audit.d.ts +20 -0
  288. package/dist/media-audit.d.ts.map +1 -0
  289. package/dist/media-audit.js +182 -0
  290. package/dist/media-audit.js.map +1 -0
  291. package/dist/metrics-exporter.d.ts +23 -0
  292. package/dist/metrics-exporter.d.ts.map +1 -0
  293. package/dist/metrics-exporter.js +297 -0
  294. package/dist/metrics-exporter.js.map +1 -0
  295. package/dist/mixed-content.d.ts +19 -0
  296. package/dist/mixed-content.d.ts.map +1 -0
  297. package/dist/mixed-content.js +86 -0
  298. package/dist/mixed-content.js.map +1 -0
  299. package/dist/motion-prefs.d.ts +21 -0
  300. package/dist/motion-prefs.d.ts.map +1 -0
  301. package/dist/motion-prefs.js +170 -0
  302. package/dist/motion-prefs.js.map +1 -0
  303. package/dist/open-graph.d.ts +40 -0
  304. package/dist/open-graph.d.ts.map +1 -0
  305. package/dist/open-graph.js +200 -0
  306. package/dist/open-graph.js.map +1 -0
  307. package/dist/orphan-assets.d.ts +17 -0
  308. package/dist/orphan-assets.d.ts.map +1 -0
  309. package/dist/orphan-assets.js +174 -0
  310. package/dist/orphan-assets.js.map +1 -0
  311. package/dist/page-object.d.ts +18 -0
  312. package/dist/page-object.d.ts.map +1 -0
  313. package/dist/page-object.js +346 -0
  314. package/dist/page-object.js.map +1 -0
  315. package/dist/pagination-audit.d.ts +24 -0
  316. package/dist/pagination-audit.d.ts.map +1 -0
  317. package/dist/pagination-audit.js +285 -0
  318. package/dist/pagination-audit.js.map +1 -0
  319. package/dist/passive-security.d.ts +19 -0
  320. package/dist/passive-security.d.ts.map +1 -0
  321. package/dist/passive-security.js +149 -0
  322. package/dist/passive-security.js.map +1 -0
  323. package/dist/pr-comment.d.ts +13 -0
  324. package/dist/pr-comment.d.ts.map +1 -0
  325. package/dist/pr-comment.js +316 -0
  326. package/dist/pr-comment.js.map +1 -0
  327. package/dist/precommit.d.ts +24 -0
  328. package/dist/precommit.d.ts.map +1 -0
  329. package/dist/precommit.js +239 -0
  330. package/dist/precommit.js.map +1 -0
  331. package/dist/prerender-audit.d.ts +22 -0
  332. package/dist/prerender-audit.d.ts.map +1 -0
  333. package/dist/prerender-audit.js +158 -0
  334. package/dist/prerender-audit.js.map +1 -0
  335. package/dist/print-audit.d.ts +21 -0
  336. package/dist/print-audit.d.ts.map +1 -0
  337. package/dist/print-audit.js +281 -0
  338. package/dist/print-audit.js.map +1 -0
  339. package/dist/protocol-audit.d.ts +17 -0
  340. package/dist/protocol-audit.d.ts.map +1 -0
  341. package/dist/protocol-audit.js +128 -0
  342. package/dist/protocol-audit.js.map +1 -0
  343. package/dist/reading-level.d.ts +37 -0
  344. package/dist/reading-level.d.ts.map +1 -0
  345. package/dist/reading-level.js +220 -0
  346. package/dist/reading-level.js.map +1 -0
  347. package/dist/redirects.d.ts +24 -0
  348. package/dist/redirects.d.ts.map +1 -0
  349. package/dist/redirects.js +119 -0
  350. package/dist/redirects.js.map +1 -0
  351. package/dist/report.d.ts +1 -1
  352. package/dist/report.d.ts.map +1 -1
  353. package/dist/report.js +736 -1
  354. package/dist/report.js.map +1 -1
  355. package/dist/reporter-plugin.d.ts +32 -0
  356. package/dist/reporter-plugin.d.ts.map +1 -0
  357. package/dist/reporter-plugin.js +120 -0
  358. package/dist/reporter-plugin.js.map +1 -0
  359. package/dist/resource-hints.d.ts +23 -0
  360. package/dist/resource-hints.d.ts.map +1 -0
  361. package/dist/resource-hints.js +225 -0
  362. package/dist/resource-hints.js.map +1 -0
  363. package/dist/retire.d.ts +22 -0
  364. package/dist/retire.d.ts.map +1 -0
  365. package/dist/retire.js +140 -0
  366. package/dist/retire.js.map +1 -0
  367. package/dist/retry.d.ts +20 -0
  368. package/dist/retry.d.ts.map +1 -0
  369. package/dist/retry.js +120 -0
  370. package/dist/retry.js.map +1 -0
  371. package/dist/robots-audit.d.ts +24 -0
  372. package/dist/robots-audit.d.ts.map +1 -0
  373. package/dist/robots-audit.js +206 -0
  374. package/dist/robots-audit.js.map +1 -0
  375. package/dist/rum.d.ts +35 -0
  376. package/dist/rum.d.ts.map +1 -0
  377. package/dist/rum.js +219 -0
  378. package/dist/rum.js.map +1 -0
  379. package/dist/schedule.d.ts +30 -0
  380. package/dist/schedule.d.ts.map +1 -0
  381. package/dist/schedule.js +238 -0
  382. package/dist/schedule.js.map +1 -0
  383. package/dist/secret-scan.d.ts +24 -0
  384. package/dist/secret-scan.d.ts.map +1 -0
  385. package/dist/secret-scan.js +202 -0
  386. package/dist/secret-scan.js.map +1 -0
  387. package/dist/service-worker.d.ts +26 -0
  388. package/dist/service-worker.d.ts.map +1 -0
  389. package/dist/service-worker.js +179 -0
  390. package/dist/service-worker.js.map +1 -0
  391. package/dist/shard.d.ts +14 -0
  392. package/dist/shard.d.ts.map +1 -0
  393. package/dist/shard.js +72 -0
  394. package/dist/shard.js.map +1 -0
  395. package/dist/sitemap-flows.d.ts +13 -0
  396. package/dist/sitemap-flows.d.ts.map +1 -0
  397. package/dist/sitemap-flows.js +157 -0
  398. package/dist/sitemap-flows.js.map +1 -0
  399. package/dist/sitemap.d.ts +27 -0
  400. package/dist/sitemap.d.ts.map +1 -0
  401. package/dist/sitemap.js +137 -0
  402. package/dist/sitemap.js.map +1 -0
  403. package/dist/slack-formatter.d.ts +35 -0
  404. package/dist/slack-formatter.d.ts.map +1 -0
  405. package/dist/slack-formatter.js +193 -0
  406. package/dist/slack-formatter.js.map +1 -0
  407. package/dist/sourcemap-scan.d.ts +24 -0
  408. package/dist/sourcemap-scan.d.ts.map +1 -0
  409. package/dist/sourcemap-scan.js +232 -0
  410. package/dist/sourcemap-scan.js.map +1 -0
  411. package/dist/sri-audit.d.ts +23 -0
  412. package/dist/sri-audit.d.ts.map +1 -0
  413. package/dist/sri-audit.js +180 -0
  414. package/dist/sri-audit.js.map +1 -0
  415. package/dist/storage-audit.d.ts +28 -0
  416. package/dist/storage-audit.d.ts.map +1 -0
  417. package/dist/storage-audit.js +263 -0
  418. package/dist/storage-audit.js.map +1 -0
  419. package/dist/storybook.d.ts +48 -0
  420. package/dist/storybook.d.ts.map +1 -0
  421. package/dist/storybook.js +191 -0
  422. package/dist/storybook.js.map +1 -0
  423. package/dist/structured-data.d.ts +25 -0
  424. package/dist/structured-data.d.ts.map +1 -0
  425. package/dist/structured-data.js +164 -0
  426. package/dist/structured-data.js.map +1 -0
  427. package/dist/svg-audit.d.ts +20 -0
  428. package/dist/svg-audit.d.ts.map +1 -0
  429. package/dist/svg-audit.js +213 -0
  430. package/dist/svg-audit.js.map +1 -0
  431. package/dist/table-audit.d.ts +18 -0
  432. package/dist/table-audit.d.ts.map +1 -0
  433. package/dist/table-audit.js +188 -0
  434. package/dist/table-audit.js.map +1 -0
  435. package/dist/teams-formatter.d.ts +66 -0
  436. package/dist/teams-formatter.d.ts.map +1 -0
  437. package/dist/teams-formatter.js +194 -0
  438. package/dist/teams-formatter.js.map +1 -0
  439. package/dist/third-party.d.ts +35 -0
  440. package/dist/third-party.d.ts.map +1 -0
  441. package/dist/third-party.js +175 -0
  442. package/dist/third-party.js.map +1 -0
  443. package/dist/tls.d.ts +33 -0
  444. package/dist/tls.d.ts.map +1 -0
  445. package/dist/tls.js +122 -0
  446. package/dist/tls.js.map +1 -0
  447. package/dist/touchtargets.d.ts +22 -0
  448. package/dist/touchtargets.d.ts.map +1 -0
  449. package/dist/touchtargets.js +80 -0
  450. package/dist/touchtargets.js.map +1 -0
  451. package/dist/tracker-sniff.d.ts +25 -0
  452. package/dist/tracker-sniff.d.ts.map +1 -0
  453. package/dist/tracker-sniff.js +355 -0
  454. package/dist/tracker-sniff.js.map +1 -0
  455. package/dist/types.d.ts +265 -1
  456. package/dist/types.d.ts.map +1 -1
  457. package/dist/visual-mask.d.ts +33 -0
  458. package/dist/visual-mask.d.ts.map +1 -0
  459. package/dist/visual-mask.js +102 -0
  460. package/dist/visual-mask.js.map +1 -0
  461. package/dist/visual-ssim.d.ts +26 -0
  462. package/dist/visual-ssim.d.ts.map +1 -0
  463. package/dist/visual-ssim.js +153 -0
  464. package/dist/visual-ssim.js.map +1 -0
  465. package/dist/watch-mode.d.ts +10 -0
  466. package/dist/watch-mode.d.ts.map +1 -0
  467. package/dist/watch-mode.js +156 -0
  468. package/dist/watch-mode.js.map +1 -0
  469. package/dist/web-worker-audit.d.ts +27 -0
  470. package/dist/web-worker-audit.d.ts.map +1 -0
  471. package/dist/web-worker-audit.js +324 -0
  472. package/dist/web-worker-audit.js.map +1 -0
  473. package/dist/webfonts.d.ts +26 -0
  474. package/dist/webfonts.d.ts.map +1 -0
  475. package/dist/webfonts.js +244 -0
  476. package/dist/webfonts.js.map +1 -0
  477. package/dist/webhook-reporter.d.ts +20 -0
  478. package/dist/webhook-reporter.d.ts.map +1 -0
  479. package/dist/webhook-reporter.js +124 -0
  480. package/dist/webhook-reporter.js.map +1 -0
  481. package/dist/websocket.d.ts +39 -0
  482. package/dist/websocket.d.ts.map +1 -0
  483. package/dist/websocket.js +233 -0
  484. package/dist/websocket.js.map +1 -0
  485. package/dist/worker-runtime.d.ts +129 -0
  486. package/dist/worker-runtime.d.ts.map +1 -0
  487. package/dist/worker-runtime.js +414 -0
  488. package/dist/worker-runtime.js.map +1 -0
  489. package/dist/zindex-audit.d.ts +28 -0
  490. package/dist/zindex-audit.d.ts.map +1 -0
  491. package/dist/zindex-audit.js +291 -0
  492. package/dist/zindex-audit.js.map +1 -0
  493. 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"}