@slashgear/gdpr-cookie-scanner 3.6.0 → 3.8.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 (149) hide show
  1. package/.dockerignore +3 -0
  2. package/.gitattributes +1 -0
  3. package/.github/workflows/website.yml +80 -0
  4. package/CHANGELOG.md +52 -0
  5. package/CLAUDE.md +12 -1
  6. package/CONTRIBUTING.md +32 -4
  7. package/NEXT_STEPS.md +37 -3
  8. package/README.md +23 -0
  9. package/dist/analyzers/colour.d.ts +36 -0
  10. package/dist/analyzers/colour.d.ts.map +1 -0
  11. package/dist/analyzers/colour.js +75 -0
  12. package/dist/analyzers/colour.js.map +1 -0
  13. package/dist/analyzers/compliance.d.ts.map +1 -1
  14. package/dist/analyzers/compliance.js +24 -6
  15. package/dist/analyzers/compliance.js.map +1 -1
  16. package/dist/analyzers/tcf-decoder.d.ts +9 -0
  17. package/dist/analyzers/tcf-decoder.d.ts.map +1 -0
  18. package/dist/analyzers/tcf-decoder.js +123 -0
  19. package/dist/analyzers/tcf-decoder.js.map +1 -0
  20. package/dist/analyzers/wording.d.ts +1 -0
  21. package/dist/analyzers/wording.d.ts.map +1 -1
  22. package/dist/analyzers/wording.js +39 -0
  23. package/dist/analyzers/wording.js.map +1 -1
  24. package/dist/report/generator.d.ts +1 -0
  25. package/dist/report/generator.d.ts.map +1 -1
  26. package/dist/report/generator.js +71 -1
  27. package/dist/report/generator.js.map +1 -1
  28. package/dist/report/html.d.ts.map +1 -1
  29. package/dist/report/html.js +123 -0
  30. package/dist/report/html.js.map +1 -1
  31. package/dist/scanner/consent-modal.d.ts.map +1 -1
  32. package/dist/scanner/consent-modal.js +4 -2
  33. package/dist/scanner/consent-modal.js.map +1 -1
  34. package/dist/scanner/index.d.ts.map +1 -1
  35. package/dist/scanner/index.js +4 -0
  36. package/dist/scanner/index.js.map +1 -1
  37. package/dist/scanner/tcf.d.ts +9 -0
  38. package/dist/scanner/tcf.d.ts.map +1 -0
  39. package/dist/scanner/tcf.js +72 -0
  40. package/dist/scanner/tcf.js.map +1 -0
  41. package/dist/types.d.ts +26 -0
  42. package/dist/types.d.ts.map +1 -1
  43. package/package.json +7 -3
  44. package/pnpm-workspace.yaml +3 -0
  45. package/scripts/build-showcase.mjs +113 -0
  46. package/src/analyzers/colour.ts +89 -0
  47. package/src/analyzers/compliance.ts +35 -10
  48. package/src/analyzers/tcf-decoder.ts +130 -0
  49. package/src/analyzers/wording.ts +44 -0
  50. package/src/report/generator.ts +83 -1
  51. package/src/report/html.ts +146 -0
  52. package/src/scanner/consent-modal.ts +3 -1
  53. package/src/scanner/index.ts +5 -0
  54. package/src/scanner/tcf.ts +80 -0
  55. package/src/types.ts +29 -0
  56. package/tests/analyzers/colour.test.ts +187 -0
  57. package/tests/analyzers/compliance.test.ts +102 -0
  58. package/tests/analyzers/tcf-decoder.test.ts +292 -0
  59. package/tests/analyzers/wording.test.ts +38 -0
  60. package/tests/scanner/button-classification.test.ts +32 -0
  61. package/website/Dockerfile +55 -0
  62. package/website/node_modules/.bin/oxfmt +21 -0
  63. package/website/node_modules/.bin/oxlint +21 -0
  64. package/website/node_modules/.bin/tsc +21 -0
  65. package/website/node_modules/.bin/tsserver +21 -0
  66. package/website/node_modules/.bin/tsx +21 -0
  67. package/website/package.json +29 -0
  68. package/{docs → website/public}/index.html +88 -50
  69. package/website/public/reports/www.20minutes.fr/after-accept.png +3 -0
  70. package/website/public/reports/www.20minutes.fr/after-reject.png +3 -0
  71. package/website/public/reports/www.20minutes.fr/gdpr-report-20minutes.fr-2026-02-22.html +907 -0
  72. package/website/public/reports/www.20minutes.fr/modal-initial.png +3 -0
  73. package/website/public/reports/www.arte.tv/after-accept.png +3 -0
  74. package/website/public/reports/www.arte.tv/after-reject.png +3 -0
  75. package/website/public/reports/www.arte.tv/gdpr-report-arte.tv-2026-02-24.html +998 -0
  76. package/website/public/reports/www.arte.tv/modal-initial.png +3 -0
  77. package/website/public/reports/www.backmarket.fr/after-accept.png +3 -0
  78. package/website/public/reports/www.backmarket.fr/after-reject.png +3 -0
  79. package/website/public/reports/www.backmarket.fr/gdpr-report-backmarket.fr-2026-02-24.html +1530 -0
  80. package/website/public/reports/www.backmarket.fr/modal-initial.png +3 -0
  81. package/website/public/reports/www.deezer.com/after-accept.png +3 -0
  82. package/website/public/reports/www.deezer.com/after-reject.png +3 -0
  83. package/website/public/reports/www.deezer.com/gdpr-report-deezer.com-2026-02-22.html +1668 -0
  84. package/website/public/reports/www.deezer.com/modal-initial.png +3 -0
  85. package/website/public/reports/www.france.tv/after-accept.png +3 -0
  86. package/website/public/reports/www.france.tv/after-reject.png +3 -0
  87. package/website/public/reports/www.france.tv/gdpr-report-france.tv-2026-02-23.html +977 -0
  88. package/website/public/reports/www.france.tv/modal-initial.png +3 -0
  89. package/website/public/reports/www.m6.fr/after-accept.png +3 -0
  90. package/website/public/reports/www.m6.fr/after-reject.png +3 -0
  91. package/website/public/reports/www.m6.fr/gdpr-report-m6.fr-2026-02-28.html +1862 -0
  92. package/website/public/reports/www.m6.fr/modal-initial.png +3 -0
  93. package/website/public/reports/www.netflix.com/after-accept.png +3 -0
  94. package/website/public/reports/www.netflix.com/after-reject.png +3 -0
  95. package/website/public/reports/www.netflix.com/gdpr-report-netflix.com-2026-02-23.html +1051 -0
  96. package/website/public/reports/www.netflix.com/modal-initial.png +3 -0
  97. package/website/public/reports/www.radiofrance.fr/after-accept.png +3 -0
  98. package/website/public/reports/www.radiofrance.fr/after-reject.png +3 -0
  99. package/website/public/reports/www.radiofrance.fr/gdpr-report-radiofrance.fr-2026-02-24.html +1146 -0
  100. package/website/public/reports/www.radiofrance.fr/modal-initial.png +3 -0
  101. package/website/public/reports/www.tf1.fr/after-accept.png +3 -0
  102. package/website/public/reports/www.tf1.fr/after-reject.png +3 -0
  103. package/website/public/reports/www.tf1.fr/gdpr-report-tf1.fr-2026-02-23.html +1512 -0
  104. package/website/public/reports/www.tf1.fr/modal-initial.png +3 -0
  105. package/website/src/index.ts +15 -0
  106. package/website/src/security.ts +26 -0
  107. package/website/tsconfig.json +14 -0
  108. package/.github/workflows/pages.yml +0 -40
  109. package/docs/reports/github.com/after-accept.png +0 -0
  110. package/docs/reports/github.com/after-reject.png +0 -0
  111. package/docs/reports/github.com/gdpr-checklist-github.com-2026-02-22.md +0 -44
  112. package/docs/reports/github.com/gdpr-cookies-github.com-2026-02-22.md +0 -29
  113. package/docs/reports/github.com/gdpr-report-github.com-2026-02-22.md +0 -102
  114. package/docs/reports/github.com/gdpr-report-github.com-2026-02-22.pdf +0 -0
  115. package/docs/reports/gitlab.com/after-accept.png +0 -0
  116. package/docs/reports/gitlab.com/after-reject.png +0 -0
  117. package/docs/reports/gitlab.com/gdpr-checklist-gitlab.com-2026-02-22.md +0 -44
  118. package/docs/reports/gitlab.com/gdpr-cookies-gitlab.com-2026-02-22.md +0 -55
  119. package/docs/reports/gitlab.com/gdpr-report-gitlab.com-2026-02-22.md +0 -200
  120. package/docs/reports/gitlab.com/gdpr-report-gitlab.com-2026-02-22.pdf +0 -0
  121. package/docs/reports/gitlab.com/modal-initial.png +0 -0
  122. package/docs/reports/npmjs.com/after-accept.png +0 -0
  123. package/docs/reports/npmjs.com/after-reject.png +0 -0
  124. package/docs/reports/npmjs.com/gdpr-checklist-npmjs.com-2026-02-22.md +0 -44
  125. package/docs/reports/npmjs.com/gdpr-cookies-npmjs.com-2026-02-22.md +0 -25
  126. package/docs/reports/npmjs.com/gdpr-report-npmjs.com-2026-02-22.md +0 -88
  127. package/docs/reports/npmjs.com/gdpr-report-npmjs.com-2026-02-22.pdf +0 -0
  128. package/docs/reports/reddit.com/after-accept.png +0 -0
  129. package/docs/reports/reddit.com/after-reject.png +0 -0
  130. package/docs/reports/reddit.com/gdpr-checklist-reddit.com-2026-02-22.md +0 -44
  131. package/docs/reports/reddit.com/gdpr-cookies-reddit.com-2026-02-22.md +0 -33
  132. package/docs/reports/reddit.com/gdpr-report-reddit.com-2026-02-22.md +0 -148
  133. package/docs/reports/reddit.com/gdpr-report-reddit.com-2026-02-22.pdf +0 -0
  134. package/docs/reports/reddit.com/modal-initial.png +0 -0
  135. package/docs/reports/stackoverflow.com/after-accept.png +0 -0
  136. package/docs/reports/stackoverflow.com/after-reject.png +0 -0
  137. package/docs/reports/stackoverflow.com/gdpr-checklist-stackoverflow.com-2026-02-22.md +0 -44
  138. package/docs/reports/stackoverflow.com/gdpr-cookies-stackoverflow.com-2026-02-22.md +0 -67
  139. package/docs/reports/stackoverflow.com/gdpr-report-stackoverflow.com-2026-02-22.md +0 -206
  140. package/docs/reports/stackoverflow.com/gdpr-report-stackoverflow.com-2026-02-22.pdf +0 -0
  141. package/docs/reports/stackoverflow.com/modal-initial.png +0 -0
  142. package/docs/reports/www.afp.com/after-accept.png +0 -0
  143. package/docs/reports/www.afp.com/after-reject.png +0 -0
  144. package/docs/reports/www.afp.com/gdpr-checklist-afp.com-2026-02-22.md +0 -44
  145. package/docs/reports/www.afp.com/gdpr-cookies-afp.com-2026-02-22.md +0 -42
  146. package/docs/reports/www.afp.com/gdpr-report-afp.com-2026-02-22.md +0 -202
  147. package/docs/reports/www.afp.com/gdpr-report-afp.com-2026-02-22.pdf +0 -0
  148. package/docs/reports/www.afp.com/modal-initial.png +0 -0
  149. /package/{docs → website/public}/style.css +0 -0
