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,285 @@
1
+ export async function auditPagination(page, opts) {
2
+ const url = page.url();
3
+ const probes = Math.max(1, opts?.scrollProbes ?? 2);
4
+ const detection = await page.evaluate(() => {
5
+ const sel = (el) => {
6
+ if (el.id)
7
+ return `#${CSS.escape(el.id)}`;
8
+ const tid = el.getAttribute('data-testid');
9
+ if (tid)
10
+ return `[data-testid="${tid}"]`;
11
+ const tag = el.tagName.toLowerCase();
12
+ const cls = Array.from(el.classList).slice(0, 2);
13
+ return cls.length ? `${tag}.${cls.map((c) => CSS.escape(c)).join('.')}` : tag;
14
+ };
15
+ const NUM_RE = /^\d+$/;
16
+ const LOADMORE_RE = /(load more|show more|see more)/i;
17
+ const TOTAL_RE = /(\bof\s+\d+|\d+\s*\/\s*\d+|page\s+\d+\s+of\s+\d+)/i;
18
+ const CURRENT_CLS_RE = /(\bactive\b|\bcurrent\b|\bis-active\b|\bis-current\b)/;
19
+ let numberedNav = null;
20
+ let pageCount = 0;
21
+ for (const nav of Array.from(document.querySelectorAll('nav'))) {
22
+ const nums = Array.from(nav.querySelectorAll('a, button')).filter((l) => NUM_RE.test((l.textContent || '').trim()));
23
+ if (nums.length >= 2) {
24
+ let max = 0;
25
+ for (const n of nums) {
26
+ const v = parseInt((n.textContent || '').trim(), 10);
27
+ if (Number.isFinite(v) && v > max)
28
+ max = v;
29
+ }
30
+ numberedNav = nav;
31
+ pageCount = max;
32
+ break;
33
+ }
34
+ }
35
+ const findCurrent = (root) => {
36
+ if (root.querySelector('[aria-current="page"]'))
37
+ return true;
38
+ for (const c of Array.from(root.querySelectorAll('a, button, span, li'))) {
39
+ const cls = typeof c.className === 'string' ? c.className.toLowerCase() : '';
40
+ if (CURRENT_CLS_RE.test(cls))
41
+ return true;
42
+ }
43
+ return false;
44
+ };
45
+ let prev = null;
46
+ let next = null;
47
+ for (const c of Array.from(document.querySelectorAll('a, button'))) {
48
+ const t = ((c.textContent || '') + ' ' + (c.getAttribute('aria-label') || '')).trim();
49
+ if (!t)
50
+ continue;
51
+ if (/previous|prev/i.test(t) && !prev)
52
+ prev = c;
53
+ if (/next/i.test(t) && !next)
54
+ next = c;
55
+ if (prev && next)
56
+ break;
57
+ }
58
+ const prevNext = next || prev;
59
+ let loadMore = null;
60
+ for (const b of Array.from(document.querySelectorAll('button, a'))) {
61
+ const t = (b.textContent || '').trim();
62
+ const l = b.getAttribute('aria-label') || '';
63
+ if (LOADMORE_RE.test(t) || LOADMORE_RE.test(l)) {
64
+ loadMore = b;
65
+ break;
66
+ }
67
+ }
68
+ let kind = 'none';
69
+ let detected = false;
70
+ let hasNavLandmark = false;
71
+ let navHasLabel = false;
72
+ let controlSelector;
73
+ let loadMoreSelector;
74
+ let hasCurrentMark = false;
75
+ let linksWithoutHref = 0;
76
+ let firstFocusSelector;
77
+ let pageCountDetected;
78
+ const labelOf = (n) => n.hasAttribute('aria-label') || n.hasAttribute('aria-labelledby');
79
+ if (numberedNav && findCurrent(numberedNav)) {
80
+ kind = 'numbered';
81
+ detected = true;
82
+ hasNavLandmark = numberedNav.tagName.toLowerCase() === 'nav';
83
+ navHasLabel = labelOf(numberedNav);
84
+ pageCountDetected = pageCount;
85
+ controlSelector = sel(numberedNav);
86
+ hasCurrentMark = true;
87
+ for (const l of Array.from(numberedNav.querySelectorAll('a'))) {
88
+ if (!l.hasAttribute('href'))
89
+ linksWithoutHref++;
90
+ }
91
+ const ff = numberedNav.querySelector('a, button');
92
+ if (ff)
93
+ firstFocusSelector = sel(ff);
94
+ }
95
+ else if (prevNext) {
96
+ kind = 'prev-next';
97
+ detected = true;
98
+ const wrap = prevNext.closest('nav');
99
+ hasNavLandmark = !!wrap;
100
+ if (wrap)
101
+ navHasLabel = labelOf(wrap);
102
+ controlSelector = sel(prevNext);
103
+ hasCurrentMark = wrap ? findCurrent(wrap) : false;
104
+ if (prevNext.tagName.toLowerCase() === 'a' && !prevNext.hasAttribute('href'))
105
+ linksWithoutHref++;
106
+ firstFocusSelector = controlSelector;
107
+ }
108
+ else if (loadMore) {
109
+ kind = 'load-more';
110
+ detected = true;
111
+ loadMoreSelector = sel(loadMore);
112
+ const wrap = loadMore.closest('nav');
113
+ hasNavLandmark = !!wrap;
114
+ if (wrap)
115
+ navHasLabel = labelOf(wrap);
116
+ firstFocusSelector = loadMoreSelector;
117
+ }
118
+ if (!loadMoreSelector && loadMore)
119
+ loadMoreSelector = sel(loadMore);
120
+ let container = null;
121
+ let bestCount = 0;
122
+ for (const l of Array.from(document.querySelectorAll('ul, ol, [role="list"], [role="feed"]'))) {
123
+ if (l.children.length > bestCount && l.children.length >= 3) {
124
+ bestCount = l.children.length;
125
+ container = l;
126
+ }
127
+ }
128
+ if (!container) {
129
+ for (const c of Array.from(document.querySelectorAll('main, section, div'))) {
130
+ const tags = new Map();
131
+ for (const ch of Array.from(c.children)) {
132
+ const t = ch.tagName.toLowerCase();
133
+ tags.set(t, (tags.get(t) || 0) + 1);
134
+ }
135
+ for (const [, count] of tags) {
136
+ if (count >= 5 && count > bestCount) {
137
+ bestCount = count;
138
+ container = c;
139
+ }
140
+ }
141
+ }
142
+ }
143
+ const hasTotalCount = TOTAL_RE.test(document.body.textContent || '');
144
+ const hasAriaLive = !!document.querySelector('[aria-live="polite"],[aria-live="assertive"],[role="status"],[role="alert"],[role="log"]');
145
+ return {
146
+ kind,
147
+ detected,
148
+ hasNavLandmark,
149
+ navHasLabel,
150
+ hasCurrentMark,
151
+ linksWithoutHref,
152
+ hasTotalCount,
153
+ hasAriaLive,
154
+ initialScrollHeight: document.body.scrollHeight,
155
+ initialItemCount: container ? container.children.length : 0,
156
+ ...(pageCountDetected !== undefined ? { pageCountDetected } : {}),
157
+ ...(controlSelector !== undefined ? { controlSelector } : {}),
158
+ ...(loadMoreSelector !== undefined ? { loadMoreSelector } : {}),
159
+ ...(container ? { containerSelector: sel(container) } : {}),
160
+ ...(firstFocusSelector !== undefined ? { firstFocusSelector } : {}),
161
+ };
162
+ });
163
+ const issues = [];
164
+ let kind = detection.kind;
165
+ let updatesUrl = false;
166
+ let supportsKeyboard = false;
167
+ let focusLost = false;
168
+ let prevHeight = detection.initialScrollHeight;
169
+ let prevCount = detection.initialItemCount;
170
+ let prevUrl = url;
171
+ let scrollGrew = false;
172
+ for (let i = 0; i < probes; i++) {
173
+ await page.evaluate(() => { window.scrollTo(0, document.body.scrollHeight); });
174
+ await page.waitForTimeout(1500);
175
+ const probe = await page.evaluate((cs) => {
176
+ let count = 0;
177
+ if (cs) {
178
+ const c = document.querySelector(cs);
179
+ if (c)
180
+ count = c.children.length;
181
+ }
182
+ const a = document.activeElement;
183
+ return { scrollHeight: document.body.scrollHeight, itemCount: count, url: window.location.href, activeIsBody: !a || a === document.body };
184
+ }, detection.containerSelector ?? null);
185
+ if (probe.scrollHeight > prevHeight + 10 || probe.itemCount > prevCount) {
186
+ scrollGrew = true;
187
+ if (probe.activeIsBody)
188
+ focusLost = true;
189
+ }
190
+ if (probe.url !== prevUrl)
191
+ updatesUrl = true;
192
+ prevHeight = probe.scrollHeight;
193
+ prevCount = probe.itemCount;
194
+ prevUrl = probe.url;
195
+ }
196
+ if (!detection.detected && scrollGrew)
197
+ kind = 'infinite-scroll';
198
+ const snap = (cs) => page.evaluate((s) => {
199
+ let count = 0;
200
+ if (s) {
201
+ const c = document.querySelector(s);
202
+ if (c)
203
+ count = c.children.length;
204
+ }
205
+ return { count, height: document.body.scrollHeight };
206
+ }, cs);
207
+ if ((kind === 'numbered' || kind === 'prev-next' || kind === 'load-more') && detection.firstFocusSelector) {
208
+ try {
209
+ const handle = await page.$(detection.firstFocusSelector);
210
+ if (handle) {
211
+ await handle.focus();
212
+ const beforeUrl = page.url();
213
+ const before = await snap(detection.containerSelector ?? null);
214
+ await page.keyboard.press('Enter');
215
+ await page.waitForTimeout(800);
216
+ const afterUrl = page.url();
217
+ const after = await snap(detection.containerSelector ?? null);
218
+ if (afterUrl !== beforeUrl || after.count !== before.count || after.height !== before.height) {
219
+ supportsKeyboard = true;
220
+ }
221
+ if (afterUrl !== beforeUrl)
222
+ updatesUrl = true;
223
+ }
224
+ }
225
+ catch {
226
+ // ignore navigation errors during keyboard probe
227
+ }
228
+ }
229
+ const cSel = detection.controlSelector;
230
+ const lmSel = detection.loadMoreSelector;
231
+ const push = (type, severity, detail, selector) => {
232
+ issues.push({ type, severity, detail, ...(selector ? { selector } : {}) });
233
+ };
234
+ if (kind === 'numbered') {
235
+ if (!detection.hasNavLandmark)
236
+ push('no-nav-landmark', 'warn', 'Numbered pagination is not wrapped in a <nav> landmark', cSel);
237
+ else if (!detection.navHasLabel)
238
+ push('missing-aria-label', 'warn', '<nav> wrapping pagination has no aria-label or aria-labelledby', cSel);
239
+ if (!detection.hasCurrentMark)
240
+ push('current-page-not-marked', 'error', 'Current page is not marked with aria-current="page"', cSel);
241
+ if (detection.linksWithoutHref > 0)
242
+ push('link-without-href', 'error', `${detection.linksWithoutHref} pagination <a> element(s) have no href`, cSel);
243
+ if (!detection.hasTotalCount)
244
+ push('no-total-count', 'info', 'Numbered pagination has no visible total count (e.g. "Page X of Y")');
245
+ }
246
+ if (kind === 'prev-next') {
247
+ if (!detection.hasNavLandmark)
248
+ push('no-nav-landmark', 'warn', 'Prev/next pagination is not wrapped in a <nav> landmark', cSel);
249
+ else if (!detection.navHasLabel)
250
+ push('missing-aria-label', 'warn', '<nav> wrapping prev/next has no aria-label or aria-labelledby', cSel);
251
+ if (detection.linksWithoutHref > 0)
252
+ push('link-without-href', 'error', 'Prev/next link element has no href attribute', cSel);
253
+ }
254
+ if (kind === 'load-more' && !detection.hasAriaLive) {
255
+ push('load-more-without-aria-live', 'info', 'Load-more control found but no aria-live region exists to announce new content', lmSel);
256
+ }
257
+ if (kind === 'infinite-scroll') {
258
+ if (!lmSel)
259
+ push('infinite-scroll-no-load-more', 'warn', 'Infinite-scroll detected but no fallback "load more" button is present');
260
+ if (!updatesUrl) {
261
+ push('infinite-scroll-no-url-update', 'warn', 'Infinite-scroll appended content but URL did not change (deep links not preserved)');
262
+ push('back-button-broken', 'warn', 'No history.pushState detected during scroll — back button will not return to scroll position');
263
+ }
264
+ if (focusLost)
265
+ push('infinite-scroll-traps-focus', 'info', 'Content was appended but document focus reverted to <body>');
266
+ if (!detection.hasAriaLive)
267
+ push('load-more-without-aria-live', 'info', 'Infinite-scroll appended content but no aria-live region announces it');
268
+ }
269
+ const passed = issues.every((i) => i.severity !== 'error');
270
+ const result = {
271
+ page: url,
272
+ kind,
273
+ detected: detection.detected || kind === 'infinite-scroll',
274
+ hasNavLandmark: detection.hasNavLandmark,
275
+ supportsKeyboard,
276
+ updatesUrl,
277
+ issues,
278
+ passed,
279
+ };
280
+ if (detection.pageCountDetected !== undefined) {
281
+ result.pageCountDetected = detection.pageCountDetected;
282
+ }
283
+ return result;
284
+ }
285
+ //# sourceMappingURL=pagination-audit.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pagination-audit.js","sourceRoot":"","sources":["../src/pagination-audit.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,19 @@
1
+ import type { Page } from 'playwright';
2
+ export interface PassiveSecurityIssue {
3
+ level: 'error' | 'warn';
4
+ type: 'sri-missing' | 'mixed-content' | 'cookie-not-secure' | 'cookie-not-httponly' | 'cookie-no-samesite' | 'target-blank-no-noopener' | 'form-action-http' | 'password-on-http';
5
+ selector?: string;
6
+ url?: string;
7
+ cookieName?: string;
8
+ message: string;
9
+ }
10
+ export interface PassiveSecurityResult {
11
+ page: string;
12
+ issues: PassiveSecurityIssue[];
13
+ scannedScripts: number;
14
+ scannedLinks: number;
15
+ cookiesChecked: number;
16
+ passed: boolean;
17
+ }
18
+ export declare function auditPassiveSecurity(page: Page): Promise<PassiveSecurityResult>;
19
+ //# sourceMappingURL=passive-security.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"passive-security.d.ts","sourceRoot":"","sources":["../src/passive-security.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,YAAY,CAAC;AAEvC,MAAM,WAAW,oBAAoB;IACnC,KAAK,EAAE,OAAO,GAAG,MAAM,CAAC;IACxB,IAAI,EACA,aAAa,GACb,eAAe,GACf,mBAAmB,GACnB,qBAAqB,GACrB,oBAAoB,GACpB,0BAA0B,GAC1B,kBAAkB,GAClB,kBAAkB,CAAC;IACvB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,OAAO,EAAE,MAAM,CAAC;CACjB;AAED,MAAM,WAAW,qBAAqB;IACpC,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,oBAAoB,EAAE,CAAC;IAC/B,cAAc,EAAE,MAAM,CAAC;IACvB,YAAY,EAAE,MAAM,CAAC;IACrB,cAAc,EAAE,MAAM,CAAC;IACvB,MAAM,EAAE,OAAO,CAAC;CACjB;AAYD,wBAAsB,oBAAoB,CAAC,IAAI,EAAE,IAAI,GAAG,OAAO,CAAC,qBAAqB,CAAC,CA0JrF"}
@@ -0,0 +1,149 @@
1
+ export async function auditPassiveSecurity(page) {
2
+ const pageUrl = page.url();
3
+ let pageHost = '';
4
+ let pageProtocol = 'about:';
5
+ try {
6
+ const u = new URL(pageUrl);
7
+ pageHost = u.hostname;
8
+ pageProtocol = u.protocol;
9
+ }
10
+ catch { }
11
+ const isHttps = pageProtocol === 'https:';
12
+ const snapshot = await page.evaluate(() => {
13
+ const cssPath = (el) => {
14
+ if (el.id)
15
+ return `#${el.id}`;
16
+ const parts = [];
17
+ let node = el;
18
+ let depth = 0;
19
+ while (node && depth < 4) {
20
+ const name = node.nodeName.toLowerCase();
21
+ const parent = node.parentElement;
22
+ if (!parent) {
23
+ parts.unshift(name);
24
+ break;
25
+ }
26
+ const current = node;
27
+ const sibs = Array.from(parent.children).filter((c) => c.nodeName === current.nodeName);
28
+ const idx = sibs.indexOf(current) + 1;
29
+ parts.unshift(sibs.length > 1 ? `${name}:nth-of-type(${idx})` : name);
30
+ node = parent;
31
+ depth++;
32
+ }
33
+ return parts.join(' > ');
34
+ };
35
+ const scripts = Array.from(document.querySelectorAll('script[src]')).map((s) => ({
36
+ src: s.src,
37
+ integrity: s.getAttribute('integrity'),
38
+ crossorigin: s.getAttribute('crossorigin'),
39
+ selector: cssPath(s),
40
+ }));
41
+ const stylesheets = Array.from(document.querySelectorAll('link[rel~="stylesheet"][href]')).map((l) => ({
42
+ href: l.href,
43
+ integrity: l.getAttribute('integrity'),
44
+ crossorigin: l.getAttribute('crossorigin'),
45
+ selector: cssPath(l),
46
+ }));
47
+ const mixed = [];
48
+ for (const sel of ['script[src]', 'img[src]', 'iframe[src]', 'source[src]', 'video[src]', 'audio[src]', 'embed[src]', 'link[href]']) {
49
+ for (const el of Array.from(document.querySelectorAll(sel))) {
50
+ const raw = el.getAttribute(el.hasAttribute('src') ? 'src' : 'href') || '';
51
+ if (raw.toLowerCase().startsWith('http://')) {
52
+ mixed.push({ url: raw, selector: cssPath(el), tag: el.nodeName.toLowerCase() });
53
+ }
54
+ }
55
+ }
56
+ const formsHttp = Array.from(document.querySelectorAll('form[action]'))
57
+ .filter((f) => (f.getAttribute('action') || '').toLowerCase().startsWith('http://'))
58
+ .map((f) => ({ action: f.getAttribute('action') || '', selector: cssPath(f) }));
59
+ const passwords = Array.from(document.querySelectorAll('input[type="password"]')).map((i) => ({ selector: cssPath(i) }));
60
+ const blankLinks = Array.from(document.querySelectorAll('a[target="_blank"]')).map((a) => ({
61
+ href: a.href,
62
+ rel: (a.getAttribute('rel') || '').toLowerCase(),
63
+ selector: cssPath(a),
64
+ }));
65
+ return { pageProtocol: location.protocol, scripts, stylesheets, mixed, formsHttp, passwords, blankLinks };
66
+ });
67
+ const issues = [];
68
+ const isCrossOrigin = (resourceUrl) => {
69
+ try {
70
+ const u = new URL(resourceUrl, pageUrl);
71
+ if (u.protocol !== 'http:' && u.protocol !== 'https:')
72
+ return false;
73
+ return u.hostname !== pageHost;
74
+ }
75
+ catch {
76
+ return false;
77
+ }
78
+ };
79
+ for (const s of snapshot.scripts) {
80
+ if (isCrossOrigin(s.src) && (!s.integrity || !s.crossorigin)) {
81
+ issues.push({
82
+ level: 'warn', type: 'sri-missing', selector: s.selector, url: s.src,
83
+ message: `cross-origin script missing ${!s.integrity ? 'integrity' : 'crossorigin'} attribute`,
84
+ });
85
+ }
86
+ }
87
+ for (const l of snapshot.stylesheets) {
88
+ if (isCrossOrigin(l.href) && (!l.integrity || !l.crossorigin)) {
89
+ issues.push({
90
+ level: 'warn', type: 'sri-missing', selector: l.selector, url: l.href,
91
+ message: `cross-origin stylesheet missing ${!l.integrity ? 'integrity' : 'crossorigin'} attribute`,
92
+ });
93
+ }
94
+ }
95
+ if (isHttps) {
96
+ for (const m of snapshot.mixed) {
97
+ issues.push({
98
+ level: 'error', type: 'mixed-content', selector: m.selector, url: m.url,
99
+ message: `mixed content: <${m.tag}> loads http:// resource on https page`,
100
+ });
101
+ }
102
+ for (const f of snapshot.formsHttp) {
103
+ issues.push({
104
+ level: 'error', type: 'form-action-http', selector: f.selector, url: f.action,
105
+ message: 'form posts to http:// from https page',
106
+ });
107
+ }
108
+ }
109
+ if (snapshot.pageProtocol === 'http:') {
110
+ for (const p of snapshot.passwords) {
111
+ issues.push({ level: 'error', type: 'password-on-http', selector: p.selector, message: 'password input served over http://' });
112
+ }
113
+ }
114
+ for (const a of snapshot.blankLinks) {
115
+ const hasNoopener = /(^|\s)noopener(\s|$)/.test(a.rel);
116
+ const hasNoreferrer = /(^|\s)noreferrer(\s|$)/.test(a.rel);
117
+ if (!hasNoopener && !hasNoreferrer) {
118
+ issues.push({
119
+ level: 'warn', type: 'target-blank-no-noopener', selector: a.selector, url: a.href,
120
+ message: 'target="_blank" without rel="noopener" enables reverse tabnabbing',
121
+ });
122
+ }
123
+ }
124
+ const cookies = await page.context().cookies();
125
+ for (const c of cookies) {
126
+ if (c.secure === false) {
127
+ issues.push({
128
+ level: isHttps ? 'error' : 'warn', type: 'cookie-not-secure', cookieName: c.name,
129
+ message: `cookie "${c.name}" missing Secure flag`,
130
+ });
131
+ }
132
+ if (c.httpOnly === false) {
133
+ issues.push({ level: 'warn', type: 'cookie-not-httponly', cookieName: c.name, message: `cookie "${c.name}" missing HttpOnly flag` });
134
+ }
135
+ const ss = c.sameSite;
136
+ if (!ss || ss === '') {
137
+ issues.push({ level: 'warn', type: 'cookie-no-samesite', cookieName: c.name, message: `cookie "${c.name}" missing SameSite attribute` });
138
+ }
139
+ }
140
+ return {
141
+ page: pageUrl,
142
+ issues,
143
+ scannedScripts: snapshot.scripts.length,
144
+ scannedLinks: snapshot.stylesheets.length,
145
+ cookiesChecked: cookies.length,
146
+ passed: !issues.some((i) => i.level === 'error'),
147
+ };
148
+ }
149
+ //# sourceMappingURL=passive-security.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"passive-security.js","sourceRoot":"","sources":["../src/passive-security.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,13 @@
1
+ import type { InspectResult } from './types.js';
2
+ export type CommentFlavor = 'github' | 'gitlab' | 'bitbucket';
3
+ export interface PrCommentOptions {
4
+ reportUrl?: string;
5
+ compareTo?: InspectResult;
6
+ maxFailures?: number;
7
+ includeSummary?: boolean;
8
+ }
9
+ export declare function renderGithubComment(result: InspectResult, opts?: PrCommentOptions): string;
10
+ export declare function renderGitlabComment(result: InspectResult, opts?: PrCommentOptions): string;
11
+ export declare function renderBitbucketComment(result: InspectResult, opts?: PrCommentOptions): string;
12
+ export declare function toPrComment(result: InspectResult, flavor: CommentFlavor, opts?: PrCommentOptions): string;
13
+ //# sourceMappingURL=pr-comment.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pr-comment.d.ts","sourceRoot":"","sources":["../src/pr-comment.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,aAAa,EAAiB,MAAM,YAAY,CAAC;AAE/D,MAAM,MAAM,aAAa,GAAG,QAAQ,GAAG,QAAQ,GAAG,WAAW,CAAC;AAE9D,MAAM,WAAW,gBAAgB;IAC/B,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,SAAS,CAAC,EAAE,aAAa,CAAC;IAC1B,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,cAAc,CAAC,EAAE,OAAO,CAAC;CAC1B;AAsSD,wBAAgB,mBAAmB,CACjC,MAAM,EAAE,aAAa,EACrB,IAAI,GAAE,gBAAqB,GAC1B,MAAM,CAoBR;AAED,wBAAgB,mBAAmB,CACjC,MAAM,EAAE,aAAa,EACrB,IAAI,GAAE,gBAAqB,GAC1B,MAAM,CAUR;AAED,wBAAgB,sBAAsB,CACpC,MAAM,EAAE,aAAa,EACrB,IAAI,GAAE,gBAAqB,GAC1B,MAAM,CAUR;AAED,wBAAgB,WAAW,CACzB,MAAM,EAAE,aAAa,EACrB,MAAM,EAAE,aAAa,EACrB,IAAI,CAAC,EAAE,gBAAgB,GACtB,MAAM,CAaR"}