Expand description of article 173

This commit is contained in:
Kuba Orlik 2022-02-10 17:07:47 +01:00
parent 5cf640e686
commit cc881fddbe
5 changed files with 641 additions and 508 deletions

View File

@ -1,5 +1,5 @@
trailingComma: "es5" trailingComma: "es5"
tabWidth: 4 tabWidth: 3
printWidth: 100 printWidth: 100
semi: true semi: true
singleQuote: true singleQuote: true

View File

@ -4,140 +4,248 @@ import { ParsedAnswers } from './parse-answers';
import { v } from './verbs'; import { v } from './verbs';
abstract class Problem { abstract class Problem {
constructor(public answers: ParsedAnswers, public clusters: Record<string, RequestCluster>) {} constructor(public answers: ParsedAnswers, public clusters: Record<string, RequestCluster>) {}
getMarkedClusters() { getMarkedClusters() {
return Object.values(this.clusters).filter((c) => c.hasMarks()); return Object.values(this.clusters).filter((c) => c.hasMarks());
} }
abstract getEmailContent(): JSX.Element; abstract getEmailContent(): JSX.Element;
abstract getNecessaryExplainers(): ExplainerKey[]; abstract getNecessaryExplainers(): ExplainerKey[];
} }
function formatRange(cluster: RequestCluster) { function formatRange(cluster: RequestCluster) {
const parts = [] as string[]; const parts = [] as string[];
console.log(cluster); console.log(cluster);
if (cluster.hasMarkedCookies()) { if (cluster.hasMarkedCookies()) {
parts.push('mojego identyfikatora internetowego pozyskanego z Cookie'); parts.push('mojego identyfikatora internetowego pozyskanego z Cookie');
} }
if (cluster.exposesOrigin()) { if (cluster.exposesOrigin()) {
parts.push('części mojej historii przeglądania'); parts.push('części mojej historii przeglądania');
} }
return parts.join(' oraz '); return parts.join(' oraz ');
} }
class NoInformationAtAllProblem extends Problem { class NoInformationAtAllProblem extends Problem {
getEmailContent() { getEmailContent() {
const _ = (word: string) => v(word, this.answers.zaimek); const _ = (word: string) => v(word, this.answers.zaimek);
return ( return (
<> <>
<h2>Brak informacji na temat przetwarzania danych osobowych</h2> <h2>Brak informacji na temat przetwarzania danych osobowych</h2>
<p> <p>
{_('Moje')} dane osobowe zostały ujawnione podmiotom, które właścicielami {_('Moje')} dane osobowe zostały ujawnione podmiotom, które właścicielami domen:
domen: </p>
</p> <ul>
<ul> {this.getMarkedClusters().map((cluster) => (
{this.getMarkedClusters().map((cluster) => ( <li key={cluster.id}>
<li key={cluster.id}> {cluster.id} (w zakresie: {formatRange(cluster)})
{cluster.id} (w zakresie: {formatRange(cluster)}) </li>
</li> ))}
))} </ul>
</ul> <p>
<p> Na stronie brakuje jednak jakichkolwiek informacji o tym, jakie cele przetwarzania
Na stronie brakuje jednak jakichkolwiek informacji o tym, jakie cele takich danych oraz jakie podstawy prawne takiego przetwarzania.
przetwarzania takich danych oraz jakie podstawy prawne takiego przetwarzania. </p>
</p> <p>Zwracam się zatem do Państwa z następującymi pytaniami:</p>
<p>Zwracam się zatem do Państwa z następującymi pytaniami:</p> <ul>
<ul> <li>Jaka jest tożsamość właścicieli tych domen?</li>
<li>Jaka jest tożsamość właścicieli tych domen?</li> <li>Jaki jest cel takiego przetwarzania danych przez Państwa stronę?</li>
<li>Jaki jest cel takiego przetwarzania danych przez Państwa stronę?</li> <li>
<li> Jaka jest podstawa prawna takiego przetwarzania moich danych osobowych przez
Jaka jest podstawa prawna takiego przetwarzania moich danych osobowych przez Państwa stronę?
Państwa stronę? </li>
</li> </ul>
</ul> </>
</> );
); }
} getNecessaryExplainers() {
getNecessaryExplainers() { const explainers = [] as Array<ExplainerKey>;
const explainers = [] as Array<ExplainerKey>;
if ( if (
this.getMarkedClusters().some((cluster) => { this.getMarkedClusters().some((cluster) => {
console.log(cluster); console.log(cluster);
return cluster.hasMarkedCookies(); return cluster.hasMarkedCookies();
}) })
) { ) {
explainers.push('cookies_are_pii'); explainers.push('cookies_are_pii');
} }
return explainers; return explainers;
} }
} }
class UnlawfulCookieAccess extends Problem { class UnlawfulCookieAccess extends Problem {
getNecessaryExplainers(): ExplainerKey[] { getNecessaryExplainers(): ExplainerKey[] {
return []; return [];
} }
getEmailContent() { getEmailContent() {
const cookie_clusters = Object.values(this.clusters).filter((c) => c.hasMarkedCookies()); const cookie_clusters = Object.values(this.clusters).filter((c) => c.hasMarkedCookies());
return ( const unnecessary_hosts = Object.entries(this.answers.hosts)
<> .filter(([, answers]) => answers.was_processing_necessary === 'no')
<h2>Dostęp do cookies niezgodny z ustawą Prawo Telekomunikacyjne</h2> .map(([host]) => host);
<p> const maybe_unnecessary_hosts = Object.entries(this.answers.hosts)
Państwa strona dokonała odczytu plików Cookie zapisanych na dysku twardym mojego .filter(([, answers]) => answers.was_processing_necessary === 'not_sure')
komputera. Dotyczy to plików cookie przypisanych do domen: .map(([host]) => host);
</p> const _ = (key: string) => v(key, this.answers.zaimek);
<ul> return (
{cookie_clusters.map((cluster, index) => { <>
const names = cluster <h2>Dostęp do cookies niezgodny z ustawą Prawo Telekomunikacyjne</h2>
.getMarkedEntries() <p>
.filter((e) => e.source === 'cookie') Państwa strona dokonała odczytu plików Cookie zapisanych na dysku twardym mojego
.map((e) => e.name); komputera. Dotyczy to plików cookie przypisanych do domen:
</p>
<ul>
{cookie_clusters.map((cluster, index) => {
const names = cluster
.getMarkedEntries()
.filter((e) => e.source === 'cookie')
.map((e) => e.name);
return ( return (
<li> <li>
{cluster.id} ({names.length > 1 ? 'pliki' : 'plik'}{' '} {cluster.id} ({names.length > 1 ? 'pliki' : 'plik'}{' '}
{names.map((name, index) => { {names.map((name, index) => {
return ( return (
<> <>
{index > 0 ? ', ' : ''} {index > 0 ? ', ' : ''}
{name} {name}
</> </>
); );
})} })}
){index === cookie_clusters.length - 1 ? '.' : ';'} ){index === cookie_clusters.length - 1 ? '.' : ';'}
</li> </li>
); );
})} })}
</ul> </ul>
</> <p>
); Zgodnie z treścią Art. 173.{' '}
} <a href="https://isap.sejm.gov.pl/isap.nsf/download.xsp/WDU20041711800/U/D20041800Lj.pdf">
static qualifies(answers: ParsedAnswers, clusters: RequestCluster[]): boolean { ustawy Prawo Telekomunikacyjne
// są cookiesy, nie było zgody, nie są konieczne do działania strony </a>
const cookie_clusters = Object.values(clusters).filter((c) => c.hasMarkedCookies()); , strona może pozyskać dostęp do treści plików cookies pod warunkiem spełnienia
return cookie_clusters.some((cluster) => { jednego z następujących warunków:
const hostAnswers = answers.hosts[cluster.id]; </p>
return ( <ol>
(hostAnswers.present == 'not_mentioned' || <li>
hostAnswers.present == 'not_before_making_a_choice' || Użytkownik wyraził zgodę na takie przetwarzanie danych <em>po</em> tym, jak został
['none', 'closed_popup', 'deny_all'].includes(hostAnswers.popup_action)) && poinformowany bezpośrednio o celu uzyskania dostępu do tej informacji;
hostAnswers.was_processing_necessary != 'yes' </li>
); <li>
}); Dostęp do treści plików cookies jest konieczny do dostarczania usługi świadczonej
} drogą elektroniczną zażądanej przez użytkownika.
</li>
</ol>
{(() => {
if (this.answers.popup_type == 'none' || this.answers.popup_type == 'page') {
return (
<p>
Jako, że strona nie pytała {_('mnie')} nigdy o zgodę, nie jest spełniony
warunek 1.
</p>
);
} else if (this.answers.popup_type === 'passive_popup') {
return (
<p>
Państwa strona nie dała mi nigdy faktycznego wyboru dotyczacego wyrażenia
lub odmówienia zgody na takie przetwarzanie danych osobowych, dlatego nie
jest spełniony warunek 1.{' '}
{this.answers.mentions_passive_consent ? (
<>
Zgody wyrażonej w sposób bierny lub milczący nie można uznać za ważną
w świetle obowiązujących przepisów rozporządzenia 2016/679. Dlatego
zaniechanie zmiany ustawień przeglądarki lub po prostu korzystanie ze
strony nie stanowi ważnej zgody. Takie jest{' '}
<a href="https://assets.midline.pl/pisma/2021-12-16%20odpowiedz%20UODO%20na%20skarg%C4%99%20i(n)Secure.pdf">
stanowisko polskiego UODO
</a>
.
</>
) : (
''
)}
</p>
);
} else if (this.answers.popup_type === 'some_choice') {
if (this.answers.popup_action === 'none') {
return (
<p>
Nie {_('wyraziłem')} zgody na takie przetwarzanie {_('moich')} danych
osobowych. W okienku pytającym o zgodję nic nie {_('kliknąłem')}. Nie
jest zatem spełniony warunek 1.
</p>
);
} else if (this.answers.popup_action === 'closed_popup') {
return (
<p>
Nie {_('wyraziłem')} zgody na takie przetwarzanie {_('moich')} danych
osobowych. {this.answers.popup_closed_how.trim()}
{this.answers.popup_closed_how.trim().at(-1) != '.' ? '.' : ''} Takiego
działania nie można uznać za ważną zgodę na przetwarzanie danych
osobowych, gdyż nie spełnia warunku jednoznaczności opisanego w Art. 4,
pkt 11 RODO. Nie jest zatem spełniony warunek 1.
</p>
);
} else if (this.answers.popup_action == 'deny_all') {
return (
<p>
{this.answers.popup_deny_all_how.trim()}
{this.answers.popup_closed_how.trim().at(-1) != '.' ? '.' : ''} Zatem nie
jest spełniony warunek 1.
</p>
);
}
}
})()}
{unnecessary_hosts.length > 0 ? (
<p>
W {_('mojej')} ocenie odczytywanie przez Państwa stronę treści plików cookies z{' '}
{unnecessary_hosts.join(', ')} nie jest konieczne do wyświetlenia treści Państwa
strony, dlatego nie jest dla nich spełniony warunek 2. Jeżeli Państwa zdaniem jest
inaczej, {_('proszę')} o wskazanie, co jest źródłem tej konieczności i co odróżnia
Państwa stronę od wielu innych stron, które realizują te same funkcjonalności{' '}
<em>bez</em> korzystania z plików Cookie.
</p>
) : (
''
)}
<p>
{_('Proszę')} o wskazanie, czy być może stosowali Państwo inną podstawę prawną do
takiego przetwarzania {_('moich')} danych osobowych, czy przetwarzali je państwo bez
ważnej podstawy prawnej?
</p>
<p>
{_('Proszę')} też o wskazanie, czy dostęp do treści plików cookie z $
{maybe_unnecessary_hosts.join(', ')} jest konieczny do poprawnego działania strony?
Jeżeli tak, to {_('proszę')} wskazać, w jaki sposób. Co sprawia, że strona nie może
działać bez nich?
</p>
</>
);
}
static qualifies(answers: ParsedAnswers, clusters: RequestCluster[]): boolean {
// są cookiesy, nie było zgody, nie są konieczne do działania strony
const cookie_clusters = Object.values(clusters).filter((c) => c.hasMarkedCookies());
return cookie_clusters.some((cluster) => {
const hostAnswers = answers.hosts[cluster.id];
return (
(hostAnswers.present == 'not_mentioned' ||
hostAnswers.present == 'not_before_making_a_choice' ||
['none', 'closed_popup', 'deny_all'].includes(hostAnswers.popup_action)) &&
hostAnswers.was_processing_necessary != 'yes'
);
});
}
} }
export default function deduceProblems( export default function deduceProblems(
answers: ParsedAnswers, answers: ParsedAnswers,
clusters: Record<string, RequestCluster> clusters: Record<string, RequestCluster>
): Problem[] { ): Problem[] {
const problems = []; const problems = [];
if (answers.popup_type === 'none') { if (answers.popup_type === 'none') {
problems.push(new NoInformationAtAllProblem(answers, clusters)); problems.push(new NoInformationAtAllProblem(answers, clusters));
} }
if (UnlawfulCookieAccess.qualifies(answers, Object.values(clusters))) { if (UnlawfulCookieAccess.qualifies(answers, Object.values(clusters))) {
problems.push(new UnlawfulCookieAccess(answers, clusters)); problems.push(new UnlawfulCookieAccess(answers, clusters));
} }
return problems; return problems;
} }