@@ -0,0 +1,998 @@
1
+ <!doctype html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="UTF-8" />
5
+ <meta name="viewport" content="width=device-width, initial-scale=1.0" />
6
+ <title>GDPR Report — arte.tv</title>
7
+ <style>
8
+ *,
9
+ *::before,
10
+ *::after {
11
+ box-sizing: border-box;
12
+ }
13
+
14
+ :root {
15
+ --grade: #ea580c;
16
+ --grade-bg: #fff7ed;
17
+ --surface: #ffffff;
18
+ --bg: #f1f5f9;
19
+ --border: #e2e8f0;
20
+ --text: #0f172a;
21
+ --text-muted: #64748b;
22
+ --critical: #dc2626;
23
+ --critical-bg: #fef2f2;
24
+ --critical-border: #fecaca;
25
+ --warning: #d97706;
26
+ --warning-bg: #fffbeb;
27
+ --warning-border: #fde68a;
28
+ --ok: #16a34a;
29
+ --ok-bg: #f0fdf4;
30
+ --ok-border: #bbf7d0;
31
+ --radius: 10px;
32
+ --shadow: 0 1px 3px rgba(0, 0, 0, 0.08), 0 1px 2px rgba(0, 0, 0, 0.06);
33
+ --shadow-md: 0 4px 6px rgba(0, 0, 0, 0.07), 0 2px 4px rgba(0, 0, 0, 0.06);
34
+ }
35
+
36
+ body {
37
+ margin: 0;
38
+ background: var(--bg);
39
+ color: var(--text);
40
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", system-ui, sans-serif;
41
+ font-size: 14px;
42
+ line-height: 1.6;
43
+ }
44
+
45
+ /* ── Layout ── */
46
+ .page {
47
+ max-width: 1000px;
48
+ margin: 0 auto;
49
+ padding: 24px 16px 64px;
50
+ }
51
+
52
+ /* ── Hero ── */
53
+ .hero {
54
+ background: var(--surface);
55
+ border-radius: var(--radius);
56
+ box-shadow: var(--shadow-md);
57
+ padding: 32px 36px;
58
+ display: flex;
59
+ align-items: center;
60
+ gap: 32px;
61
+ margin-bottom: 20px;
62
+ border-top: 4px solid var(--grade);
63
+ }
64
+ .grade-badge {
65
+ flex-shrink: 0;
66
+ width: 80px;
67
+ height: 80px;
68
+ border-radius: 16px;
69
+ background: var(--grade);
70
+ color: #fff;
71
+ font-size: 42px;
72
+ font-weight: 800;
73
+ display: flex;
74
+ align-items: center;
75
+ justify-content: center;
76
+ letter-spacing: -2px;
77
+ }
78
+ .hero-info {
79
+ flex: 1;
80
+ min-width: 0;
81
+ }
82
+ .hero-info h1 {
83
+ margin: 0 0 4px;
84
+ font-size: 22px;
85
+ font-weight: 700;
86
+ color: var(--text);
87
+ white-space: nowrap;
88
+ overflow: hidden;
89
+ text-overflow: ellipsis;
90
+ }
91
+ .hero-meta {
92
+ font-size: 13px;
93
+ color: var(--text-muted);
94
+ margin: 0;
95
+ }
96
+ .hero-score {
97
+ flex-shrink: 0;
98
+ text-align: right;
99
+ }
100
+ .hero-score .score-num {
101
+ font-size: 40px;
102
+ font-weight: 800;
103
+ color: var(--grade);
104
+ line-height: 1;
105
+ }
106
+ .hero-score .score-den {
107
+ font-size: 18px;
108
+ color: var(--text-muted);
109
+ font-weight: 400;
110
+ }
111
+ .hero-score .score-label {
112
+ font-size: 12px;
113
+ color: var(--text-muted);
114
+ margin-top: 4px;
115
+ }
116
+
117
+ /* ── Score grid ── */
118
+ .score-grid {
119
+ display: grid;
120
+ grid-template-columns: repeat(4, 1fr);
121
+ gap: 12px;
122
+ margin-bottom: 20px;
123
+ }
124
+ @media (max-width: 640px) {
125
+ .score-grid {
126
+ grid-template-columns: repeat(2, 1fr);
127
+ }
128
+ }
129
+ .score-card {
130
+ background: var(--surface);
131
+ border-radius: var(--radius);
132
+ padding: 16px 18px;
133
+ box-shadow: var(--shadow);
134
+ }
135
+ .score-card-label {
136
+ font-size: 11px;
137
+ font-weight: 600;
138
+ text-transform: uppercase;
139
+ letter-spacing: 0.05em;
140
+ color: var(--text-muted);
141
+ margin-bottom: 8px;
142
+ }
143
+ .score-card-value {
144
+ font-size: 22px;
145
+ font-weight: 700;
146
+ color: var(--text);
147
+ margin-bottom: 8px;
148
+ }
149
+ .score-card-value span {
150
+ font-size: 14px;
151
+ font-weight: 400;
152
+ color: var(--text-muted);
153
+ }
154
+ .progress-track {
155
+ height: 6px;
156
+ background: var(--border);
157
+ border-radius: 3px;
158
+ overflow: hidden;
159
+ }
160
+ .progress-fill {
161
+ height: 100%;
162
+ border-radius: 3px;
163
+ background: var(--grade-color, #64748b);
164
+ }
165
+
166
+ /* ── Section ── */
167
+ .section {
168
+ background: var(--surface);
169
+ border-radius: var(--radius);
170
+ box-shadow: var(--shadow);
171
+ margin-bottom: 16px;
172
+ overflow: hidden;
173
+ }
174
+ .section-header {
175
+ padding: 16px 20px;
176
+ border-bottom: 1px solid var(--border);
177
+ display: flex;
178
+ align-items: center;
179
+ gap: 10px;
180
+ }
181
+ .section-header h2 {
182
+ margin: 0;
183
+ font-size: 15px;
184
+ font-weight: 600;
185
+ color: var(--text);
186
+ }
187
+ .section-body {
188
+ padding: 20px;
189
+ }
190
+ .section-body.no-pad {
191
+ padding: 0;
192
+ }
193
+
194
+ /* ── Badges ── */
195
+ .badge {
196
+ display: inline-flex;
197
+ align-items: center;
198
+ gap: 4px;
199
+ padding: 2px 8px;
200
+ border-radius: 99px;
201
+ font-size: 11px;
202
+ font-weight: 600;
203
+ }
204
+ .badge-critical {
205
+ background: var(--critical-bg);
206
+ color: var(--critical);
207
+ border: 1px solid var(--critical-border);
208
+ }
209
+ .badge-warning {
210
+ background: var(--warning-bg);
211
+ color: var(--warning);
212
+ border: 1px solid var(--warning-border);
213
+ }
214
+ .badge-ok {
215
+ background: var(--ok-bg);
216
+ color: var(--ok);
217
+ border: 1px solid var(--ok-border);
218
+ }
219
+ .badge-muted {
220
+ background: var(--bg);
221
+ color: var(--text-muted);
222
+ border: 1px solid var(--border);
223
+ }
224
+ .count-badge {
225
+ background: var(--bg);
226
+ color: var(--text-muted);
227
+ font-size: 12px;
228
+ font-weight: 600;
229
+ padding: 1px 8px;
230
+ border-radius: 99px;
231
+ margin-left: auto;
232
+ }
233
+
234
+ /* ── Issue cards ── */
235
+ .issue-list {
236
+ display: flex;
237
+ flex-direction: column;
238
+ gap: 10px;
239
+ }
240
+ .issue-card {
241
+ border-radius: 8px;
242
+ padding: 14px 16px;
243
+ border: 1px solid;
244
+ }
245
+ .issue-card.critical {
246
+ background: var(--critical-bg);
247
+ border-color: var(--critical-border);
248
+ }
249
+ .issue-card.warning {
250
+ background: var(--warning-bg);
251
+ border-color: var(--warning-border);
252
+ }
253
+ .issue-title {
254
+ font-size: 13px;
255
+ font-weight: 600;
256
+ margin-bottom: 4px;
257
+ }
258
+ .issue-card.critical .issue-title {
259
+ color: var(--critical);
260
+ }
261
+ .issue-card.warning .issue-title {
262
+ color: var(--warning);
263
+ }
264
+ .issue-evidence {
265
+ font-size: 12px;
266
+ color: var(--text-muted);
267
+ margin: 0;
268
+ }
269
+ .no-issues {
270
+ display: flex;
271
+ align-items: center;
272
+ gap: 8px;
273
+ color: var(--ok);
274
+ font-weight: 500;
275
+ font-size: 14px;
276
+ }
277
+
278
+ /* ── Tables ── */
279
+ .data-table {
280
+ width: 100%;
281
+ border-collapse: collapse;
282
+ font-size: 13px;
283
+ }
284
+ .data-table th {
285
+ background: var(--bg);
286
+ padding: 10px 14px;
287
+ text-align: left;
288
+ font-size: 11px;
289
+ font-weight: 600;
290
+ text-transform: uppercase;
291
+ letter-spacing: 0.04em;
292
+ color: var(--text-muted);
293
+ border-bottom: 1px solid var(--border);
294
+ white-space: nowrap;
295
+ }
296
+ .data-table td {
297
+ padding: 10px 14px;
298
+ border-bottom: 1px solid var(--border);
299
+ vertical-align: top;
300
+ }
301
+ .data-table tr:last-child td {
302
+ border-bottom: none;
303
+ }
304
+ .data-table tr:hover td {
305
+ background: #fafafa;
306
+ }
307
+ code {
308
+ font-family: "SFMono-Regular", Consolas, "Liberation Mono", monospace;
309
+ font-size: 12px;
310
+ background: var(--bg);
311
+ padding: 1px 6px;
312
+ border-radius: 4px;
313
+ border: 1px solid var(--border);
314
+ }
315
+ .empty-state {
316
+ text-align: center;
317
+ padding: 32px;
318
+ color: var(--text-muted);
319
+ font-size: 13px;
320
+ }
321
+
322
+ /* ── Checklist status ── */
323
+ .status-ok {
324
+ color: var(--ok);
325
+ font-weight: 600;
326
+ }
327
+ .status-ko {
328
+ color: var(--critical);
329
+ font-weight: 600;
330
+ }
331
+ .status-warn {
332
+ color: var(--warning);
333
+ font-weight: 600;
334
+ }
335
+
336
+ /* ── Info grid ── */
337
+ .info-grid {
338
+ display: grid;
339
+ grid-template-columns: 1fr 1fr;
340
+ gap: 12px;
341
+ }
342
+ @media (max-width: 640px) {
343
+ .info-grid {
344
+ grid-template-columns: 1fr;
345
+ }
346
+ }
347
+ .info-item {
348
+ }
349
+ .info-label {
350
+ font-size: 11px;
351
+ font-weight: 600;
352
+ text-transform: uppercase;
353
+ letter-spacing: 0.04em;
354
+ color: var(--text-muted);
355
+ margin-bottom: 2px;
356
+ }
357
+ .info-value {
358
+ font-size: 13px;
359
+ color: var(--text);
360
+ }
361
+
362
+ /* ── Buttons table ── */
363
+ .btn-chip {
364
+ display: inline-block;
365
+ padding: 2px 10px;
366
+ border-radius: 4px;
367
+ font-size: 12px;
368
+ font-weight: 600;
369
+ }
370
+ .btn-chip.accept {
371
+ background: #dcfce7;
372
+ color: #166534;
373
+ }
374
+ .btn-chip.reject {
375
+ background: #fee2e2;
376
+ color: #991b1b;
377
+ }
378
+ .btn-chip.preferences {
379
+ background: #dbeafe;
380
+ color: #1e40af;
381
+ }
382
+ .btn-chip.unknown,
383
+ .btn-chip.close {
384
+ background: var(--bg);
385
+ color: var(--text-muted);
386
+ }
387
+
388
+ /* ── Recommendations ── */
389
+ .rec-list {
390
+ list-style: none;
391
+ margin: 0;
392
+ padding: 0;
393
+ display: flex;
394
+ flex-direction: column;
395
+ gap: 10px;
396
+ }
397
+ .rec-item {
398
+ display: flex;
399
+ gap: 12px;
400
+ padding: 12px 14px;
401
+ background: #f8fafc;
402
+ border-radius: 8px;
403
+ border: 1px solid var(--border);
404
+ font-size: 13px;
405
+ }
406
+ .rec-num {
407
+ flex-shrink: 0;
408
+ width: 24px;
409
+ height: 24px;
410
+ border-radius: 50%;
411
+ background: var(--text);
412
+ color: #fff;
413
+ font-size: 12px;
414
+ font-weight: 700;
415
+ display: flex;
416
+ align-items: center;
417
+ justify-content: center;
418
+ margin-top: 1px;
419
+ }
420
+
421
+ /* ── Footer ── */
422
+ .footer {
423
+ text-align: center;
424
+ margin-top: 40px;
425
+ font-size: 12px;
426
+ color: var(--text-muted);
427
+ }
428
+ .footer a {
429
+ color: var(--text-muted);
430
+ }
431
+
432
+ @media print {
433
+ body {
434
+ background: #fff;
435
+ }
436
+ .section {
437
+ box-shadow: none;
438
+ border: 1px solid var(--border);
439
+ }
440
+ .page {
441
+ padding: 0;
442
+ }
443
+ }
444
+ </style>
445
+ </head>
446
+ <body>
447
+ <div class="page">
448
+ <div class="hero">
449
+ <div class="grade-badge" style="background: #ea580c">D</div>
450
+ <div class="hero-info">
451
+ <h1>arte.tv</h1>
452
+ <p class="hero-meta">Scanned on 24 February 2026 at 15:00 · 14.6s</p>
453
+ </div>
454
+ <div class="hero-score">
455
+ <div>
456
+ <span class="score-num" style="color: #ea580c">53</span
457
+ ><span class="score-den">/100</span>
458
+ </div>
459
+ <div class="score-label">Compliance score</div>
460
+ </div>
461
+ </div>
462
+
463
+ <div class="score-grid">
464
+ <div class="score-card">
465
+ <div class="score-card-label">Consent validity</div>
466
+ <div class="score-card-value">10<span>/25</span></div>
467
+ <div class="progress-track">
468
+ <div class="progress-fill" style="width: 40%; background: #ea580c"></div>
469
+ </div>
470
+ </div>
471
+ <div class="score-card">
472
+ <div class="score-card-label">Easy refusal</div>
473
+ <div class="score-card-value">25<span>/25</span></div>
474
+ <div class="progress-track">
475
+ <div class="progress-fill" style="width: 100%; background: #16a34a"></div>
476
+ </div>
477
+ </div>
478
+ <div class="score-card">
479
+ <div class="score-card-label">Transparency</div>
480
+ <div class="score-card-value">0<span>/25</span></div>
481
+ <div class="progress-track">
482
+ <div class="progress-fill" style="width: 0%; background: #dc2626"></div>
483
+ </div>
484
+ </div>
485
+ <div class="score-card">
486
+ <div class="score-card-label">Cookie behavior</div>
487
+ <div class="score-card-value">18<span>/25</span></div>
488
+ <div class="progress-track">
489
+ <div class="progress-fill" style="width: 72%; background: #ca8a04"></div>
490
+ </div>
491
+ </div>
492
+ </div>
493
+
494
+ <div class="section">
495
+ <div class="section-header">
496
+ <h2>Issues</h2>
497
+ <span class="count-badge">7</span>
498
+ <span class="badge badge-critical">2 critical</span>
499
+ <span class="badge badge-warning">5 warnings</span>
500
+ </div>
501
+ <div class="section-body">
502
+ <div class="issue-list">
503
+ <div class="issue-card critical">
504
+ <div class="issue-title">
505
+ 1 non-essential cookie(s) deposited before any interaction
506
+ </div>
507
+ <p class="issue-evidence">ABV (unknown)</p>
508
+ </div>
509
+ <div class="issue-card critical">
510
+ <div class="issue-title">1 non-essential cookie(s) persist after rejection</div>
511
+ <p class="issue-evidence">ABV (unknown)</p>
512
+ </div>
513
+ <div class="issue-card warning">
514
+ <div class="issue-title">Missing required information: &quot;purposes&quot;</div>
515
+ <p class="issue-evidence">The consent text does not mention purposes</p>
516
+ </div>
517
+ <div class="issue-card warning">
518
+ <div class="issue-title">Missing required information: &quot;third-parties&quot;</div>
519
+ <p class="issue-evidence">The consent text does not mention third-parties</p>
520
+ </div>
521
+ <div class="issue-card warning">
522
+ <div class="issue-title">Missing required information: &quot;duration&quot;</div>
523
+ <p class="issue-evidence">The consent text does not mention duration</p>
524
+ </div>
525
+ <div class="issue-card warning">
526
+ <div class="issue-title">Missing required information: &quot;withdrawal&quot;</div>
527
+ <p class="issue-evidence">The consent text does not mention withdrawal</p>
528
+ </div>
529
+ <div class="issue-card warning">
530
+ <div class="issue-title">No privacy policy link found in the consent modal</div>
531
+ <p class="issue-evidence">
532
+ GDPR Art. 13 requires the privacy policy to be accessible from the consent interface
533
+ </p>
534
+ </div>
535
+ </div>
536
+ </div>
537
+ </div>
538
+
539
+ <div class="section">
540
+ <div class="section-header">
541
+ <h2>Consent modal</h2>
542
+ <span class="badge badge-ok">Detected</span>
543
+ </div>
544
+ <div class="section-body">
545
+ <img src="modal-initial.png" alt="Consent modal screenshot" class="modal-screenshot" />
546
+ <div class="info-grid" style="margin-bottom: 20px">
547
+ <div class="info-item">
548
+ <div class="info-label">Selector</div>
549
+ <div class="info-value"><code>div.ds-eg90wg</code></div>
550
+ </div>
551
+ <div class="info-item">
552
+ <div class="info-label">Granular controls</div>
553
+ <div class="info-value"><span class="status-warn">✗ No</span></div>
554
+ </div>
555
+ <div class="info-item">
556
+ <div class="info-label">Privacy policy link</div>
557
+ <div class="info-value"><span class="badge badge-warning">Not found</span></div>
558
+ </div>
559
+ <div class="info-item">
560
+ <div class="info-label">Pre-ticked checkboxes</div>
561
+ <div class="info-value"><span class="status-ok">✓ None</span></div>
562
+ </div>
563
+ </div>
564
+ <div
565
+ style="
566
+ font-size: 12px;
567
+ font-weight: 600;
568
+ text-transform: uppercase;
569
+ letter-spacing: 0.04em;
570
+ color: var(--text-muted);
571
+ margin-bottom: 8px;
572
+ "
573
+ >
574
+ Buttons
575
+ </div>
576
+ <table class="data-table">
577
+ <thead>
578
+ <tr>
579
+ <th>Type</th>
580
+ <th>Label</th>
581
+ <th>Font size</th>
582
+ <th>Contrast</th>
583
+ <th>Clicks</th>
584
+ </tr>
585
+ </thead>
586
+ <tbody>
587
+ <tr>
588
+ <td><span class="btn-chip accept">accept</span></td>
589
+ <td>J'accepte</td>
590
+ <td>16px</td>
591
+ <td>21:1</td>
592
+ <td>1</td>
593
+ </tr>
594
+ <tr>
595
+ <td><span class="btn-chip reject">reject</span></td>
596
+ <td>Je refuse</td>
597
+ <td>16px</td>
598
+ <td>21:1</td>
599
+ <td>1</td>
600
+ </tr>
601
+ <tr>
602
+ <td><span class="btn-chip unknown">unknown</span></td>
603
+ <td>Paramétrer mes choix</td>
604
+ <td>16px</td>
605
+ <td>21:1</td>
606
+ <td>1</td>
607
+ </tr>
608
+ </tbody>
609
+ </table>
610
+ </div>
611
+ </div>
612
+
613
+ <div class="section">
614
+ <div class="section-header"><h2>Cookies</h2></div>
615
+ <div class="section-body">
616
+ <div style="margin-bottom: 24px">
617
+ <div
618
+ style="
619
+ font-size: 12px;
620
+ font-weight: 600;
621
+ text-transform: uppercase;
622
+ letter-spacing: 0.04em;
623
+ color: var(--text-muted);
624
+ margin-bottom: 8px;
625
+ display: flex;
626
+ align-items: center;
627
+ gap: 8px;
628
+ "
629
+ >
630
+ Before interaction
631
+ <span class="count-badge">4</span>
632
+ <span class="badge badge-critical">1 non-essential</span>
633
+ </div>
634
+ <table class="data-table">
635
+ <thead>
636
+ <tr>
637
+ <th>Name</th>
638
+ <th>Domain</th>
639
+ <th>Category</th>
640
+ <th>Expiry</th>
641
+ <th>Consent</th>
642
+ </tr>
643
+ </thead>
644
+ <tbody>
645
+ <tr>
646
+ <td><code>AUTH_SESSION_ID</code></td>
647
+ <td style="color: var(--text-muted)">id.arte.tv</td>
648
+ <td><span class="badge badge-muted">strictly-necessary</span></td>
649
+ <td style="color: var(--text-muted)">Session</td>
650
+ <td><span class="badge badge-muted">No</span></td>
651
+ </tr>
652
+ <tr>
653
+ <td><code>KC_AUTH_SESSION_HASH</code></td>
654
+ <td style="color: var(--text-muted)">id.arte.tv</td>
655
+ <td><span class="badge badge-muted">unknown</span></td>
656
+ <td style="color: var(--text-muted)">< 1 day</td>
657
+ <td><span class="badge badge-muted">No</span></td>
658
+ </tr>
659
+ <tr>
660
+ <td><code>ABV</code></td>
661
+ <td style="color: var(--text-muted)">.arte.tv</td>
662
+ <td><span class="badge badge-muted">unknown</span></td>
663
+ <td style="color: var(--text-muted)">Session</td>
664
+ <td><span class="badge badge-warning">Required</span></td>
665
+ </tr>
666
+ <tr>
667
+ <td><code>validated-age</code></td>
668
+ <td style="color: var(--text-muted)">www.arte.tv</td>
669
+ <td><span class="badge badge-muted">unknown</span></td>
670
+ <td style="color: var(--text-muted)">12mo</td>
671
+ <td><span class="badge badge-muted">No</span></td>
672
+ </tr>
673
+ </tbody>
674
+ </table>
675
+ </div>
676
+ <div style="margin-bottom: 24px">
677
+ <div
678
+ style="
679
+ font-size: 12px;
680
+ font-weight: 600;
681
+ text-transform: uppercase;
682
+ letter-spacing: 0.04em;
683
+ color: var(--text-muted);
684
+ margin-bottom: 8px;
685
+ display: flex;
686
+ align-items: center;
687
+ gap: 8px;
688
+ "
689
+ >
690
+ After reject
691
+ <span class="count-badge">5</span>
692
+
693
+ <span class="badge badge-critical">1 non-essential</span>
694
+ </div>
695
+ <table class="data-table">
696
+ <thead>
697
+ <tr>
698
+ <th>Name</th>
699
+ <th>Domain</th>
700
+ <th>Category</th>
701
+ <th>Expiry</th>
702
+ <th>Consent</th>
703
+ </tr>
704
+ </thead>
705
+ <tbody>
706
+ <tr>
707
+ <td><code>AUTH_SESSION_ID</code></td>
708
+ <td style="color: var(--text-muted)">id.arte.tv</td>
709
+ <td><span class="badge badge-muted">strictly-necessary</span></td>
710
+ <td style="color: var(--text-muted)">Session</td>
711
+ <td><span class="badge badge-muted">No</span></td>
712
+ </tr>
713
+ <tr>
714
+ <td><code>KC_AUTH_SESSION_HASH</code></td>
715
+ <td style="color: var(--text-muted)">id.arte.tv</td>
716
+ <td><span class="badge badge-muted">unknown</span></td>
717
+ <td style="color: var(--text-muted)">< 1 day</td>
718
+ <td><span class="badge badge-muted">No</span></td>
719
+ </tr>
720
+ <tr>
721
+ <td><code>ABV</code></td>
722
+ <td style="color: var(--text-muted)">.arte.tv</td>
723
+ <td><span class="badge badge-muted">unknown</span></td>
724
+ <td style="color: var(--text-muted)">Session</td>
725
+ <td><span class="badge badge-warning">Required</span></td>
726
+ </tr>
727
+ <tr>
728
+ <td><code>validated-age</code></td>
729
+ <td style="color: var(--text-muted)">www.arte.tv</td>
730
+ <td><span class="badge badge-muted">unknown</span></td>
731
+ <td style="color: var(--text-muted)">12mo</td>
732
+ <td><span class="badge badge-muted">No</span></td>
733
+ </tr>
734
+ <tr>
735
+ <td><code>user_consent</code></td>
736
+ <td style="color: var(--text-muted)">www.arte.tv</td>
737
+ <td><span class="badge badge-muted">unknown</span></td>
738
+ <td style="color: var(--text-muted)">12mo</td>
739
+ <td><span class="badge badge-muted">No</span></td>
740
+ </tr>
741
+ </tbody>
742
+ </table>
743
+ </div>
744
+ <div style="margin-bottom: 24px">
745
+ <div
746
+ style="
747
+ font-size: 12px;
748
+ font-weight: 600;
749
+ text-transform: uppercase;
750
+ letter-spacing: 0.04em;
751
+ color: var(--text-muted);
752
+ margin-bottom: 8px;
753
+ display: flex;
754
+ align-items: center;
755
+ gap: 8px;
756
+ "
757
+ >
758
+ After accept
759
+ <span class="count-badge">5</span>
760
+ </div>
761
+ <table class="data-table">
762
+ <thead>
763
+ <tr>
764
+ <th>Name</th>
765
+ <th>Domain</th>
766
+ <th>Category</th>
767
+ <th>Expiry</th>
768
+ <th>Consent</th>
769
+ </tr>
770
+ </thead>
771
+ <tbody>
772
+ <tr>
773
+ <td><code>AUTH_SESSION_ID</code></td>
774
+ <td style="color: var(--text-muted)">id.arte.tv</td>
775
+ <td><span class="badge badge-muted">strictly-necessary</span></td>
776
+ <td style="color: var(--text-muted)">Session</td>
777
+ <td><span class="badge badge-muted">No</span></td>
778
+ </tr>
779
+ <tr>
780
+ <td><code>KC_AUTH_SESSION_HASH</code></td>
781
+ <td style="color: var(--text-muted)">id.arte.tv</td>
782
+ <td><span class="badge badge-muted">unknown</span></td>
783
+ <td style="color: var(--text-muted)">< 1 day</td>
784
+ <td><span class="badge badge-muted">No</span></td>
785
+ </tr>
786
+ <tr>
787
+ <td><code>ABV</code></td>
788
+ <td style="color: var(--text-muted)">.arte.tv</td>
789
+ <td><span class="badge badge-muted">unknown</span></td>
790
+ <td style="color: var(--text-muted)">Session</td>
791
+ <td><span class="badge badge-warning">Required</span></td>
792
+ </tr>
793
+ <tr>
794
+ <td><code>validated-age</code></td>
795
+ <td style="color: var(--text-muted)">www.arte.tv</td>
796
+ <td><span class="badge badge-muted">unknown</span></td>
797
+ <td style="color: var(--text-muted)">12mo</td>
798
+ <td><span class="badge badge-muted">No</span></td>
799
+ </tr>
800
+ <tr>
801
+ <td><code>user_consent</code></td>
802
+ <td style="color: var(--text-muted)">www.arte.tv</td>
803
+ <td><span class="badge badge-muted">unknown</span></td>
804
+ <td style="color: var(--text-muted)">12mo</td>
805
+ <td><span class="badge badge-muted">No</span></td>
806
+ </tr>
807
+ </tbody>
808
+ </table>
809
+ </div>
810
+ </div>
811
+ </div>
812
+
813
+ <div class="section">
814
+ <div class="section-header">
815
+ <h2>Network trackers</h2>
816
+ <span class="count-badge">2</span>
817
+ </div>
818
+ <div class="section-body no-pad">
819
+ <table class="data-table">
820
+ <thead>
821
+ <tr>
822
+ <th>Tracker</th>
823
+ <th>Category</th>
824
+ <th>Phase</th>
825
+ <th>URL</th>
826
+ </tr>
827
+ </thead>
828
+ <tbody>
829
+ <tr>
830
+ <td>Tracking Pixel</td>
831
+ <td><span class="badge badge-muted">pixel</span></td>
832
+ <td><span class="badge badge-muted">after-reject</span></td>
833
+ <td style="font-size: 11px; color: var(--text-muted); word-break: break-all">
834
+ <code>https://event.arte.tv/api/server-side-tracking/v1/tracking</code>
835
+ </td>
836
+ </tr>
837
+ <tr>
838
+ <td>Tracking Pixel</td>
839
+ <td><span class="badge badge-muted">pixel</span></td>
840
+ <td><span class="badge badge-muted">after-accept</span></td>
841
+ <td style="font-size: 11px; color: var(--text-muted); word-break: break-all">
842
+ <code>https://event.arte.tv/api/server-side-tracking/v1/tracking</code>
843
+ </td>
844
+ </tr>
845
+ </tbody>
846
+ </table>
847
+ </div>
848
+ </div>
849
+
850
+ <div class="section">
851
+ <div class="section-header">
852
+ <h2>Recommendations</h2>
853
+ <span class="count-badge">3</span>
854
+ </div>
855
+ <div class="section-body">
856
+ <ul class="rec-list">
857
+ <li class="rec-item">
858
+ <span class="rec-num">1</span>
859
+ <span
860
+ >Do not set any non-essential cookie before consent. Gate the initialisation of
861
+ third-party scripts on the acceptance callback.</span
862
+ >
863
+ </li>
864
+ <li class="rec-item">
865
+ <span class="rec-num">2</span>
866
+ <span
867
+ >Complete the modal information: processing purposes, identity of sub-processors,
868
+ retention period, right to withdraw.</span
869
+ >
870
+ </li>
871
+ <li class="rec-item">
872
+ <span class="rec-num">3</span>
873
+ <span
874
+ >Remove or block non-essential cookies after rejection, and verify consent handling
875
+ server-side.</span
876
+ >
877
+ </li>
878
+ </ul>
879
+ </div>
880
+ </div>
881
+
882
+ <div class="section">
883
+ <div class="section-header">
884
+ <h2>Compliance checklist</h2>
885
+ <span class="count-badge">12 rules</span>
886
+ <span class="badge badge-ok">8 ✓</span>
887
+ <span class="badge badge-critical">2 ✗</span>
888
+ <span class="badge badge-warning">2 ⚠</span>
889
+ </div>
890
+ <div class="section-body no-pad">
891
+ <table class="data-table">
892
+ <thead>
893
+ <tr>
894
+ <th>Category</th>
895
+ <th>Rule</th>
896
+ <th>Status</th>
897
+ <th>Detail</th>
898
+ </tr>
899
+ </thead>
900
+ <tbody>
901
+ <tr>
902
+ <td
903
+ rowspan="3"
904
+ style="font-weight: 600; vertical-align: top; background: var(--bg)"
905
+ >
906
+ Consent
907
+ </td>
908
+ <td>Consent modal detected</td>
909
+ <td><span class="status-ok">✓ Compliant</span></td>
910
+ <td style="color: var(--text-muted); font-size: 12px">Detected (div.ds-eg90wg)</td>
911
+ </tr>
912
+ <tr>
913
+ <td>No pre-ticked checkboxes</td>
914
+ <td><span class="status-ok">✓ Compliant</span></td>
915
+ <td style="color: var(--text-muted); font-size: 12px">None detected</td>
916
+ </tr>
917
+ <tr>
918
+ <td>Unambiguous accept label</td>
919
+ <td><span class="status-ok">✓ Compliant</span></td>
920
+ <td style="color: var(--text-muted); font-size: 12px">J'accepte</td>
921
+ </tr>
922
+ <tr>
923
+ <td
924
+ rowspan="3"
925
+ style="font-weight: 600; vertical-align: top; background: var(--bg)"
926
+ >
927
+ Easy refusal
928
+ </td>
929
+ <td>Reject button at first layer</td>
930
+ <td><span class="status-ok">✓ Compliant</span></td>
931
+ <td style="color: var(--text-muted); font-size: 12px">Je refuse</td>
932
+ </tr>
933
+ <tr>
934
+ <td>Reject ≤ clicks than accept</td>
935
+ <td><span class="status-ok">✓ Compliant</span></td>
936
+ <td style="color: var(--text-muted); font-size: 12px">Accept: 1 · Reject: 1</td>
937
+ </tr>
938
+ <tr>
939
+ <td>Button size symmetry</td>
940
+ <td><span class="status-ok">✓ Compliant</span></td>
941
+ <td style="color: var(--text-muted); font-size: 12px">Comparable sizes</td>
942
+ </tr>
943
+ <tr>
944
+ <td
945
+ rowspan="3"
946
+ style="font-weight: 600; vertical-align: top; background: var(--bg)"
947
+ >
948
+ Transparency
949
+ </td>
950
+ <td>Granular controls available</td>
951
+ <td><span class="status-warn">⚠ Warning</span></td>
952
+ <td style="color: var(--text-muted); font-size: 12px">No granular controls</td>
953
+ </tr>
954
+ <tr>
955
+ <td>Privacy policy in modal</td>
956
+ <td><span class="status-warn">⚠ Warning</span></td>
957
+ <td style="color: var(--text-muted); font-size: 12px">Not found</td>
958
+ </tr>
959
+ <tr>
960
+ <td>Privacy policy on page</td>
961
+ <td><span class="status-ok">✓ Compliant</span></td>
962
+ <td style="color: var(--text-muted); font-size: 12px">
963
+ https://www.arte.tv/sites/corporate/credits/
964
+ </td>
965
+ </tr>
966
+ <tr>
967
+ <td
968
+ rowspan="3"
969
+ style="font-weight: 600; vertical-align: top; background: var(--bg)"
970
+ >
971
+ Cookie behavior
972
+ </td>
973
+ <td>No non-essential cookie before consent</td>
974
+ <td><span class="status-ko">✗ Non-compliant</span></td>
975
+ <td style="color: var(--text-muted); font-size: 12px">1: ABV</td>
976
+ </tr>
977
+ <tr>
978
+ <td>Non-essential cookies removed after reject</td>
979
+ <td><span class="status-ko">✗ Non-compliant</span></td>
980
+ <td style="color: var(--text-muted); font-size: 12px">1 persisting</td>
981
+ </tr>
982
+ <tr>
983
+ <td>No tracker before consent</td>
984
+ <td><span class="status-ok">✓ Compliant</span></td>
985
+ <td style="color: var(--text-muted); font-size: 12px">None</td>
986
+ </tr>
987
+ </tbody>
988
+ </table>
989
+ </div>
990
+ </div>
991
+
992
+ <div class="footer">
993
+ Generated by <a href="https://github.com/Slashgear/gdpr-report">gdpr-cookie-scanner</a>
994
+ on 24 February 2026 at 15:00
995
+ </div>
996
+ </div>
997
+ </body>
998
+ </html>