View File

@ -1,366 +1,381 @@
function generateHostPage( function generateHostPage(
host: string, host: string,
index: number, index: number,
all_hosts: string[] all_hosts: string[]
): { title: string; elements: any[]; visibleIf: string } { ): { title: string; elements: any[]; visibleIf: string } {
function f(name: string, h = host) { function f(name: string, h = host) {
return `${h.replace(/\./g, '_')}|${name}`; return `${h.replace(/\./g, '_')}|${name}`;
} }
const previous_host: string | null = index > 0 ? all_hosts[index - 1] : null; const previous_host: string | null = index > 0 ? all_hosts[index - 1] : null;
function defaultValue(name: string) { function defaultValue(name: string) {
if (!previous_host) { if (!previous_host) {
return {}; return {};
} }
return { defaultValueExpression: `{${f(name, previous_host)}}` }; return { defaultValueExpression: `{${f(name, previous_host)}}` };
} }
return { return {
title: host, title: host,
visibleIf: "{popup_type} != 'none'", elements: [
elements: [ {
{ type: 'radiogroup',
type: 'radiogroup', name: f('present'),
name: f('present'), isRequired: true,
isRequired: true, title: `Cel ujawnienia danych właścicielowi domeny ${host}`,
title: `Cel ujawnienia danych właścicielowi domeny ${host}`, ...defaultValue('present'),
...defaultValue('present'), choices: [
choices: [ {
{ value: 'not_mentioned',
value: 'not_mentioned', text: 'nie jest podany nigdzie na stronie',
text: 'nie jest podany nigdzie na stronie', visibleIf: "{policy_readable} = 'yes' ",
visibleIf: "{policy_readable} = 'yes' ", },
}, {
{ value: 'not_before_making_a_choice',
value: 'not_before_making_a_choice', text: 'nie jest podany w żadnym miejscu na stronie, do którego można się dostać bez podejmowania wyboru dotyczącego przetwarzania danych osobowych',
text: 'nie jest podany w żadnym miejscu na stronie, do którego można się dostać bez podejmowania wyboru dotyczącego przetwarzania danych osobowych', },
}, {
{ value: 'mentioned_in_policy',
value: 'mentioned_in_policy', text: 'jest podany w polityce prywatności',
text: 'jest podany w polityce prywatności', visibleIf: "{policy_readable} = 'yes' ",
visibleIf: "{policy_readable} = 'yes' ", },
},
{ {
value: 'mentioned_in_popup', value: 'mentioned_in_popup',
text: 'jest podany w okienku RODO', text: 'jest podany w okienku RODO',
visibleIf: "{popup_type} != 'none' ", visibleIf: "{popup_type} != 'none' ",
}, },
], ],
}, },
{ {
type: 'radiogroup', type: 'radiogroup',
name: f('legal_basis_type'), name: f('legal_basis_type'),
...defaultValue('legal_basis_type'), ...defaultValue('legal_basis_type'),
isRequired: true, isRequired: true,
title: `Podstawa prawna dla tego konkretnego celu`, title: `Podstawa prawna dla tego konkretnego celu`,
visibleIf: `{${f('present')}} notempty and {${f( visibleIf: `{${f('present')}} notempty and {${f(
'present' 'present'
)}} != "not_mentioned" and {${f('present')}} != "not_before_making_a_choice"`, )}} != "not_mentioned" and {${f('present')}} != "not_before_making_a_choice"`,
choices: [ choices: [
{ value: 'consent', text: 'to zgoda.' }, { value: 'consent', text: 'to zgoda.' },
{ {
value: 'legitimate_interest', value: 'legitimate_interest',
text: 'to uzasadniony interes.', text: 'to uzasadniony interes.',
}, },
{ value: 'not_mentioned', text: 'nie jest wskazana nigdzie na stronie.' }, { value: 'not_mentioned', text: 'nie jest wskazana nigdzie na stronie.' },
], ],
}, },
{ {
type: 'radiogroup', type: 'radiogroup',
name: f('consent_problems'), name: f('consent_problems'),
...defaultValue('consent_problems'), ...defaultValue('consent_problems'),
isRequired: true, isRequired: true,
title: `Jak ma się ta podstawa prawna do stanu faktycznego?`, title: `Jak ma się ta podstawa prawna do stanu faktycznego?`,
visibleIf: `{${f('legal_basis_type')}} = "consent"`, visibleIf: `{${f('legal_basis_type')}} = "consent"`,
defaultValueExpression: defaultValueExpression:
'iif({popup_action} = "none" or {popup_action} = "closed_popup", "claims_consent_but_sends_before_consent", iif({popup_action} = "accept_all" and {rejection_is_hard} = "yes", "claims_consent_but_there_was_no_easy_refuse", ""))', 'iif({popup_action} = "none" or {popup_action} = "closed_popup", "claims_consent_but_sends_before_consent", iif({popup_action} = "accept_all" and {rejection_is_hard} = "yes", "claims_consent_but_there_was_no_easy_refuse", ""))',
choices: [ choices: [
{ {
value: 'claims_consent_but_sends_before_consent', value: 'claims_consent_but_sends_before_consent',
text: `Strona wysłała {moje} dane do ${host} zanim {wyraziłem} na to zgodę`, text: `Strona wysłała {moje} dane do ${host} zanim {wyraziłem} na to zgodę`,
}, },
{ {
value: 'claims_consent_but_there_was_no_easy_refuse', value: 'claims_consent_but_there_was_no_easy_refuse',
text: '{Kliknąłem} przycisk od wyrażania zgody, ale w okienku o zgodę nie było natychmiastowo dostępnego przycisku do niewyrażenia zgody jednym kliknięciem', text: '{Kliknąłem} przycisk od wyrażania zgody, ale w okienku o zgodę nie było natychmiastowo dostępnego przycisku do niewyrażenia zgody jednym kliknięciem',
}, },
{ value: 'none', text: 'żadne z powyższych.' }, { value: 'none', text: 'żadne z powyższych.' },
], ],
}, },
{ {
type: 'radiogroup', type: 'radiogroup',
name: f('legitimate_interest_activity_specified'), name: f('legitimate_interest_activity_specified'),
...defaultValue('legitimate_interest_activity_specified'), ...defaultValue('legitimate_interest_activity_specified'),
isRequired: true, isRequired: true,
title: /* HTML */ `Czy administrator strony opisał szczegółowo, na czym polega title: /* HTML */ `Czy administrator strony opisał szczegółowo, na czym polega
uzasadniony interes w kontekście tego celu?`, uzasadniony interes w kontekście tego celu?`,
visibleIf: `{${f('legal_basis_type')}} = "legitimate_interest"`, visibleIf: `{${f('legal_basis_type')}} = "legitimate_interest"`,
choices: [ choices: [
{ {
value: 'precise', value: 'precise',
text: /* HTML */ `Tak, wskazuje jasno na bieżące działania lub korzyści text: /* HTML */ `Tak, wskazuje jasno na bieżące działania lub korzyści wynikające
wynikające z takiego przetwarzania danych.`, z takiego przetwarzania danych.`,
}, },
{ {
value: 'vague', value: 'vague',
text: `Wskazuje tylko ogólnie, jak np. „marketing” czy „statystyki”.`, text: `Wskazuje tylko ogólnie, jak np. „marketing” czy „statystyki”.`,
}, },
{ {
value: 'no', value: 'no',
text: `Nie. Nie wiadomo, na czym ten uzasadniony interes polega.`, text: `Nie. Nie wiadomo, na czym ten uzasadniony interes polega.`,
}, },
], ],
}, },
{ {
type: 'text', type: 'text',
title: `Jak administrator opisał to, na czym polega uzasadniony interes w kontekście ${host}?`, title: `Jak administrator opisał to, na czym polega uzasadniony interes w kontekście ${host}?`,
name: f('legitimate_interest_description'), name: f('legitimate_interest_description'),
visibleIf: `{${f('legitimate_interest_activity_specified')}} = 'vague'`, visibleIf: `{${f('legitimate_interest_activity_specified')}} = 'vague'`,
placeholder: 'marketing', placeholder: 'marketing',
defaultValueExpression: defaultValueExpression:
index == 0 index == 0
? 'marketing' ? 'marketing'
: `{${f('legitimate_interest_description', previous_host)}}`, : `{${f('legitimate_interest_description', previous_host)}}`,
}, },
{ {
type: 'radiogroup', type: 'radiogroup',
title: `Czy domena ${host} należy do podmiotu spoza Europy (np. Google, Facebook)?`, title: `Czy domena ${host} należy do podmiotu spoza Europy (np. Google, Facebook)?`,
name: f('outside_eu'), name: f('outside_eu'),
...defaultValue('outside_eu'), ...defaultValue('outside_eu'),
visibleIf: `{${f('legitimate_interest_activity_specified')}} = "precise" or {${f( visibleIf: `{${f('legitimate_interest_activity_specified')}} = "precise" or {${f(
'consent_problems' 'consent_problems'
)}} = "none"`, )}} = "none"`,
choices: [ choices: [
{ value: 'yes', text: 'Tak' }, { value: 'yes', text: 'Tak' },
{ value: 'no', text: 'Nie' }, { value: 'no', text: 'Nie' },
{ value: 'not_sure', text: 'Nie wiem' }, { value: 'not_sure', text: 'Nie wiem' },
], ],
}, },
{ {
type: 'radiogroup', type: 'radiogroup',
title: `Czy w {Twojej} ocenie wysłanie {Twoich} danych do właściciela domeny ${host} było konieczne do świadczenia zażądanej przez {Ciebie} usługi drogą elektroniczną?`, title: `Czy w {Twojej} ocenie wysłanie {Twoich} danych do właściciela domeny ${host} było konieczne do świadczenia zażądanej przez {Ciebie} usługi drogą elektroniczną?`,
name: f('was_processing_necessary'), name: f('was_processing_necessary'),
...defaultValue('was_processing_necessary'), ...defaultValue('was_processing_necessary'),
visibleIf: `{${f('legal_basis_type')}} = "legitimate_interest" or {${f( visibleIf: `{${f('legal_basis_type')}} = "legitimate_interest" or {${f(
'present' 'present'
)}} = "not_mentioned"`, )}} = "not_mentioned"`,
choices: [ choices: [
{ value: 'yes', text: 'Tak, było konieczne' }, { value: 'yes', text: 'Tak, było konieczne' },
{ value: 'no', text: 'Nie, nie było konieczne' }, { value: 'no', text: 'Nie, nie było konieczne' },
{ value: 'not_sure', text: 'Nie mam zdania' }, { value: 'not_sure', text: 'Nie mam zdania' },
], ],
}, },
], ],
}; };
} }
export default function generateSurveyQuestions(hosts: string[]) { export default function generateSurveyQuestions(hosts: string[]) {
return { return {
showQuestionNumbers: 'off', showQuestionNumbers: 'off',
showProgressBar: 'top', showProgressBar: 'top',
clearInvisibleValues: 'onHidden', clearInvisibleValues: 'onHidden',
pages: [ pages: [
{ {
title: 'Zaimki', title: 'Zaimki',
elements: [ elements: [
{ {
type: 'radiogroup', type: 'radiogroup',
name: 'zaimek', name: 'zaimek',
title: 'Forma czasownika:', title: 'Forma czasownika:',
isRequired: true, isRequired: true,
choices: [ choices: [
{ value: 0, text: 'Wysłałem' }, { value: 0, text: 'Wysłałem' },
{ value: 1, text: 'Wysłałam' }, { value: 1, text: 'Wysłałam' },
{ value: 2, text: 'Wysłałom' }, { value: 2, text: 'Wysłałom' },
{ value: 3, text: 'Wysłaliśmy' }, { value: 3, text: 'Wysłaliśmy' },
], ],
}, },
], ],
}, },
{ {
title: 'Obowiązek informacyjny i machanizm pozyskiwania zgody', title: 'Obowiązek informacyjny i machanizm pozyskiwania zgody',
elements: [ elements: [
{ {
type: 'radiogroup', type: 'radiogroup',
title: 'Jaką formę informacji o przetwarzaniu danych osobowych stosuje ta strona?', title: 'Jaką formę informacji o przetwarzaniu danych osobowych stosuje ta strona?',
name: 'popup_type', name: 'popup_type',
isRequired: true, isRequired: true,
choices: [ choices: [
{ value: 'none', text: 'Brak informacji' }, { value: 'none', text: 'Brak informacji' },
{ {
value: 'page', value: 'page',
text: 'Tylko w postaci tekstu na podstronie np. "prywatność" lub "polityka cookies"', text: 'Tylko w postaci tekstu na podstronie np. "prywatność" lub "polityka cookies"',
}, },
{ {
value: 'passive_popup', value: 'passive_popup',
text: /* HTML */ `Okienko o cookiesach, bez możliwości podjęcia text: /* HTML */ `Okienko o cookiesach, bez możliwości podjęcia żadnego
żadnego wyboru (np. tylko opcja zamknij)`, wyboru (np. tylko opcja zamknij)`,
}, },
{ {
value: 'some_choice', value: 'some_choice',
text: 'Okienko o cookiesach, z możliwością podjęcia wyboru', text: 'Okienko o cookiesach, z możliwością podjęcia wyboru',
}, },
], ],
}, },
{ {
type: 'checkbox', type: 'checkbox',
title: /* HTML */ `Istnieje możliwość, że okienko z informacjami i wyborami title: /* HTML */ `Istnieje możliwość, że okienko z informacjami i wyborami
dotyczącymi przetwarzania {Twoich} danych osobowych ukazało się dawno temu w dotyczącymi przetwarzania {Twoich} danych osobowych ukazało się dawno temu w
trakcie {twojej} wcześniejszej wizyty i wtedy je {odkliknąłeś}. {Otwórz} trakcie {twojej} wcześniejszej wizyty i wtedy je {odkliknąłeś}. {Otwórz} samą
samą stronę w Trybie Prywatnym (Incognito). Co {widzisz}?`, stronę w Trybie Prywatnym (Incognito). Co {widzisz}?`,
visibleIf: "{popup_type} = 'none' or {popup_type} = 'page'", visibleIf: "{popup_type} = 'none' or {popup_type} = 'page'",
name: 'is_incognito_different', name: 'is_incognito_different',
isRequired: true, isRequired: true,
choices: [ choices: [
{ {
value: 'incognito_is_the_same', value: 'incognito_is_the_same',
text: 'W Trybie prywatnym {widzę} to samo, co {widziałem} w normalnym trybie', text: 'W Trybie prywatnym {widzę} to samo, co {widziałem} w normalnym trybie',
}, },
], ],
}, },
{ {
type: 'html', type: 'html',
visibleIf: visibleIf:
'{is_incognito_different} != "no" and ({popup_type} = "none" or {popup_type} = "page") ', '{is_incognito_different} != "no" and ({popup_type} = "none" or {popup_type} = "page") ',
html: /* HTML */ `Jeżeli w trybie incognito widzisz więcej okienek z html: /* HTML */ `Jeżeli w trybie incognito widzisz więcej okienek z informacjami
informacjami o przetwarzaniu danych osobowych, wykonaj analizę w o przetwarzaniu danych osobowych, wykonaj analizę w normalnym trybie ponownie -
normalnym trybie ponownie - ale najpierw usuń pliki cookies tej strony. ale najpierw usuń pliki cookies tej strony.
<a <a
href="https://support.mozilla.org/pl/kb/usuwanie-ciasteczek-i-danych-stron-firefox?redirectslug=usuwanie-ciasteczek&redirectlocale=pl" href="https://support.mozilla.org/pl/kb/usuwanie-ciasteczek-i-danych-stron-firefox?redirectslug=usuwanie-ciasteczek&redirectlocale=pl"
target="_blank" target="_blank"
> >
Zobacz, jak to zrobić Zobacz, jak to zrobić
</a>`, </a>`,
}, },
{ {
type: 'radiogroup', type: 'radiogroup',
name: 'mentions_passive_consent', name: 'mentions_passive_consent',
isRequired: true, isRequired: true,
visibleIf: '{popup_type} = "passive_popup"', visibleIf: '{popup_type} = "passive_popup"',
title: 'Czy treść okienka wskazuje na zgodę wyrażoną pasywnie, np. „Korzystając z naszej strony wyrażasz zgodę”, „Brak zmiany ustawień przeglądarki oznacza zgodę”, „Klikając przycisk "X" (zamknij) wyrażasz zgodę”?', title: 'Czy treść okienka wskazuje na zgodę wyrażoną pasywnie, np. „Korzystając z naszej strony wyrażasz zgodę”, „Brak zmiany ustawień przeglądarki oznacza zgodę”, „Klikając przycisk "X" (zamknij) wyrażasz zgodę”?',
choices: [ choices: [
{ {
value: 'yes', value: 'yes',
text: 'Tak', text: 'Tak',
}, },
{ {
value: 'no', value: 'no',
text: 'Nie', text: 'Nie',
}, },
], ],
}, },
{ {
type: 'text', type: 'text',
name: 'passive_consent_description', name: 'passive_consent_description',
isRequired: true, isRequired: true,
visibleIf: '{mentions_passive_consent} = "yes"', visibleIf: '{mentions_passive_consent} = "yes"',
title: 'Jakimi słowami administrator opisuje to pasywne wyrażenie zgody? Zacytuj wprost', title: 'Jakimi słowami administrator opisuje to pasywne wyrażenie zgody? Zacytuj wprost. Na przykład: „Korzystając ze strony wyrażasz zgodę”, albo „Pozostawiając ustawienia przeglądarki bez zmian (..) wyrażasz zgodę”',
defaultValue: 'Korzystając ze strony wyrażasz zgodę', defaultValue: 'Korzystając ze strony wyrażasz zgodę',
}, },
{ {
type: 'radiogroup', type: 'radiogroup',
name: 'cookie_wall', name: 'cookie_wall',
isRequired: true, isRequired: true,
visibleIf: '{popup_type} = "passive_popup"', visibleIf: '{popup_type} = "passive_popup"',
title: 'Czy treść strony jest wygodnie czytelna bez odkliknięcia tego okienka o RODO?', title: 'Czy treść strony jest wygodnie czytelna bez odkliknięcia tego okienka o RODO?',
choices: [ choices: [
{ {
value: 'no', // wiem, że tu jest "no", a odpowiedź brzmi "tak" - ale nazwa pytania dotyczy obecności cookie walla value: 'no', // wiem, że tu jest "no", a odpowiedź brzmi "tak" - ale nazwa pytania dotyczy obecności cookie walla
text: 'Tak, jest czytelna', text: 'Tak, jest czytelna',
}, },
{ {
value: 'yes', value: 'yes',
text: 'Nie. Jest zupełnie niewidoczna albo jest przesłonięta w stopniu uniemożliwiającym lub znacznie utrudniającym czytanie treści strony.', text: 'Nie. Jest zupełnie niewidoczna albo jest przesłonięta w stopniu uniemożliwiającym lub znacznie utrudniającym czytanie treści strony.',
}, },
], ],
}, },
{ {
type: 'radiogroup', type: 'radiogroup',
name: 'rejection_is_hard', name: 'rejection_is_hard',
isRequired: true, isRequired: true,
visibleIf: '{popup_type} = "some_choice"', visibleIf: '{popup_type} = "some_choice"',
title: 'Czy wyrażenie zgody na wszystkie cele jest dokładnie tak samo łatwe, jak odmowa zgody na wszystkie cele?', title: 'Czy wyrażenie zgody na wszystkie cele jest dokładnie tak samo łatwe, jak odmowa zgody na wszystkie cele?',
choices: [ choices: [
{ {
value: 'no', // wiem, że tu jest "no", a odpowiedź brzmi "tak" - ale nazwa pytania dotyczy braku równowagi value: 'no', // wiem, że tu jest "no", a odpowiedź brzmi "tak" - ale nazwa pytania dotyczy braku równowagi
text: 'Tak. Opcja odmowy zgody na wszystkie cele jest równie widoczna i łatwo dostępna, co opcja wyrażenia zgody.', text: 'Tak. Opcja odmowy zgody na wszystkie cele jest równie widoczna i łatwo dostępna, co opcja wyrażenia zgody.',
}, },
{ {
value: 'yes', value: 'yes',
text: 'Nie. Muszę wykonać więcej czynności aby odmówić wszystkich zgód, albo opcja niewyrażenia zgody jest mało widoczna.', text: 'Nie. Muszę wykonać więcej czynności aby odmówić wszystkich zgód, albo opcja niewyrażenia zgody jest mało widoczna.',
}, },
], ],
}, },
{ {
type: 'radiogroup', type: 'radiogroup',
name: 'popup_action', name: 'popup_action',
isRequired: true, isRequired: true,
visibleIf: '{popup_type} = "some_choice" or {popup_type} = "passive_popup"', visibleIf: '{popup_type} = "some_choice" or {popup_type} = "passive_popup"',
title: 'Jaką akcję {podjąłeś} w ramach wyskakującego okienka?', title: 'Jaką akcję {podjąłeś} w ramach wyskakującego okienka?',
choices: [ choices: [
{ {
value: 'none', value: 'none',
text: 'Nic nie {kliknąłem}', text: 'Nic nie {kliknąłem}',
}, },
{ {
value: 'closed_popup', value: 'closed_popup',
text: '{Zamknąłem} okienko za pomocą przycisku „X” lub „Zamknij”, lub podobnego', text: '{Zamknąłem} okienko za pomocą przycisku „X” lub „Zamknij”, lub podobnego',
}, },
{ {
value: 'accept_all', value: 'accept_all',
text: '{Kliknąłem} przycisk od akceptacji wszystkich zgód', text: '{Kliknąłem} przycisk od akceptacji wszystkich zgód',
}, },
{ {
value: 'deny_all', value: 'deny_all',
text: '{Kliknąłem} przycisk do odmówienia zgody na wszystkie cele', text: '{Odmówiłem} wyrażenia zgody na wszystkie cele',
}, },
{ {
value: 'other', value: 'other',
text: 'Coś innego', text: 'Coś innego',
}, },
], ],
}, },
{ {
type: 'radiogroup', type: 'text',
name: 'administrator_identity_available_before_choice', name: 'popup_closed_how',
isRequired: true, isRequired: true,
visibleIf: '{popup_type} = "some_choice"', visibleIf: '{popup_action} = "closed_popup"',
title: 'Czy przed podjęciem wyboru dot. {Twoich} danych masz możliwość poznać tożsamość administratora strony?', title: 'W jaki sposób {zamknąłeś} okienko o zgodę? Opisz pełnym zdaniem',
choices: [ defaultValueExpression: '{Kliknąłem} przycisk „X”.',
{ },
value: 'yes', {
text: 'Tak.', type: 'text',
}, name: 'popup_deny_all_how',
{ isRequired: true,
value: 'no', visibleIf: '{popup_action} = "deny_all"',
text: 'Nie.', title: 'W jaki sposób {zamknąłeś} okienko o zgodę? Opisz pełnym zdaniem, np.: „{Kliknąłem} przycisk <Odrzuć wszystkie>” lub „{Odznaczyłem} wszystkie opcje w ustawieniach zaawansowanych”',
}, defaultValueExpression: '{Kliknąłem} przycisk „odmawiam wyrażenia zgody”.',
], },
}, {
], type: 'radiogroup',
}, name: 'administrator_identity_available_before_choice',
{ isRequired: true,
title: 'Obowiązek informacyjny, polityka prywatności', visibleIf: '{popup_type} = "some_choice"',
visibleIf: "{popup_type} != 'none'", title: 'Czy przed podjęciem wyboru dot. {Twoich} danych masz możliwość poznać tożsamość administratora strony?',
elements: [ choices: [
{ {
type: 'radiogroup', value: 'yes',
title: 'Czy polityka prywatności jest dostępna i czytelna?', text: 'Tak.',
name: 'policy_readable', },
isRequired: true, {
choices: [ value: 'no',
{ value: 'yes', text: 'dostępna i czytelna' }, text: 'Nie.',
{ },
value: 'entirely_obscured_by_popup', ],
text: 'dostępna, ale nieczytelna. Zasłania ją całkowicie lub prawie całkowicie popup o RODO lub nie można się do niej doklikać bez podjęcia wyboru w okienku', },
}, ],
{ },
value: 'cant_find', {
text: `Niedostępna. {Szukałem}, ale nie {znalazłem} jej na stronie`, title: 'Obowiązek informacyjny, polityka prywatności',
}, visibleIf: "{popup_type} != 'none'",
], elements: [
}, {
], type: 'radiogroup',
}, title: 'Czy polityka prywatności jest dostępna i czytelna?',
...hosts.map(generateHostPage), name: 'policy_readable',
], isRequired: true,
}; choices: [
{ value: 'yes', text: 'dostępna i czytelna' },
{
value: 'entirely_obscured_by_popup',
text: 'dostępna, ale nieczytelna. Zasłania ją całkowicie lub prawie całkowicie popup o RODO lub nie można się do niej doklikać bez podjęcia wyboru w okienku',
},
{
value: 'cant_find',
text: `Niedostępna. {Szukałem}, ale nie {znalazłem} jej na stronie`,
},
],
},
],
},
...hosts.map(generateHostPage),
],
};
} }

View File

@ -19,6 +19,8 @@ export type BasicRawAnswers = {
is_incognito_different: [] | ['incognito_is_the_same']; is_incognito_different: [] | ['incognito_is_the_same'];
policy_readable: 'yes' | 'vague' | 'cant_find'; policy_readable: 'yes' | 'vague' | 'cant_find';
popup_action: 'none' | 'closed_popup' | 'accept_all' | 'deny_all' | 'other'; popup_action: 'none' | 'closed_popup' | 'accept_all' | 'deny_all' | 'other';
popup_closed_how: string;
popup_deny_all_how: string;
} & ( } & (
| ({ | ({
popup_type: 'passive_popup'; popup_type: 'passive_popup';

View File

@ -1,36 +1,44 @@
const words = { const words = {
zrobiłem: ['zrobiłem', 'zrobiłam', 'zrobiłom', 'zrobiliśmy'], zrobiłem: ['zrobiłem', 'zrobiłam', 'zrobiłom', 'zrobiliśmy'],
szukałem: ['szukałem', 'szukałam', 'szukałom', 'szukaliśmy'], szukałem: ['szukałem', 'szukałam', 'szukałom', 'szukaliśmy'],
znalazłem: ['znalazłem', 'znalazłam', 'znalazłom', 'znaleźliśmy'], znalazłem: ['znalazłem', 'znalazłam', 'znalazłom', 'znaleźliśmy'],
moje: ['moje', 'moje', 'moje', 'nasze'], moje: ['moje', 'moje', 'moje', 'nasze'],
wyraziłem: ['wyraziłem', 'wyraziłam', 'wyraziłom', 'wyraziliśmy'], mojej: ['mojej', 'mojej', 'mojej', 'naszej'],
kliknąłem: ['kliknąłem', 'kliknęłam', 'klinkęłom', 'kliknęliśmy'], wyraziłem: ['wyraziłem', 'wyraziłam', 'wyraziłom', 'wyraziliśmy'],
odwiedzałeś: ['odwiedzałeś', 'odwiedzałaś', 'odwiedzałoś', 'odwiedzaliście'], kliknąłem: ['kliknąłem', 'kliknęłam', 'klinkęłom', 'kliknęliśmy'],
wyraziłeś: ['wyraziłeś', 'wyraziłaś', 'wyraziłoś', 'wyraziliście'], odwiedzałeś: ['odwiedzałeś', 'odwiedzałaś', 'odwiedzałoś', 'odwiedzaliście'],
jesteś: ['jesteś', 'jesteś', 'jesteś', 'jesteście'], wyraziłeś: ['wyraziłeś', 'wyraziłaś', 'wyraziłoś', 'wyraziliście'],
twoich: ['twoich', 'twoich', 'twoich', 'waszych'], jesteś: ['jesteś', 'jesteś', 'jesteś', 'jesteście'],
tobie: ['tobie', 'tobie', 'tobie', 'wam'], twoich: ['twoich', 'twoich', 'twoich', 'waszych'],
twojej: ['twojej', 'twojej', 'twojej', 'waszej'], tobie: ['tobie', 'tobie', 'tobie', 'wam'],
odkliknąłeś: ['odkliknąłeś', 'odkliknęłaś', 'odklikęłoś', 'odkliknęliście'], twojej: ['twojej', 'twojej', 'twojej', 'waszej'],
otwórz: ['otwórz', 'otwórz', 'otwórz', 'otwórzcie'], odkliknąłeś: ['odkliknąłeś', 'odkliknęłaś', 'odklikęłoś', 'odkliknęliście'],
widzisz: ['widzisz', 'widzisz', 'widzisz', 'widzicie'], otwórz: ['otwórz', 'otwórz', 'otwórz', 'otwórzcie'],
widzę: ['widzę', 'widzę', 'widzę', 'widzimy'], widzisz: ['widzisz', 'widzisz', 'widzisz', 'widzicie'],
widziałem: ['widziałem', 'widziałam', 'widziałom', 'widzieliśmy'], widzę: ['widzę', 'widzę', 'widzę', 'widzimy'],
odwiedziłem: ['odwiedziłem', 'odwiedziłam', 'odwiedziłom', 'odwiedziliśmy'], widziałem: ['widziałem', 'widziałam', 'widziałom', 'widzieliśmy'],
mam: ['mam', 'mam', 'mam', 'mamy'], odwiedziłem: ['odwiedziłem', 'odwiedziłam', 'odwiedziłom', 'odwiedziliśmy'],
podjąłeś: ['podjąłeś', 'podjęłaś', 'podjęłoś', 'podjęliście'], mam: ['mam', 'mam', 'mam', 'mamy'],
zamknąłem: ['zamknąłem', 'zamknęłaś', 'zamknęłoś', 'zamknęliście'], podjąłeś: ['podjąłeś', 'podjęłaś', 'podjęłoś', 'podjęliście'],
zwracam: ['zwracam', 'zwracam', 'zwracam', 'zwracamy'], zamknąłem: ['zamknąłem', 'zamknęłom', 'zamknęłom', 'zamknęliśmy'],
moich: ['moich', 'moich', 'moich', 'naszych'], zamknąłeś: ['zamknąłeś', 'zamknęłaś', 'zamknęłoś', 'zamknęliście'],
ciebie: ['ciebie', 'ciebie', 'ciebie', 'was'], zwracam: ['zwracam', 'zwracam', 'zwracam', 'zwracamy'],
moich: ['moich', 'moich', 'moich', 'naszych'],
ciebie: ['ciebie', 'ciebie', 'ciebie', 'was'],
mnie: ['mnie', 'mnie', 'mnie', 'nas'],
podjąłem: ['podjąłem', 'podjęłam', 'podjęłom', 'podjęliśmy'],
dokonałeś: ['dokonałeś', 'dokonałaś', 'dokonałoś', 'dokonaliście'],
odmówiłeś: ['odmówiłeś', 'odmówiłaś', 'odmówiłoś', 'odmówiliście'],
odznaczyłem: ['odznaczyłem', 'odznaczyłam', 'odznaczyłom', 'odznaczyliśmy'],
proszę: ['proszę', 'proszę', 'proszę', 'prosimy'],
} as { [key: string]: string[] }; } as { [key: string]: string[] };
export default words; export default words;
export function v(key: string, index: number) { export function v(key: string, index: number) {
let result = words[key.toLowerCase()]?.[index] || key; let result = words[key.toLowerCase()]?.[index] || key;
if (key[0] == key[0].toUpperCase()) { if (key[0] == key[0].toUpperCase()) {
result = [result[0].toUpperCase(), ...result.slice(1)].join(''); result = [result[0].toUpperCase(), ...result.slice(1)].join('');
} }
return result; return result;
} }