Desktop private modes

(default settings)
brave logo
brave
1.61

private
brave logo
brave
1.61

Tor
chrome logo
chrome
120.0

private
edge logo
edge
120.0

private
firefox logo
firefox
121.0

private
librewolf logo
librewolf
121.0-1

private
mullvad logo
mullvad
13.0

private
opera logo
opera
106.0

private
safari logo
safari
17.2

private
tor logo
tor
13.0

private
ungoogled logo
ungoogled
120.0

private
vivaldi logo
vivaldi
6.5

private
State Partitioning testsWhich browsers isolate websites to prevent them from sharing data to track you?
A common vulnerability of web browsers is that they allow tracking companies to 'tag' your browser with some data ('state') that identifies you. When third-party trackers are embedded in websites, they can see this identifying data as you browse to different websites. Fortunately, it is possible for this category of leaks to be fixed by partitioning all data stored in the browser such that no data can be shared between websites.
        
Alt-Svc
Alt-Svc allows the server to indicate to the web browser that a resource should be loaded on a different server. Because this is a persistent setting, it could be used to track users across websites if it is not correctly partitioned.
              
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same first party: h3, h3, h3, h3, h3

result, different first party: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different first party: h2, h2, h2, h2, h2

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same first party: h3, h3, h3, h3, h3

result, different first party: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same first party: h3, h3, h3, h3, h3

result, different first party: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same first party: h3, h3, h3, h3, h3

result, different first party: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same first party: h3, h3, h3, h3, h3

result, different first party: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same first party: h3, h3, h3, h3, h3

result, different first party: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same first party: h3, h3, h3, h3, h3

result, different first party: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same first party: Error: Unsupported

result, different first party: h3

unsupported: true

passed: undefined

test failed: false
Unsupported
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different first party: h2, h2, h2

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same first party: h3, h3, h3, h3, h3

result, different first party: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same first party: h3, h3, h3, h3, h3

result, different first party: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
blob
A 'blob URL' is a local reference to some raw data. Trackers can use a blob URL to share data between websites.
              
Passed
write: (secret) => {
      try {
        let blobURL = URL.createObjectURL(new Blob([secret]));
        fetch(`${baseURI}blob?mode=write&key=${secret}&blobUrl=${encodeURIComponent(blobURL)}`);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async (secret) => {
      let response = await fetch(`${baseURI}blob?mode=read&key=${secret}`);
      let result = await response.json();
      let blobUrl = decodeURIComponent(result.blobUrl);
      let blobResponse = await fetch(blobUrl);
      return blobResponse.text();
    }

result, same first party: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c,
6e0d0559-9ac2-4abc-9025-582a29e22a8f,
066762a4-4e2b-4344-82a6-b11540d563ab,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2

result, different first party: 
Error: Failed to fetch,
Error: Failed to fetch,
Error: Failed to fetch,
Error: Failed to fetch,
Error: Failed to fetch

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      try {
        let blobURL = URL.createObjectURL(new Blob([secret]));
        fetch(`${baseURI}blob?mode=write&key=${secret}&blobUrl=${encodeURIComponent(blobURL)}`);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async (secret) => {
      let response = await fetch(`${baseURI}blob?mode=read&key=${secret}`);
      let result = await response.json();
      let blobUrl = decodeURIComponent(result.blobUrl);
      let blobResponse = await fetch(blobUrl);
      return blobResponse.text();
    }

result, same first party: 
ead45dd4-f0d1-4ab5-af71-11d423c35903,
139bd1c9-0bf8-414e-809a-d12f37b34aab,
33d68d0c-5918-4ede-a519-aff107d2ce1c,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5,
fd597d68-f9be-48a0-9656-b7d0b2e140be

result, different first party: 
Error: Failed to fetch,
Error: Failed to fetch,
Error: Failed to fetch,
Error: Failed to fetch,
Error: Failed to fetch

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Failed
write: (secret) => {
      try {
        let blobURL = URL.createObjectURL(new Blob([secret]));
        fetch(`${baseURI}blob?mode=write&key=${secret}&blobUrl=${encodeURIComponent(blobURL)}`);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async (secret) => {
      let response = await fetch(`${baseURI}blob?mode=read&key=${secret}`);
      let result = await response.json();
      let blobUrl = decodeURIComponent(result.blobUrl);
      let blobResponse = await fetch(blobUrl);
      return blobResponse.text();
    }

result, same first party: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb,
42c2adb9-619b-4b8f-a843-84ede2df77e1,
8b52f9c3-d53f-4dad-91bb-5e162d085e46,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f,
57161958-02cf-461a-8b94-db8e32150e06

result, different first party: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb,
42c2adb9-619b-4b8f-a843-84ede2df77e1,
8b52f9c3-d53f-4dad-91bb-5e162d085e46,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f,
57161958-02cf-461a-8b94-db8e32150e06

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Failed
write: (secret) => {
      try {
        let blobURL = URL.createObjectURL(new Blob([secret]));
        fetch(`${baseURI}blob?mode=write&key=${secret}&blobUrl=${encodeURIComponent(blobURL)}`);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async (secret) => {
      let response = await fetch(`${baseURI}blob?mode=read&key=${secret}`);
      let result = await response.json();
      let blobUrl = decodeURIComponent(result.blobUrl);
      let blobResponse = await fetch(blobUrl);
      return blobResponse.text();
    }

result, same first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40,
142f7096-d2d9-4c89-ba22-d0a3b8055308,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346

result, different first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40,
142f7096-d2d9-4c89-ba22-d0a3b8055308,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Passed
write: (secret) => {
      try {
        let blobURL = URL.createObjectURL(new Blob([secret]));
        fetch(`${baseURI}blob?mode=write&key=${secret}&blobUrl=${encodeURIComponent(blobURL)}`);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async (secret) => {
      let response = await fetch(`${baseURI}blob?mode=read&key=${secret}`);
      let result = await response.json();
      let blobUrl = decodeURIComponent(result.blobUrl);
      let blobResponse = await fetch(blobUrl);
      return blobResponse.text();
    }

result, same first party: 
784004e0-9aeb-47e5-9c5f-12b883e353ac,
eab30b13-9ecc-48af-9a1a-cae23815efbc,
a652f390-6821-4125-9fd0-c118d2a1e7a2,
09045837-46dd-45c8-ad94-7b7c171aeef0,
7cb67be0-5127-4489-aeb1-7164e3b5af92

result, different first party: 
Error: NetworkError when attempting to fetch resource.,
Error: NetworkError when attempting to fetch resource.,
Error: NetworkError when attempting to fetch resource.,
Error: NetworkError when attempting to fetch resource.,
Error: NetworkError when attempting to fetch resource.

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      try {
        let blobURL = URL.createObjectURL(new Blob([secret]));
        fetch(`${baseURI}blob?mode=write&key=${secret}&blobUrl=${encodeURIComponent(blobURL)}`);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async (secret) => {
      let response = await fetch(`${baseURI}blob?mode=read&key=${secret}`);
      let result = await response.json();
      let blobUrl = decodeURIComponent(result.blobUrl);
      let blobResponse = await fetch(blobUrl);
      return blobResponse.text();
    }

result, same first party: 
79152685-f997-47b4-aef1-9e391daef62f,
3863988a-95b2-4b54-a63c-ce0727dc2084,
218debd1-a05b-48c0-b827-e4249b90263c,
6d855fa7-5396-4b31-ac62-f63859f5a3ef,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75

result, different first party: 
Error: NetworkError when attempting to fetch resource.,
Error: NetworkError when attempting to fetch resource.,
Error: NetworkError when attempting to fetch resource.,
Error: NetworkError when attempting to fetch resource.,
Error: NetworkError when attempting to fetch resource.

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      try {
        let blobURL = URL.createObjectURL(new Blob([secret]));
        fetch(`${baseURI}blob?mode=write&key=${secret}&blobUrl=${encodeURIComponent(blobURL)}`);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async (secret) => {
      let response = await fetch(`${baseURI}blob?mode=read&key=${secret}`);
      let result = await response.json();
      let blobUrl = decodeURIComponent(result.blobUrl);
      let blobResponse = await fetch(blobUrl);
      return blobResponse.text();
    }

result, same first party: 
0a2e161c-e0f2-4394-8989-cbcb8cc210b9,
51502f82-4cca-4007-a857-3c34c3db9bf8,
37efbb8c-8f64-4994-8d72-bf40bf93d2bb,
3b85b9c3-ce06-4354-ab1d-998c750dab4a,
70cc2ae2-5861-4824-85db-4f905ce0c785

result, different first party: 
Error: NetworkError when attempting to fetch resource.,
Error: NetworkError when attempting to fetch resource.,
Error: NetworkError when attempting to fetch resource.,
Error: NetworkError when attempting to fetch resource.,
Error: NetworkError when attempting to fetch resource.

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Failed
write: (secret) => {
      try {
        let blobURL = URL.createObjectURL(new Blob([secret]));
        fetch(`${baseURI}blob?mode=write&key=${secret}&blobUrl=${encodeURIComponent(blobURL)}`);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async (secret) => {
      let response = await fetch(`${baseURI}blob?mode=read&key=${secret}`);
      let result = await response.json();
      let blobUrl = decodeURIComponent(result.blobUrl);
      let blobResponse = await fetch(blobUrl);
      return blobResponse.text();
    }

result, same first party: 
92ee92f4-0792-4f9a-ba56-45989cd80572,
606b3a57-c3cf-45fe-a079-3b1566f6abd1,
d40bde18-63e0-419d-b233-1dc09c4c5f94,
df73d3b8-d807-46cf-9f77-65d30bc06fa4,
ba872d52-1a40-47cf-9575-e78f927ff5b3

result, different first party: 
92ee92f4-0792-4f9a-ba56-45989cd80572,
606b3a57-c3cf-45fe-a079-3b1566f6abd1,
d40bde18-63e0-419d-b233-1dc09c4c5f94,
df73d3b8-d807-46cf-9f77-65d30bc06fa4,
ba872d52-1a40-47cf-9575-e78f927ff5b3

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Passed
write: (secret) => {
      try {
        let blobURL = URL.createObjectURL(new Blob([secret]));
        fetch(`${baseURI}blob?mode=write&key=${secret}&blobUrl=${encodeURIComponent(blobURL)}`);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async (secret) => {
      let response = await fetch(`${baseURI}blob?mode=read&key=${secret}`);
      let result = await response.json();
      let blobUrl = decodeURIComponent(result.blobUrl);
      let blobResponse = await fetch(blobUrl);
      return blobResponse.text();
    }

result, same first party: Error: Load failed

result, different first party: Error: Load failed

unsupported: false

passed: undefined

test failed: true
Passed
write: (secret) => {
      try {
        let blobURL = URL.createObjectURL(new Blob([secret]));
        fetch(`${baseURI}blob?mode=write&key=${secret}&blobUrl=${encodeURIComponent(blobURL)}`);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async (secret) => {
      let response = await fetch(`${baseURI}blob?mode=read&key=${secret}`);
      let result = await response.json();
      let blobUrl = decodeURIComponent(result.blobUrl);
      let blobResponse = await fetch(blobUrl);
      return blobResponse.text();
    }

result, same first party: 
37c72365-5852-42a4-9c27-47b4aa879b24,
323291dd-ef8a-484c-aaf9-641497dea358,
bd429ec2-5e93-4afc-8a05-5781fb0abad6

result, different first party: 
Error: NetworkError when attempting to fetch resource.,
Error: NetworkError when attempting to fetch resource.,
Error: NetworkError when attempting to fetch resource.

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Failed
write: (secret) => {
      try {
        let blobURL = URL.createObjectURL(new Blob([secret]));
        fetch(`${baseURI}blob?mode=write&key=${secret}&blobUrl=${encodeURIComponent(blobURL)}`);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async (secret) => {
      let response = await fetch(`${baseURI}blob?mode=read&key=${secret}`);
      let result = await response.json();
      let blobUrl = decodeURIComponent(result.blobUrl);
      let blobResponse = await fetch(blobUrl);
      return blobResponse.text();
    }

result, same first party: 
3a254b81-9486-4572-a38a-b4bd82252b3e,
c0ee6f27-d50e-4975-9a0a-523696e05cbf,
89d605d6-80b6-488e-8454-c4de9c5dfe6b,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6,
d0752b25-d241-400f-a69f-21c3ff242a56

result, different first party: 
3a254b81-9486-4572-a38a-b4bd82252b3e,
c0ee6f27-d50e-4975-9a0a-523696e05cbf,
89d605d6-80b6-488e-8454-c4de9c5dfe6b,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6,
d0752b25-d241-400f-a69f-21c3ff242a56

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Failed
write: (secret) => {
      try {
        let blobURL = URL.createObjectURL(new Blob([secret]));
        fetch(`${baseURI}blob?mode=write&key=${secret}&blobUrl=${encodeURIComponent(blobURL)}`);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async (secret) => {
      let response = await fetch(`${baseURI}blob?mode=read&key=${secret}`);
      let result = await response.json();
      let blobUrl = decodeURIComponent(result.blobUrl);
      let blobResponse = await fetch(blobUrl);
      return blobResponse.text();
    }

result, same first party: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078,
fad257e3-bcc3-4aa7-b541-2b4c42aee488,
3da92bf0-3ab6-4d33-9452-fac73e80ebba,
d276eb0f-b849-45c6-97b9-57596abf3678,
b64d49aa-de07-400a-9631-2935db968706

result, different first party: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078,
fad257e3-bcc3-4aa7-b541-2b4c42aee488,
3da92bf0-3ab6-4d33-9452-fac73e80ebba,
d276eb0f-b849-45c6-97b9-57596abf3678,
b64d49aa-de07-400a-9631-2935db968706

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
BroadcastChannel
A BroadcastChannel is designed to send messages between tabs. In some browsers it can be used for cross-site communication and tracking.
              
Passed
write: (secret) => {
      try {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data === "request") {
            bc.postMessage(secret);
          }
        };
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () =>
      new Promise((resolve, reject) => {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data !== "request") {
            resolve(event.data);
          }
        };
        bc.postMessage("request");
        setTimeout(() => reject({message: "no BroadcastChannel message"}), 3000);
      })

result, same first party: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c,
6e0d0559-9ac2-4abc-9025-582a29e22a8f,
066762a4-4e2b-4344-82a6-b11540d563ab,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2

result, different first party: 
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      try {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data === "request") {
            bc.postMessage(secret);
          }
        };
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () =>
      new Promise((resolve, reject) => {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data !== "request") {
            resolve(event.data);
          }
        };
        bc.postMessage("request");
        setTimeout(() => reject({message: "no BroadcastChannel message"}), 3000);
      })

result, same first party: 
ead45dd4-f0d1-4ab5-af71-11d423c35903,
139bd1c9-0bf8-414e-809a-d12f37b34aab,
33d68d0c-5918-4ede-a519-aff107d2ce1c,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5,
fd597d68-f9be-48a0-9656-b7d0b2e140be

result, different first party: 
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      try {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data === "request") {
            bc.postMessage(secret);
          }
        };
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () =>
      new Promise((resolve, reject) => {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data !== "request") {
            resolve(event.data);
          }
        };
        bc.postMessage("request");
        setTimeout(() => reject({message: "no BroadcastChannel message"}), 3000);
      })

result, same first party: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb,
42c2adb9-619b-4b8f-a843-84ede2df77e1,
8b52f9c3-d53f-4dad-91bb-5e162d085e46,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f,
57161958-02cf-461a-8b94-db8e32150e06

result, different first party: 
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      try {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data === "request") {
            bc.postMessage(secret);
          }
        };
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () =>
      new Promise((resolve, reject) => {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data !== "request") {
            resolve(event.data);
          }
        };
        bc.postMessage("request");
        setTimeout(() => reject({message: "no BroadcastChannel message"}), 3000);
      })

result, same first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40,
142f7096-d2d9-4c89-ba22-d0a3b8055308,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346

result, different first party: 
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      try {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data === "request") {
            bc.postMessage(secret);
          }
        };
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () =>
      new Promise((resolve, reject) => {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data !== "request") {
            resolve(event.data);
          }
        };
        bc.postMessage("request");
        setTimeout(() => reject({message: "no BroadcastChannel message"}), 3000);
      })

result, same first party: 
784004e0-9aeb-47e5-9c5f-12b883e353ac,
eab30b13-9ecc-48af-9a1a-cae23815efbc,
a652f390-6821-4125-9fd0-c118d2a1e7a2,
09045837-46dd-45c8-ad94-7b7c171aeef0,
7cb67be0-5127-4489-aeb1-7164e3b5af92

result, different first party: 
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      try {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data === "request") {
            bc.postMessage(secret);
          }
        };
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () =>
      new Promise((resolve, reject) => {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data !== "request") {
            resolve(event.data);
          }
        };
        bc.postMessage("request");
        setTimeout(() => reject({message: "no BroadcastChannel message"}), 3000);
      })

result, same first party: 
79152685-f997-47b4-aef1-9e391daef62f,
3863988a-95b2-4b54-a63c-ce0727dc2084,
218debd1-a05b-48c0-b827-e4249b90263c,
6d855fa7-5396-4b31-ac62-f63859f5a3ef,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75

result, different first party: 
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      try {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data === "request") {
            bc.postMessage(secret);
          }
        };
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () =>
      new Promise((resolve, reject) => {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data !== "request") {
            resolve(event.data);
          }
        };
        bc.postMessage("request");
        setTimeout(() => reject({message: "no BroadcastChannel message"}), 3000);
      })

result, same first party: 
0a2e161c-e0f2-4394-8989-cbcb8cc210b9,
51502f82-4cca-4007-a857-3c34c3db9bf8,
37efbb8c-8f64-4994-8d72-bf40bf93d2bb,
3b85b9c3-ce06-4354-ab1d-998c750dab4a,
70cc2ae2-5861-4824-85db-4f905ce0c785

result, different first party: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      try {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data === "request") {
            bc.postMessage(secret);
          }
        };
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () =>
      new Promise((resolve, reject) => {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data !== "request") {
            resolve(event.data);
          }
        };
        bc.postMessage("request");
        setTimeout(() => reject({message: "no BroadcastChannel message"}), 3000);
      })

result, same first party: 
92ee92f4-0792-4f9a-ba56-45989cd80572,
606b3a57-c3cf-45fe-a079-3b1566f6abd1,
d40bde18-63e0-419d-b233-1dc09c4c5f94,
df73d3b8-d807-46cf-9f77-65d30bc06fa4,
ba872d52-1a40-47cf-9575-e78f927ff5b3

result, different first party: 
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      try {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data === "request") {
            bc.postMessage(secret);
          }
        };
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () =>
      new Promise((resolve, reject) => {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data !== "request") {
            resolve(event.data);
          }
        };
        bc.postMessage("request");
        setTimeout(() => reject({message: "no BroadcastChannel message"}), 3000);
      })

result, same first party: Error: no BroadcastChannel message

result, different first party: Error: no BroadcastChannel message

unsupported: false

passed: undefined

test failed: true
Passed
write: (secret) => {
      try {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data === "request") {
            bc.postMessage(secret);
          }
        };
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () =>
      new Promise((resolve, reject) => {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data !== "request") {
            resolve(event.data);
          }
        };
        bc.postMessage("request");
        setTimeout(() => reject({message: "no BroadcastChannel message"}), 3000);
      })

result, same first party: 
37c72365-5852-42a4-9c27-47b4aa879b24,
323291dd-ef8a-484c-aaf9-641497dea358,
bd429ec2-5e93-4afc-8a05-5781fb0abad6

result, different first party: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: (secret) => {
      try {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data === "request") {
            bc.postMessage(secret);
          }
        };
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () =>
      new Promise((resolve, reject) => {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data !== "request") {
            resolve(event.data);
          }
        };
        bc.postMessage("request");
        setTimeout(() => reject({message: "no BroadcastChannel message"}), 3000);
      })

result, same first party: 
3a254b81-9486-4572-a38a-b4bd82252b3e,
c0ee6f27-d50e-4975-9a0a-523696e05cbf,
89d605d6-80b6-488e-8454-c4de9c5dfe6b,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6,
d0752b25-d241-400f-a69f-21c3ff242a56

result, different first party: 
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      try {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data === "request") {
            bc.postMessage(secret);
          }
        };
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () =>
      new Promise((resolve, reject) => {
        let bc = new BroadcastChannel("secrets");
        bc.onmessage = (event) => {
          if (event.data !== "request") {
            resolve(event.data);
          }
        };
        bc.postMessage("request");
        setTimeout(() => reject({message: "no BroadcastChannel message"}), 3000);
      })

result, same first party: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078,
fad257e3-bcc3-4aa7-b541-2b4c42aee488,
3da92bf0-3ab6-4d33-9452-fac73e80ebba,
d276eb0f-b849-45c6-97b9-57596abf3678,
b64d49aa-de07-400a-9631-2935db968706

result, different first party: 
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message,
Error: no BroadcastChannel message

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
CacheStorage
The Cache API is a content storage mechanism originally introduced to support ServiceWorkers. If the same Cache object is accessible to multiple websites, it can be abused to track users.
              
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same first party: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c,
6e0d0559-9ac2-4abc-9025-582a29e22a8f,
066762a4-4e2b-4344-82a6-b11540d563ab,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2

result, different first party: 
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same first party: 
ead45dd4-f0d1-4ab5-af71-11d423c35903,
139bd1c9-0bf8-414e-809a-d12f37b34aab,
33d68d0c-5918-4ede-a519-aff107d2ce1c,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5,
fd597d68-f9be-48a0-9656-b7d0b2e140be

result, different first party: 
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same first party: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb,
42c2adb9-619b-4b8f-a843-84ede2df77e1,
8b52f9c3-d53f-4dad-91bb-5e162d085e46,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f,
57161958-02cf-461a-8b94-db8e32150e06

result, different first party: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40,
142f7096-d2d9-4c89-ba22-d0a3b8055308,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346

result, different first party: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same first party: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

result, different first party: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same first party: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

result, different first party: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same first party: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

result, different first party: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same first party: 
92ee92f4-0792-4f9a-ba56-45989cd80572,
606b3a57-c3cf-45fe-a079-3b1566f6abd1,
d40bde18-63e0-419d-b233-1dc09c4c5f94,
df73d3b8-d807-46cf-9f77-65d30bc06fa4,
ba872d52-1a40-47cf-9575-e78f927ff5b3

result, different first party: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same first party: Error: undefined is not an object (evaluating 'cacheKeys[0].url')

result, different first party: Error: undefined is not an object (evaluating 'cacheKeys[0].url')

unsupported: false

passed: undefined

test failed: true
Unsupported
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same first party: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

result, different first party: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same first party: 
3a254b81-9486-4572-a38a-b4bd82252b3e,
c0ee6f27-d50e-4975-9a0a-523696e05cbf,
89d605d6-80b6-488e-8454-c4de9c5dfe6b,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6,
d0752b25-d241-400f-a69f-21c3ff242a56

result, different first party: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same first party: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078,
fad257e3-bcc3-4aa7-b541-2b4c42aee488,
3da92bf0-3ab6-4d33-9452-fac73e80ebba,
d276eb0f-b849-45c6-97b9-57596abf3678,
b64d49aa-de07-400a-9631-2935db968706

result, different first party: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
cookie (HTTP)
The cookie, first introduced by Netscape in 1994, is a small amount of data stored by your browser on a website's behalf. It has legitimate uses, but it is also the classic cross-site tracking mechanism, and today still the most popular method of tracking users across websites. Browsers can stop cookies from being used for cross-site tracking by either blocking or partitioning them.
              
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same first party: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c_http,
6e0d0559-9ac2-4abc-9025-582a29e22a8f_http,
066762a4-4e2b-4344-82a6-b11540d563ab_http,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee_http,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2_http

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same first party: 
ead45dd4-f0d1-4ab5-af71-11d423c35903_http,
139bd1c9-0bf8-414e-809a-d12f37b34aab_http,
33d68d0c-5918-4ede-a519-aff107d2ce1c_http,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5_http,
fd597d68-f9be-48a0-9656-b7d0b2e140be_http

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same first party: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb_http,
42c2adb9-619b-4b8f-a843-84ede2df77e1_http,
8b52f9c3-d53f-4dad-91bb-5e162d085e46_http,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f_http,
57161958-02cf-461a-8b94-db8e32150e06_http

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Failed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40_http,
142f7096-d2d9-4c89-ba22-d0a3b8055308_http,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c_http,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd_http,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346_http

result, different first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40_http,
142f7096-d2d9-4c89-ba22-d0a3b8055308_http,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c_http,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd_http,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346_http

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same first party: 
784004e0-9aeb-47e5-9c5f-12b883e353ac_http,
eab30b13-9ecc-48af-9a1a-cae23815efbc_http,
a652f390-6821-4125-9fd0-c118d2a1e7a2_http,
09045837-46dd-45c8-ad94-7b7c171aeef0_http,
7cb67be0-5127-4489-aeb1-7164e3b5af92_http

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same first party: 
79152685-f997-47b4-aef1-9e391daef62f_http,
3863988a-95b2-4b54-a63c-ce0727dc2084_http,
218debd1-a05b-48c0-b827-e4249b90263c_http,
6d855fa7-5396-4b31-ac62-f63859f5a3ef_http,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75_http

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same first party: 
0a2e161c-e0f2-4394-8989-cbcb8cc210b9_http,
51502f82-4cca-4007-a857-3c34c3db9bf8_http,
37efbb8c-8f64-4994-8d72-bf40bf93d2bb_http,
3b85b9c3-ce06-4354-ab1d-998c750dab4a_http,
70cc2ae2-5861-4824-85db-4f905ce0c785_http

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same first party: 
92ee92f4-0792-4f9a-ba56-45989cd80572_http,
606b3a57-c3cf-45fe-a079-3b1566f6abd1_http,
d40bde18-63e0-419d-b233-1dc09c4c5f94_http,
df73d3b8-d807-46cf-9f77-65d30bc06fa4_http,
ba872d52-1a40-47cf-9575-e78f927ff5b3_http

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same first party: null

result, different first party: null

unsupported: false

passed: undefined

test failed: true
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same first party: 
37c72365-5852-42a4-9c27-47b4aa879b24_http,
323291dd-ef8a-484c-aaf9-641497dea358_http,
bd429ec2-5e93-4afc-8a05-5781fb0abad6_http

result, different first party: , , 

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same first party: 
3a254b81-9486-4572-a38a-b4bd82252b3e_http,
c0ee6f27-d50e-4975-9a0a-523696e05cbf_http,
89d605d6-80b6-488e-8454-c4de9c5dfe6b_http,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6_http,
d0752b25-d241-400f-a69f-21c3ff242a56_http

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same first party: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078_http,
fad257e3-bcc3-4aa7-b541-2b4c42aee488_http,
3da92bf0-3ab6-4d33-9452-fac73e80ebba_http,
d276eb0f-b849-45c6-97b9-57596abf3678_http,
b64d49aa-de07-400a-9631-2935db968706_http

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
cookie (JS)
The cookie, first introduced by Netscape in 1994, is a small amount of data stored by your browser on a website's behalf. It has legitimate uses, but it is also the classic cross-site tracking mechanism, and today still the most popular method of tracking users across websites. Browsers can stop cookies from being used for cross-site tracking by either blocking or partitioning them.
              
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same first party: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c_js,
6e0d0559-9ac2-4abc-9025-582a29e22a8f_js,
066762a4-4e2b-4344-82a6-b11540d563ab_js,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee_js,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2_js

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same first party: 
ead45dd4-f0d1-4ab5-af71-11d423c35903_js,
139bd1c9-0bf8-414e-809a-d12f37b34aab_js,
33d68d0c-5918-4ede-a519-aff107d2ce1c_js,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5_js,
fd597d68-f9be-48a0-9656-b7d0b2e140be_js

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same first party: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb_js,
42c2adb9-619b-4b8f-a843-84ede2df77e1_js,
8b52f9c3-d53f-4dad-91bb-5e162d085e46_js,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f_js,
57161958-02cf-461a-8b94-db8e32150e06_js

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Failed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40_js,
142f7096-d2d9-4c89-ba22-d0a3b8055308_js,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c_js,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd_js,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346_js

result, different first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40_js,
142f7096-d2d9-4c89-ba22-d0a3b8055308_js,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c_js,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd_js,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346_js

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same first party: 
784004e0-9aeb-47e5-9c5f-12b883e353ac_js,
eab30b13-9ecc-48af-9a1a-cae23815efbc_js,
a652f390-6821-4125-9fd0-c118d2a1e7a2_js,
09045837-46dd-45c8-ad94-7b7c171aeef0_js,
7cb67be0-5127-4489-aeb1-7164e3b5af92_js

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same first party: 
79152685-f997-47b4-aef1-9e391daef62f_js,
3863988a-95b2-4b54-a63c-ce0727dc2084_js,
218debd1-a05b-48c0-b827-e4249b90263c_js,
6d855fa7-5396-4b31-ac62-f63859f5a3ef_js,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75_js

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same first party: 
0a2e161c-e0f2-4394-8989-cbcb8cc210b9_js,
51502f82-4cca-4007-a857-3c34c3db9bf8_js,
37efbb8c-8f64-4994-8d72-bf40bf93d2bb_js,
3b85b9c3-ce06-4354-ab1d-998c750dab4a_js,
70cc2ae2-5861-4824-85db-4f905ce0c785_js

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same first party: 
92ee92f4-0792-4f9a-ba56-45989cd80572_js,
606b3a57-c3cf-45fe-a079-3b1566f6abd1_js,
d40bde18-63e0-419d-b233-1dc09c4c5f94_js,
df73d3b8-d807-46cf-9f77-65d30bc06fa4_js,
ba872d52-1a40-47cf-9575-e78f927ff5b3_js

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same first party: null

result, different first party: null

unsupported: false

passed: undefined

test failed: true
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same first party: 
37c72365-5852-42a4-9c27-47b4aa879b24_js,
323291dd-ef8a-484c-aaf9-641497dea358_js,
bd429ec2-5e93-4afc-8a05-5781fb0abad6_js

result, different first party: , , 

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same first party: 
3a254b81-9486-4572-a38a-b4bd82252b3e_js,
c0ee6f27-d50e-4975-9a0a-523696e05cbf_js,
89d605d6-80b6-488e-8454-c4de9c5dfe6b_js,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6_js,
d0752b25-d241-400f-a69f-21c3ff242a56_js

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same first party: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078_js,
fad257e3-bcc3-4aa7-b541-2b4c42aee488_js,
3da92bf0-3ab6-4d33-9452-fac73e80ebba_js,
d276eb0f-b849-45c6-97b9-57596abf3678_js,
b64d49aa-de07-400a-9631-2935db968706_js

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
CookieStore
The Cookie Store API is an alternative asynchronous API for managing cookies, supported by some browsers.
              
Passed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same first party: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c,
6e0d0559-9ac2-4abc-9025-582a29e22a8f,
066762a4-4e2b-4344-82a6-b11540d563ab,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same first party: 
ead45dd4-f0d1-4ab5-af71-11d423c35903,
139bd1c9-0bf8-414e-809a-d12f37b34aab,
33d68d0c-5918-4ede-a519-aff107d2ce1c,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5,
fd597d68-f9be-48a0-9656-b7d0b2e140be

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same first party: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb,
42c2adb9-619b-4b8f-a843-84ede2df77e1,
8b52f9c3-d53f-4dad-91bb-5e162d085e46,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f,
57161958-02cf-461a-8b94-db8e32150e06

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Failed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40,
142f7096-d2d9-4c89-ba22-d0a3b8055308,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346

result, different first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40,
142f7096-d2d9-4c89-ba22-d0a3b8055308,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same first party: 
92ee92f4-0792-4f9a-ba56-45989cd80572,
606b3a57-c3cf-45fe-a079-3b1566f6abd1,
d40bde18-63e0-419d-b233-1dc09c4c5f94,
df73d3b8-d807-46cf-9f77-65d30bc06fa4,
ba872d52-1a40-47cf-9575-e78f927ff5b3

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same first party: Error: Unsupported

result, different first party: Error: Unsupported

unsupported: true

passed: undefined

test failed: false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Passed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same first party: 
3a254b81-9486-4572-a38a-b4bd82252b3e,
c0ee6f27-d50e-4975-9a0a-523696e05cbf,
89d605d6-80b6-488e-8454-c4de9c5dfe6b,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6,
d0752b25-d241-400f-a69f-21c3ff242a56

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same first party: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078,
fad257e3-bcc3-4aa7-b541-2b4c42aee488,
3da92bf0-3ab6-4d33-9452-fac73e80ebba,
d276eb0f-b849-45c6-97b9-57596abf3678,
b64d49aa-de07-400a-9631-2935db968706

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
CSS cache
CSS stylesheets are cached, and if that cache is shared between websites, it can be used to track users across sites.
              
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same first party: 
fake_07799567495243065,
fake_8673154376160528,
fake_7526641367119917,
fake_5571568211409275,
fake_1577629022970859

result, different first party: 
fake_08726511467376508,
fake_501138773978149,
fake_7886341215032246,
fake_9538753229694468,
fake_1360922296574425

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same first party: 
fake_2543439211795755,
fake_5570103118366942,
fake_4446354762661302,
fake_26449432070588097,
fake_12778365000889647

result, different first party: 
fake_6117722829416457,
fake_9839238338787657,
fake_8885701010315497,
fake_45985720494496585,
fake_5533185575659723

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same first party: 
fake_5335951983679681,
fake_8193042157707182,
fake_6852735881034362,
fake_7749420697080194,
fake_5890443894285433

result, different first party: 
fake_15653167426756975,
fake_11655042361600554,
fake_9053678258904012,
fake_11345114547945134,
fake_33027606744650906

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same first party: 
fake_10340319999845171,
fake_1494665825219168,
fake_8377748709851747,
fake_3829122165084864,
fake_5504457967935605

result, different first party: 
fake_6548341924698589,
fake_6947546085759635,
fake_14856565527295817,
fake_7635020295757327,
fake_4871132818088746

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same first party: 
fake_39857838129963863,
fake_15394478283090818,
fake_41446350556763334,
fake_38263028788539377,
fake_9467882845525231

result, different first party: 
fake_6710618842415697,
fake_7865371252229845,
fake_15872212046883627,
fake_41847820191986385,
fake_9735890175553588

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same first party: 
fake_5387845685539836,
fake_2206937161654956,
fake_42987701513496734,
fake_5116791455300023,
fake_6293689986773812

result, different first party: 
fake_445896999759527,
fake_44050114116406114,
fake_8942926042905524,
fake_6152308565954363,
fake_5793336089557319

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same first party: 
fake_5672735529126212,
fake_7105539461901222,
fake_024217881294009924,
fake_8621590002487538,
fake_8888717582025429

result, different first party: 
fake_8453166190078618,
fake_6085887767260363,
fake_7393102816742281,
fake_984031171892646,
fake_2574971759849507

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same first party: 
fake_9837911934270462,
fake_8390993113200158,
fake_8874450320464911,
fake_6044319197660601,
fake_10772715932081467

result, different first party: 
fake_9083340907818929,
fake_4390896000609408,
fake_4366067399176796,
fake_9442816727149186,
fake_7869068624897504

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same first party: fake_8863090416507016

result, different first party: fake_014865151201008286

unsupported: false

passed: true

test failed: false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same first party: 
fake_41227619531392645,
fake_48525589403463676,
fake_8347679969287813

result, different first party: 
fake_43363195478183725,
fake_16227076071265834,
fake_8012590025913133

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same first party: 
fake_5015518382546111,
fake_4691632387290017,
fake_09328342791042643,
fake_5123414426910102,
fake_5341223186957997

result, different first party: 
fake_7499558686925996,
fake_16991824468615002,
fake_9333306991334847,
fake_02241495880223665,
fake_6530779117414336

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same first party: 
fake_6222700119394466,
fake_811452555524633,
fake_44068590063967195,
fake_03690101231006415,
fake_08994360004965585

result, different first party: 
fake_912334304208049,
fake_5466154503066787,
fake_6240470931705899,
fake_7950114173575351,
fake_004118680896003646

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
favicon cache
A favicon is an icon that represents a website, typically shown in browser tab and bookmarks menu. If the favicon cache is not partitioned, it can be used to track users across websites.
              
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same first party: 1

result, different first party: 2

unsupported: false

passed: true

test failed: false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same first party: 1, 1, 1

result, different first party: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same first party: 2, 2, 2, 2, 2

result, different first party: 3, 3, 3, 3, 3

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
fetch cache
When a resource is received via the Fetch API, it is frequently cached. That cache can potentially be abused for cross-site tracking.
              
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 2

result, different first party: 3

unsupported: false

passed: true

test failed: false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1

result, different first party: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
font cache
Web fonts are sometimes stored in their own cache, which is vulnerable to being abused for cross-site tracking.
              
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 2

result, different first party: 3

unsupported: false

passed: true

test failed: false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1

result, different first party: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
getDirectory
navigator.storage.getDirectory exposes a location for storing files to web content. In some cases, these files may be shared across tabs.
              
Passed
write: async (secret) => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt", { create: true });
        const stream = await fileHandle.createWritable();
        await stream.write(secret);
        await stream.close();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt");
        const file = await fileHandle.getFile();
        return file.text();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

result, same first party: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c,
6e0d0559-9ac2-4abc-9025-582a29e22a8f,
066762a4-4e2b-4344-82a6-b11540d563ab,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt", { create: true });
        const stream = await fileHandle.createWritable();
        await stream.write(secret);
        await stream.close();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt");
        const file = await fileHandle.getFile();
        return file.text();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

result, same first party: 
ead45dd4-f0d1-4ab5-af71-11d423c35903,
139bd1c9-0bf8-414e-809a-d12f37b34aab,
33d68d0c-5918-4ede-a519-aff107d2ce1c,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5,
fd597d68-f9be-48a0-9656-b7d0b2e140be

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt", { create: true });
        const stream = await fileHandle.createWritable();
        await stream.write(secret);
        await stream.close();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt");
        const file = await fileHandle.getFile();
        return file.text();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

result, same first party: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb,
42c2adb9-619b-4b8f-a843-84ede2df77e1,
8b52f9c3-d53f-4dad-91bb-5e162d085e46,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f,
57161958-02cf-461a-8b94-db8e32150e06

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt", { create: true });
        const stream = await fileHandle.createWritable();
        await stream.write(secret);
        await stream.close();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt");
        const file = await fileHandle.getFile();
        return file.text();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

result, same first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40,
142f7096-d2d9-4c89-ba22-d0a3b8055308,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt", { create: true });
        const stream = await fileHandle.createWritable();
        await stream.write(secret);
        await stream.close();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt");
        const file = await fileHandle.getFile();
        return file.text();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

result, same first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt", { create: true });
        const stream = await fileHandle.createWritable();
        await stream.write(secret);
        await stream.close();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt");
        const file = await fileHandle.getFile();
        return file.text();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

result, same first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt", { create: true });
        const stream = await fileHandle.createWritable();
        await stream.write(secret);
        await stream.close();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt");
        const file = await fileHandle.getFile();
        return file.text();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

result, same first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt", { create: true });
        const stream = await fileHandle.createWritable();
        await stream.write(secret);
        await stream.close();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt");
        const file = await fileHandle.getFile();
        return file.text();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

result, same first party: 
92ee92f4-0792-4f9a-ba56-45989cd80572,
606b3a57-c3cf-45fe-a079-3b1566f6abd1,
d40bde18-63e0-419d-b233-1dc09c4c5f94,
df73d3b8-d807-46cf-9f77-65d30bc06fa4,
ba872d52-1a40-47cf-9575-e78f927ff5b3

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt", { create: true });
        const stream = await fileHandle.createWritable();
        await stream.write(secret);
        await stream.close();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt");
        const file = await fileHandle.getFile();
        return file.text();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

result, same first party: Error: Unsupported

result, different first party: Error: Unsupported

unsupported: true

passed: undefined

test failed: false
Unsupported
write: async (secret) => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt", { create: true });
        const stream = await fileHandle.createWritable();
        await stream.write(secret);
        await stream.close();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt");
        const file = await fileHandle.getFile();
        return file.text();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

result, same first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Passed
write: async (secret) => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt", { create: true });
        const stream = await fileHandle.createWritable();
        await stream.write(secret);
        await stream.close();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt");
        const file = await fileHandle.getFile();
        return file.text();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

result, same first party: 
3a254b81-9486-4572-a38a-b4bd82252b3e,
c0ee6f27-d50e-4975-9a0a-523696e05cbf,
89d605d6-80b6-488e-8454-c4de9c5dfe6b,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6,
d0752b25-d241-400f-a69f-21c3ff242a56

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt", { create: true });
        const stream = await fileHandle.createWritable();
        await stream.write(secret);
        await stream.close();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      try {
        const root = await navigator.storage.getDirectory();
        const fileHandle = await root.getFileHandle("secret.txt");
        const file = await fileHandle.getFile();
        return file.text();
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

result, same first party: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078,
fad257e3-bcc3-4aa7-b541-2b4c42aee488,
3da92bf0-3ab6-4d33-9452-fac73e80ebba,
d276eb0f-b849-45c6-97b9-57596abf3678,
b64d49aa-de07-400a-9631-2935db968706

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
H1 connection
HTTP/1.x are the classic web connection protocols. If these connections are re-used across websites, they can be used to track users.
              
Passed
write: async (secret) => {
      await fetch(`https://h1.privacytests2.org:8901/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h1.privacytests2.org:8901/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c,
6e0d0559-9ac2-4abc-9025-582a29e22a8f,
066762a4-4e2b-4344-82a6-b11540d563ab,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h1.privacytests2.org:8901/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h1.privacytests2.org:8901/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
ead45dd4-f0d1-4ab5-af71-11d423c35903,
139bd1c9-0bf8-414e-809a-d12f37b34aab,
33d68d0c-5918-4ede-a519-aff107d2ce1c,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5,
fd597d68-f9be-48a0-9656-b7d0b2e140be

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h1.privacytests2.org:8901/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h1.privacytests2.org:8901/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb,
42c2adb9-619b-4b8f-a843-84ede2df77e1,
8b52f9c3-d53f-4dad-91bb-5e162d085e46,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f,
57161958-02cf-461a-8b94-db8e32150e06

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h1.privacytests2.org:8901/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h1.privacytests2.org:8901/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40,
142f7096-d2d9-4c89-ba22-d0a3b8055308,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h1.privacytests2.org:8901/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h1.privacytests2.org:8901/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
784004e0-9aeb-47e5-9c5f-12b883e353ac,
eab30b13-9ecc-48af-9a1a-cae23815efbc,
a652f390-6821-4125-9fd0-c118d2a1e7a2,
09045837-46dd-45c8-ad94-7b7c171aeef0,
7cb67be0-5127-4489-aeb1-7164e3b5af92

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h1.privacytests2.org:8901/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h1.privacytests2.org:8901/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
79152685-f997-47b4-aef1-9e391daef62f,
3863988a-95b2-4b54-a63c-ce0727dc2084,
218debd1-a05b-48c0-b827-e4249b90263c,
6d855fa7-5396-4b31-ac62-f63859f5a3ef,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h1.privacytests2.org:8901/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h1.privacytests2.org:8901/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
0a2e161c-e0f2-4394-8989-cbcb8cc210b9,
51502f82-4cca-4007-a857-3c34c3db9bf8,
37efbb8c-8f64-4994-8d72-bf40bf93d2bb,
3b85b9c3-ce06-4354-ab1d-998c750dab4a,
70cc2ae2-5861-4824-85db-4f905ce0c785

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h1.privacytests2.org:8901/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h1.privacytests2.org:8901/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
92ee92f4-0792-4f9a-ba56-45989cd80572,
606b3a57-c3cf-45fe-a079-3b1566f6abd1,
d40bde18-63e0-419d-b233-1dc09c4c5f94,
df73d3b8-d807-46cf-9f77-65d30bc06fa4,
ba872d52-1a40-47cf-9575-e78f927ff5b3

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h1.privacytests2.org:8901/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h1.privacytests2.org:8901/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 

result, different first party: 

unsupported: false

passed: undefined

test failed: true
Passed
write: async (secret) => {
      await fetch(`https://h1.privacytests2.org:8901/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h1.privacytests2.org:8901/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
37c72365-5852-42a4-9c27-47b4aa879b24,
323291dd-ef8a-484c-aaf9-641497dea358,
bd429ec2-5e93-4afc-8a05-5781fb0abad6

result, different first party: , , 

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h1.privacytests2.org:8901/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h1.privacytests2.org:8901/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
3a254b81-9486-4572-a38a-b4bd82252b3e,
c0ee6f27-d50e-4975-9a0a-523696e05cbf,
89d605d6-80b6-488e-8454-c4de9c5dfe6b,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6,
d0752b25-d241-400f-a69f-21c3ff242a56

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h1.privacytests2.org:8901/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h1.privacytests2.org:8901/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078,
fad257e3-bcc3-4aa7-b541-2b4c42aee488,
3da92bf0-3ab6-4d33-9452-fac73e80ebba,
d276eb0f-b849-45c6-97b9-57596abf3678,
b64d49aa-de07-400a-9631-2935db968706

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
H2 connection
HTTP/2 is a web connection protocol introduced in 2015. Some browsers re-use HTTP/2 connections across websites and can thus be used to track users.
              
Passed
write: async (secret) => {
      await fetch(`https://h2.privacytests2.org:8902/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h2.privacytests2.org:8902/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c,
6e0d0559-9ac2-4abc-9025-582a29e22a8f,
066762a4-4e2b-4344-82a6-b11540d563ab,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h2.privacytests2.org:8902/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h2.privacytests2.org:8902/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
ead45dd4-f0d1-4ab5-af71-11d423c35903,
139bd1c9-0bf8-414e-809a-d12f37b34aab,
33d68d0c-5918-4ede-a519-aff107d2ce1c,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5,
fd597d68-f9be-48a0-9656-b7d0b2e140be

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h2.privacytests2.org:8902/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h2.privacytests2.org:8902/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb,
42c2adb9-619b-4b8f-a843-84ede2df77e1,
8b52f9c3-d53f-4dad-91bb-5e162d085e46,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f,
57161958-02cf-461a-8b94-db8e32150e06

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h2.privacytests2.org:8902/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h2.privacytests2.org:8902/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40,
142f7096-d2d9-4c89-ba22-d0a3b8055308,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h2.privacytests2.org:8902/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h2.privacytests2.org:8902/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
784004e0-9aeb-47e5-9c5f-12b883e353ac,
eab30b13-9ecc-48af-9a1a-cae23815efbc,
a652f390-6821-4125-9fd0-c118d2a1e7a2,
09045837-46dd-45c8-ad94-7b7c171aeef0,
7cb67be0-5127-4489-aeb1-7164e3b5af92

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h2.privacytests2.org:8902/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h2.privacytests2.org:8902/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
79152685-f997-47b4-aef1-9e391daef62f,
3863988a-95b2-4b54-a63c-ce0727dc2084,
218debd1-a05b-48c0-b827-e4249b90263c,
6d855fa7-5396-4b31-ac62-f63859f5a3ef,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h2.privacytests2.org:8902/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h2.privacytests2.org:8902/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
0a2e161c-e0f2-4394-8989-cbcb8cc210b9,
51502f82-4cca-4007-a857-3c34c3db9bf8,
37efbb8c-8f64-4994-8d72-bf40bf93d2bb,
3b85b9c3-ce06-4354-ab1d-998c750dab4a,
70cc2ae2-5861-4824-85db-4f905ce0c785

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h2.privacytests2.org:8902/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h2.privacytests2.org:8902/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
92ee92f4-0792-4f9a-ba56-45989cd80572,
606b3a57-c3cf-45fe-a079-3b1566f6abd1,
d40bde18-63e0-419d-b233-1dc09c4c5f94,
df73d3b8-d807-46cf-9f77-65d30bc06fa4,
ba872d52-1a40-47cf-9575-e78f927ff5b3

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h2.privacytests2.org:8902/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h2.privacytests2.org:8902/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 

result, different first party: 

unsupported: false

passed: undefined

test failed: true
Passed
write: async (secret) => {
      await fetch(`https://h2.privacytests2.org:8902/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h2.privacytests2.org:8902/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
37c72365-5852-42a4-9c27-47b4aa879b24,
323291dd-ef8a-484c-aaf9-641497dea358,
bd429ec2-5e93-4afc-8a05-5781fb0abad6

result, different first party: , , 

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h2.privacytests2.org:8902/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h2.privacytests2.org:8902/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
3a254b81-9486-4572-a38a-b4bd82252b3e,
c0ee6f27-d50e-4975-9a0a-523696e05cbf,
89d605d6-80b6-488e-8454-c4de9c5dfe6b,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6,
d0752b25-d241-400f-a69f-21c3ff242a56

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      await fetch(`https://h2.privacytests2.org:8902/?mode=write&secret=${secret}`, {cache: "no-store"});
    }

read: async () => {
      let response = await fetch(`https://h2.privacytests2.org:8902/?mode=read`, {cache: "no-store"});
      return await response.text();
    }

result, same first party: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078,
fad257e3-bcc3-4aa7-b541-2b4c42aee488,
3da92bf0-3ab6-4d33-9452-fac73e80ebba,
d276eb0f-b849-45c6-97b9-57596abf3678,
b64d49aa-de07-400a-9631-2935db968706

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
H3 connection
HTTP/3 is a new standard HTTP connection protocol, still in draft but widely supported by browsers. If it is not partitioned, it can be used to track users across websites.
              
Passed
write: async (secret) => {
      // Ensure that we can switch over to h3 via alt-svc:
      for (let i = 0; i<3; ++i) {
        await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
        await sleepMs(500);
      }
      // Are we now connecting over h3?
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
      let text = await response.text();
      // Empty response text indicates we are not connecting over h3:
      if (text.trim() === "") {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`);
      return await response.text();
    }

result, same first party: 
f3f3bc3ff2c9228c39d6b2e9630ee5b1,
ad7ba18d2c9145db7cc074d80607bccf,
685c4f520cd2491ee404ca8ff2dbca3d,
0e91b0381d9cd0022f62a6f08473aaa7,
28f6c2285934d23e14e77b4a27ec6bad

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      // Ensure that we can switch over to h3 via alt-svc:
      for (let i = 0; i<3; ++i) {
        await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
        await sleepMs(500);
      }
      // Are we now connecting over h3?
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
      let text = await response.text();
      // Empty response text indicates we are not connecting over h3:
      if (text.trim() === "") {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`);
      return await response.text();
    }

result, same first party: , , , , 

result, different first party: , , , , 

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Ensure that we can switch over to h3 via alt-svc:
      for (let i = 0; i<3; ++i) {
        await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
        await sleepMs(500);
      }
      // Are we now connecting over h3?
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
      let text = await response.text();
      // Empty response text indicates we are not connecting over h3:
      if (text.trim() === "") {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`);
      return await response.text();
    }

result, same first party: 
d01eec9dd199b3cfa0cdcad59ec50f07,
fd38994ee84a7069a9027f4503805697,
ab3432a325bd8a1fcd43813d1c483f10,
c24c491511d76fd1a93ab1d7861e9591,
b523bf1e4fdaff43959ed769976e8ca7

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Ensure that we can switch over to h3 via alt-svc:
      for (let i = 0; i<3; ++i) {
        await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
        await sleepMs(500);
      }
      // Are we now connecting over h3?
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
      let text = await response.text();
      // Empty response text indicates we are not connecting over h3:
      if (text.trim() === "") {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`);
      return await response.text();
    }

result, same first party: 
2a0bccabaa4f20e6a588b3b2d818759c,
12b3164cb0ae5a801edd8fd762e991e5,
b64f0cade018a7be35d69ceed00d77a5,
9cf759660143eb2010983420d9c97bde,
c60265efa555f0156ad16e72682bf774

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Ensure that we can switch over to h3 via alt-svc:
      for (let i = 0; i<3; ++i) {
        await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
        await sleepMs(500);
      }
      // Are we now connecting over h3?
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
      let text = await response.text();
      // Empty response text indicates we are not connecting over h3:
      if (text.trim() === "") {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`);
      return await response.text();
    }

result, same first party: 
0aca8cef88a250c7ba3596cc7e23ef4d,
ebde3a7f92fab7aed199e3d91401e042,
1ee4262f9a1010989510f92b24330140,
00b1f9cfdd5082a7a38735ba47152bd9,
c25d81afafc649a80eed3d8ff44e5787

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Ensure that we can switch over to h3 via alt-svc:
      for (let i = 0; i<3; ++i) {
        await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
        await sleepMs(500);
      }
      // Are we now connecting over h3?
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
      let text = await response.text();
      // Empty response text indicates we are not connecting over h3:
      if (text.trim() === "") {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`);
      return await response.text();
    }

result, same first party: 
57b2f8f5bcc2fa05ee3c59d1600a6beb,
68d7bb42a53a309a76d8b47e62e46a28,
09da0a252cb3b36df7bb18c3cafeec44,
1c5f2aedec097326182a3000eea5c496,
644f840b7f38870afa719efbd9abf51c

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Ensure that we can switch over to h3 via alt-svc:
      for (let i = 0; i<3; ++i) {
        await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
        await sleepMs(500);
      }
      // Are we now connecting over h3?
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
      let text = await response.text();
      // Empty response text indicates we are not connecting over h3:
      if (text.trim() === "") {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`);
      return await response.text();
    }

result, same first party: 
1eba6fd7cb7a2792569ff1f272257662,
400ed27accb1045e3838a423552deca5,
57ef94af8dcc8ce892c6f8a7ae2b1f99,
7e7823fd851e4e7959e3416f91af2530,
375de32266a97fd5c901cacc5227f65c

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Ensure that we can switch over to h3 via alt-svc:
      for (let i = 0; i<3; ++i) {
        await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
        await sleepMs(500);
      }
      // Are we now connecting over h3?
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
      let text = await response.text();
      // Empty response text indicates we are not connecting over h3:
      if (text.trim() === "") {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`);
      return await response.text();
    }

result, same first party: 
b16ea085bc30d0542e4c18707f4c2b14,
fb39fe192a1611ceefcbe7d79fff5f99,
3e1b17dc25599f2f3fde788cf6872074,
182f4d8c67fe1a7b654e49106c6bc2c1,
220d664bb3aed5bcbfde44b84db211fd

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      // Ensure that we can switch over to h3 via alt-svc:
      for (let i = 0; i<3; ++i) {
        await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
        await sleepMs(500);
      }
      // Are we now connecting over h3?
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
      let text = await response.text();
      // Empty response text indicates we are not connecting over h3:
      if (text.trim() === "") {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`);
      return await response.text();
    }

result, same first party: 

result, different first party: 

unsupported: true

passed: undefined

test failed: false
Unsupported
write: async (secret) => {
      // Ensure that we can switch over to h3 via alt-svc:
      for (let i = 0; i<3; ++i) {
        await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
        await sleepMs(500);
      }
      // Are we now connecting over h3?
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
      let text = await response.text();
      // Empty response text indicates we are not connecting over h3:
      if (text.trim() === "") {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`);
      return await response.text();
    }

result, same first party: , , 

result, different first party: , , 

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Passed
write: async (secret) => {
      // Ensure that we can switch over to h3 via alt-svc:
      for (let i = 0; i<3; ++i) {
        await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
        await sleepMs(500);
      }
      // Are we now connecting over h3?
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
      let text = await response.text();
      // Empty response text indicates we are not connecting over h3:
      if (text.trim() === "") {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`);
      return await response.text();
    }

result, same first party: 
ef8b16c0237775750620386eedb7b317,
6929af9c688d4ea186b729be6fee0a14,
0b41ce61c94d710a58a3015a97baeb3d,
1088df2387a281d0dfdd69fdcacc6849,
4288c1d99734534ac9ecfc3e1e694cf3

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Ensure that we can switch over to h3 via alt-svc:
      for (let i = 0; i<3; ++i) {
        await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
        await sleepMs(500);
      }
      // Are we now connecting over h3?
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`, {cache: "no-store"});
      let text = await response.text();
      // Empty response text indicates we are not connecting over h3:
      if (text.trim() === "") {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let response = await fetch(`https://h3.privacytests2.org:4434/connection_id`);
      return await response.text();
    }

result, same first party: 
607437a52ec1605f7b8851d8d0ba7c2b,
20524ecbbd05182b746ad2d8ffc0717e,
ced7800dd1cbe6a4e29fec5be5bb38ed,
1c0ccf3c3f99e5b87c6fee7079dbe357,
4117624f1860d0f42866e2b6633fd35e

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
HSTS cache
The HTTP Strict-Transport-Security response header allows a website to signal that it should only be accessed via HTTPS. The browser remembers this directive in a database, but if this database is not partitioned, then it can be used to track users across websites."
              
Passed
write: set HSTS flag

read: read HSTS flag

result, same first party: not tested, not tested, not tested, not tested, not tested

result, different first party: Used http, Used http, Used http, Used http, Used http

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: null

read: null

result, same first party: , , , , 

result, different first party: 
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected

unsupported: , , , , 

passed: true, true, true, true, true

test failed: false, false, false, false, false
Failed
write: set HSTS flag

read: read HSTS flag

result, same first party: not tested, not tested, not tested, not tested, not tested

result, different first party: 
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Failed
write: set HSTS flag

read: read HSTS flag

result, same first party: not tested, not tested, not tested, not tested, not tested

result, different first party: 
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Passed
write: set HSTS flag

read: read HSTS flag

result, same first party: not tested, not tested, not tested, not tested, not tested

result, different first party: Used http, Used http, Used http, Used http, Used http

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: null

read: null

result, same first party: , , , , 

result, different first party: 
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected

unsupported: , , , , 

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: null

read: null

result, same first party: , , , , 

result, different first party: 
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected

unsupported: , , , , 

passed: true, true, true, true, true

test failed: false, false, false, false, false
Failed
write: set HSTS flag

read: read HSTS flag

result, same first party: not tested, not tested, not tested, not tested, not tested

result, different first party: 
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Passed
write: set HSTS flag

read: read HSTS flag

result, same first party: not tested

result, different first party: Used http

unsupported: false

passed: true

test failed: false
Passed
write: null

read: null

result, same first party: , , 

result, different first party: 
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected

unsupported: , , 

passed: true, true, true

test failed: false, false, false
Failed
write: set HSTS flag

read: read HSTS flag

result, same first party: not tested, not tested, not tested, not tested, not tested

result, different first party: 
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Failed
write: set HSTS flag

read: read HSTS flag

result, same first party: not tested, not tested, not tested, not tested, not tested

result, different first party: 
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
HSTS cache (fetch)
The HTTP Strict-Transport-Security response header allows a website to signal that it should only be accessed via HTTPS. The browser remembers this directive in a database, but if this database is not partitioned, then it can be used to track users across websites."
              
Passed
write: set HSTS flag

read: read HSTS flag

result, same first party: not tested, not tested, not tested, not tested, not tested

result, different first party: Used http, Used http, Used http, Used http, Used http

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: null

read: null

result, same first party: , , , , 

result, different first party: 
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected

unsupported: , , , , 

passed: true, true, true, true, true

test failed: false, false, false, false, false
Failed
write: set HSTS flag

read: read HSTS flag

result, same first party: not tested, not tested, not tested, not tested, not tested

result, different first party: 
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Failed
write: set HSTS flag

read: read HSTS flag

result, same first party: not tested, not tested, not tested, not tested, not tested

result, different first party: 
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Passed
write: set HSTS flag

read: read HSTS flag

result, same first party: not tested, not tested, not tested, not tested, not tested

result, different first party: Used http, Used http, Used http, Used http, Used http

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: null

read: null

result, same first party: , , , , 

result, different first party: 
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected

unsupported: , , , , 

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: null

read: null

result, same first party: , , , , 

result, different first party: 
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected

unsupported: , , , , 

passed: true, true, true, true, true

test failed: false, false, false, false, false
Failed
write: set HSTS flag

read: read HSTS flag

result, same first party: not tested, not tested, not tested, not tested, not tested

result, different first party: 
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Passed
write: set HSTS flag

read: read HSTS flag

result, same first party: not tested

result, different first party: Used http

unsupported: false

passed: true

test failed: false
Passed
write: null

read: null

result, same first party: , , 

result, different first party: 
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected,
HTTPS used by default; no HSTS cache issue expected

unsupported: , , 

passed: true, true, true

test failed: false, false, false
Failed
write: set HSTS flag

read: read HSTS flag

result, same first party: not tested, not tested, not tested, not tested, not tested

result, different first party: 
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Failed
write: set HSTS flag

read: read HSTS flag

result, same first party: not tested, not tested, not tested, not tested, not tested

result, different first party: 
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https,
Upgraded to https

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
iframe cache
An iframe is an element in a web page than allows websites to embed a second web page. Caching of this web page could be abused for cross-site tracking.
              
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 2

result, different first party: 3

unsupported: false

passed: true

test failed: false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1

result, different first party: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
image cache
Caching of images in web browsers is a standard behavior. But if that cache leaks between websites, it can be abused for cross-site tracking.
              
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 2

result, different first party: 3

unsupported: false

passed: true

test failed: false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1

result, different first party: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
indexedDB
The IndexedDB API exposes a transactional database to web pages. That database can be used to track users across websites, unless it is partitioned.
              
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same first party: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c,
6e0d0559-9ac2-4abc-9025-582a29e22a8f,
066762a4-4e2b-4344-82a6-b11540d563ab,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2

result, different first party: 
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same first party: 
ead45dd4-f0d1-4ab5-af71-11d423c35903,
139bd1c9-0bf8-414e-809a-d12f37b34aab,
33d68d0c-5918-4ede-a519-aff107d2ce1c,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5,
fd597d68-f9be-48a0-9656-b7d0b2e140be

result, different first party: 
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same first party: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb,
42c2adb9-619b-4b8f-a843-84ede2df77e1,
8b52f9c3-d53f-4dad-91bb-5e162d085e46,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f,
57161958-02cf-461a-8b94-db8e32150e06

result, different first party: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40,
142f7096-d2d9-4c89-ba22-d0a3b8055308,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346

result, different first party: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same first party: 
784004e0-9aeb-47e5-9c5f-12b883e353ac,
eab30b13-9ecc-48af-9a1a-cae23815efbc,
a652f390-6821-4125-9fd0-c118d2a1e7a2,
09045837-46dd-45c8-ad94-7b7c171aeef0,
7cb67be0-5127-4489-aeb1-7164e3b5af92

result, different first party: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same first party: 
79152685-f997-47b4-aef1-9e391daef62f,
3863988a-95b2-4b54-a63c-ce0727dc2084,
218debd1-a05b-48c0-b827-e4249b90263c,
6d855fa7-5396-4b31-ac62-f63859f5a3ef,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75

result, different first party: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same first party: 
0a2e161c-e0f2-4394-8989-cbcb8cc210b9,
51502f82-4cca-4007-a857-3c34c3db9bf8,
37efbb8c-8f64-4994-8d72-bf40bf93d2bb,
3b85b9c3-ce06-4354-ab1d-998c750dab4a,
70cc2ae2-5861-4824-85db-4f905ce0c785

result, different first party: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same first party: 
92ee92f4-0792-4f9a-ba56-45989cd80572,
606b3a57-c3cf-45fe-a079-3b1566f6abd1,
d40bde18-63e0-419d-b233-1dc09c4c5f94,
df73d3b8-d807-46cf-9f77-65d30bc06fa4,
ba872d52-1a40-47cf-9575-e78f927ff5b3

result, different first party: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same first party: undefined

result, different first party: undefined

unsupported: false

passed: undefined

test failed: true
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same first party: 
37c72365-5852-42a4-9c27-47b4aa879b24,
323291dd-ef8a-484c-aaf9-641497dea358,
bd429ec2-5e93-4afc-8a05-5781fb0abad6

result, different first party: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same first party: 
3a254b81-9486-4572-a38a-b4bd82252b3e,
c0ee6f27-d50e-4975-9a0a-523696e05cbf,
89d605d6-80b6-488e-8454-c4de9c5dfe6b,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6,
d0752b25-d241-400f-a69f-21c3ff242a56

result, different first party: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same first party: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078,
fad257e3-bcc3-4aa7-b541-2b4c42aee488,
3da92bf0-3ab6-4d33-9452-fac73e80ebba,
d276eb0f-b849-45c6-97b9-57596abf3678,
b64d49aa-de07-400a-9631-2935db968706

result, different first party: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
localStorage
The localStorage API gives websites access to a key-value database that will remain available across visits. If the localStorage API is not partitioned or blocked, it can also be used to track users across websites.
              
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same first party: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c,
6e0d0559-9ac2-4abc-9025-582a29e22a8f,
066762a4-4e2b-4344-82a6-b11540d563ab,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same first party: 
ead45dd4-f0d1-4ab5-af71-11d423c35903,
139bd1c9-0bf8-414e-809a-d12f37b34aab,
33d68d0c-5918-4ede-a519-aff107d2ce1c,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5,
fd597d68-f9be-48a0-9656-b7d0b2e140be

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same first party: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb,
42c2adb9-619b-4b8f-a843-84ede2df77e1,
8b52f9c3-d53f-4dad-91bb-5e162d085e46,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f,
57161958-02cf-461a-8b94-db8e32150e06

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40,
142f7096-d2d9-4c89-ba22-d0a3b8055308,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same first party: 
784004e0-9aeb-47e5-9c5f-12b883e353ac,
eab30b13-9ecc-48af-9a1a-cae23815efbc,
a652f390-6821-4125-9fd0-c118d2a1e7a2,
09045837-46dd-45c8-ad94-7b7c171aeef0,
7cb67be0-5127-4489-aeb1-7164e3b5af92

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same first party: 
79152685-f997-47b4-aef1-9e391daef62f,
3863988a-95b2-4b54-a63c-ce0727dc2084,
218debd1-a05b-48c0-b827-e4249b90263c,
6d855fa7-5396-4b31-ac62-f63859f5a3ef,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same first party: 
0a2e161c-e0f2-4394-8989-cbcb8cc210b9,
51502f82-4cca-4007-a857-3c34c3db9bf8,
37efbb8c-8f64-4994-8d72-bf40bf93d2bb,
3b85b9c3-ce06-4354-ab1d-998c750dab4a,
70cc2ae2-5861-4824-85db-4f905ce0c785

result, different first party: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same first party: 
92ee92f4-0792-4f9a-ba56-45989cd80572,
606b3a57-c3cf-45fe-a079-3b1566f6abd1,
d40bde18-63e0-419d-b233-1dc09c4c5f94,
df73d3b8-d807-46cf-9f77-65d30bc06fa4,
ba872d52-1a40-47cf-9575-e78f927ff5b3

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same first party: null

result, different first party: null

unsupported: false

passed: undefined

test failed: true
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same first party: 
37c72365-5852-42a4-9c27-47b4aa879b24,
323291dd-ef8a-484c-aaf9-641497dea358,
bd429ec2-5e93-4afc-8a05-5781fb0abad6

result, different first party: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same first party: 
3a254b81-9486-4572-a38a-b4bd82252b3e,
c0ee6f27-d50e-4975-9a0a-523696e05cbf,
89d605d6-80b6-488e-8454-c4de9c5dfe6b,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6,
d0752b25-d241-400f-a69f-21c3ff242a56

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same first party: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078,
fad257e3-bcc3-4aa7-b541-2b4c42aee488,
3da92bf0-3ab6-4d33-9452-fac73e80ebba,
d276eb0f-b849-45c6-97b9-57596abf3678,
b64d49aa-de07-400a-9631-2935db968706

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
locks
navigator.locks (only supported in some browsers) allows scripts on multiple tabs to coordinate. If this API is not partitioned, it can be used for cross-site tracking.
              
Passed
write: async (key) => {
      if (navigator.locks) {
        navigator.locks.request(key, lock => new Promise((f,r) => {}));
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].clientId;
      } else {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      if (navigator.locks) {
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].name;
      }
    }

result, same first party: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c,
6e0d0559-9ac2-4abc-9025-582a29e22a8f,
066762a4-4e2b-4344-82a6-b11540d563ab,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2

result, different first party: 
Error: The request was denied.,
Error: The request was denied.,
Error: The request was denied.,
Error: The request was denied.,
Error: The request was denied.

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      if (navigator.locks) {
        navigator.locks.request(key, lock => new Promise((f,r) => {}));
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].clientId;
      } else {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      if (navigator.locks) {
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].name;
      }
    }

result, same first party: 
ead45dd4-f0d1-4ab5-af71-11d423c35903,
139bd1c9-0bf8-414e-809a-d12f37b34aab,
33d68d0c-5918-4ede-a519-aff107d2ce1c,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5,
fd597d68-f9be-48a0-9656-b7d0b2e140be

result, different first party: 
Error: The request was denied.,
Error: The request was denied.,
Error: The request was denied.,
Error: The request was denied.,
Error: The request was denied.

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      if (navigator.locks) {
        navigator.locks.request(key, lock => new Promise((f,r) => {}));
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].clientId;
      } else {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      if (navigator.locks) {
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].name;
      }
    }

result, same first party: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb,
42c2adb9-619b-4b8f-a843-84ede2df77e1,
8b52f9c3-d53f-4dad-91bb-5e162d085e46,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f,
57161958-02cf-461a-8b94-db8e32150e06

result, different first party: 
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      if (navigator.locks) {
        navigator.locks.request(key, lock => new Promise((f,r) => {}));
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].clientId;
      } else {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      if (navigator.locks) {
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].name;
      }
    }

result, same first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40,
142f7096-d2d9-4c89-ba22-d0a3b8055308,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346

result, different first party: 
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      if (navigator.locks) {
        navigator.locks.request(key, lock => new Promise((f,r) => {}));
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].clientId;
      } else {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      if (navigator.locks) {
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].name;
      }
    }

result, same first party: 
784004e0-9aeb-47e5-9c5f-12b883e353ac,
eab30b13-9ecc-48af-9a1a-cae23815efbc,
a652f390-6821-4125-9fd0-c118d2a1e7a2,
09045837-46dd-45c8-ad94-7b7c171aeef0,
7cb67be0-5127-4489-aeb1-7164e3b5af92

result, different first party: 
Error: LockManager.query: query() is not allowed in this context,
Error: LockManager.query: query() is not allowed in this context,
Error: LockManager.query: query() is not allowed in this context,
Error: LockManager.query: query() is not allowed in this context,
Error: LockManager.query: query() is not allowed in this context

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      if (navigator.locks) {
        navigator.locks.request(key, lock => new Promise((f,r) => {}));
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].clientId;
      } else {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      if (navigator.locks) {
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].name;
      }
    }

result, same first party: 
79152685-f997-47b4-aef1-9e391daef62f,
3863988a-95b2-4b54-a63c-ce0727dc2084,
218debd1-a05b-48c0-b827-e4249b90263c,
6d855fa7-5396-4b31-ac62-f63859f5a3ef,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75

result, different first party: 
Error: LockManager.query: query() is not allowed in this context,
Error: LockManager.query: query() is not allowed in this context,
Error: LockManager.query: query() is not allowed in this context,
Error: LockManager.query: query() is not allowed in this context,
Error: LockManager.query: query() is not allowed in this context

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      if (navigator.locks) {
        navigator.locks.request(key, lock => new Promise((f,r) => {}));
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].clientId;
      } else {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      if (navigator.locks) {
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].name;
      }
    }

result, same first party: 
0a2e161c-e0f2-4394-8989-cbcb8cc210b9,
51502f82-4cca-4007-a857-3c34c3db9bf8,
37efbb8c-8f64-4994-8d72-bf40bf93d2bb,
3b85b9c3-ce06-4354-ab1d-998c750dab4a,
70cc2ae2-5861-4824-85db-4f905ce0c785

result, different first party: 
Error: LockManager.query: query() is not allowed in this context,
Error: LockManager.query: query() is not allowed in this context,
Error: LockManager.query: query() is not allowed in this context,
Error: LockManager.query: query() is not allowed in this context,
Error: LockManager.query: query() is not allowed in this context

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      if (navigator.locks) {
        navigator.locks.request(key, lock => new Promise((f,r) => {}));
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].clientId;
      } else {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      if (navigator.locks) {
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].name;
      }
    }

result, same first party: 
92ee92f4-0792-4f9a-ba56-45989cd80572,
606b3a57-c3cf-45fe-a079-3b1566f6abd1,
d40bde18-63e0-419d-b233-1dc09c4c5f94,
df73d3b8-d807-46cf-9f77-65d30bc06fa4,
ba872d52-1a40-47cf-9575-e78f927ff5b3

result, different first party: 
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      if (navigator.locks) {
        navigator.locks.request(key, lock => new Promise((f,r) => {}));
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].clientId;
      } else {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      if (navigator.locks) {
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].name;
      }
    }

result, same first party: Error: undefined is not an object (evaluating 'queryResult.held[0].name')

result, different first party: Error: undefined is not an object (evaluating 'queryResult.held[0].name')

unsupported: false

passed: undefined

test failed: true
Passed
write: async (key) => {
      if (navigator.locks) {
        navigator.locks.request(key, lock => new Promise((f,r) => {}));
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].clientId;
      } else {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      if (navigator.locks) {
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].name;
      }
    }

result, same first party: 
37c72365-5852-42a4-9c27-47b4aa879b24,
323291dd-ef8a-484c-aaf9-641497dea358,
bd429ec2-5e93-4afc-8a05-5781fb0abad6

result, different first party: 
Error: LockManager.query: query() is not allowed in this context,
Error: LockManager.query: query() is not allowed in this context,
Error: LockManager.query: query() is not allowed in this context

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (key) => {
      if (navigator.locks) {
        navigator.locks.request(key, lock => new Promise((f,r) => {}));
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].clientId;
      } else {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      if (navigator.locks) {
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].name;
      }
    }

result, same first party: 
3a254b81-9486-4572-a38a-b4bd82252b3e,
c0ee6f27-d50e-4975-9a0a-523696e05cbf,
89d605d6-80b6-488e-8454-c4de9c5dfe6b,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6,
d0752b25-d241-400f-a69f-21c3ff242a56

result, different first party: 
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      if (navigator.locks) {
        navigator.locks.request(key, lock => new Promise((f,r) => {}));
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].clientId;
      } else {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      if (navigator.locks) {
        let queryResult = await navigator.locks.query();
        return queryResult.held[0].name;
      }
    }

result, same first party: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078,
fad257e3-bcc3-4aa7-b541-2b4c42aee488,
3da92bf0-3ab6-4d33-9452-fac73e80ebba,
d276eb0f-b849-45c6-97b9-57596abf3678,
b64d49aa-de07-400a-9631-2935db968706

result, different first party: 
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name'),
Error: Cannot read properties of undefined (reading 'name')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
prefetch cache
A <link rel='prefetch'...> suggests to browsers they should fetch a resource ahead of time and cache it. But if browsers don't partition this cache, it can be used to track users across websites.
              
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same first party: 
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received

result, different first party: 
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same first party: 
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received

result, different first party: 
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same first party: Error: No requests received

result, different first party: Error: No requests received

unsupported: true

passed: undefined

test failed: false
Unsupported
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same first party: 
Error: No requests received,
Error: No requests received,
Error: No requests received

result, different first party: 
Error: No requests received,
Error: No requests received,
Error: No requests received

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
ServiceWorker
The ServiceWorker API allows websites to run code in the background and store content in the browser for offline use. If a ServiceWorker can be accessed from multiple websites, it can be abused to track users across sites.
              
Passed
write: async (key) => {
      if (!navigator.serviceWorker) {
        throw new Error("Unsupported");
      }
      let registration = await navigator.serviceWorker.register(
        'serviceWorker.js');
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      await fetch(`serviceworker-write?secret=${key}`);
    }

read: async () => {
      console.log("trying to register the serviceworker now...");
      const registration = await Promise.race([
        navigator.serviceWorker.register('serviceWorker.js'),
        sleepMs(500)
      ]);
      if (registration === undefined) {
        // We timed out or otherwise failed.
        throw new Error("ServiceWorker registration failed");
      }
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      let response = await fetch("serviceworker-read");
      return await response.text();
    }

result, same first party: 
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->


result, different first party: 
Error: Failed to register a ServiceWorker for scope ('https://test-pages.privacytests2.org/') with script ('https://test-pages.privacytests2.org/serviceWorker.js'): The user denied permission to use Service Worker.,
Error: Failed to register a ServiceWorker for scope ('https://test-pages.privacytests2.org/') with script ('https://test-pages.privacytests2.org/serviceWorker.js'): The user denied permission to use Service Worker.,
Error: Failed to register a ServiceWorker for scope ('https://test-pages.privacytests2.org/') with script ('https://test-pages.privacytests2.org/serviceWorker.js'): The user denied permission to use Service Worker.,
Error: Failed to register a ServiceWorker for scope ('https://test-pages.privacytests2.org/') with script ('https://test-pages.privacytests2.org/serviceWorker.js'): The user denied permission to use Service Worker.,
Error: Failed to register a ServiceWorker for scope ('https://test-pages.privacytests2.org/') with script ('https://test-pages.privacytests2.org/serviceWorker.js'): The user denied permission to use Service Worker.

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      if (!navigator.serviceWorker) {
        throw new Error("Unsupported");
      }
      let registration = await navigator.serviceWorker.register(
        'serviceWorker.js');
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      await fetch(`serviceworker-write?secret=${key}`);
    }

read: async () => {
      console.log("trying to register the serviceworker now...");
      const registration = await Promise.race([
        navigator.serviceWorker.register('serviceWorker.js'),
        sleepMs(500)
      ]);
      if (registration === undefined) {
        // We timed out or otherwise failed.
        throw new Error("ServiceWorker registration failed");
      }
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      let response = await fetch("serviceworker-read");
      return await response.text();
    }

result, same first party: 
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->


result, different first party: 
Error: Failed to register a ServiceWorker for scope ('https://test-pages.privacytests2.org/') with script ('https://test-pages.privacytests2.org/serviceWorker.js'): The user denied permission to use Service Worker.,
Error: Failed to register a ServiceWorker for scope ('https://test-pages.privacytests2.org/') with script ('https://test-pages.privacytests2.org/serviceWorker.js'): The user denied permission to use Service Worker.,
Error: Failed to register a ServiceWorker for scope ('https://test-pages.privacytests2.org/') with script ('https://test-pages.privacytests2.org/serviceWorker.js'): The user denied permission to use Service Worker.,
Error: Failed to register a ServiceWorker for scope ('https://test-pages.privacytests2.org/') with script ('https://test-pages.privacytests2.org/serviceWorker.js'): The user denied permission to use Service Worker.,
Error: Failed to register a ServiceWorker for scope ('https://test-pages.privacytests2.org/') with script ('https://test-pages.privacytests2.org/serviceWorker.js'): The user denied permission to use Service Worker.

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      if (!navigator.serviceWorker) {
        throw new Error("Unsupported");
      }
      let registration = await navigator.serviceWorker.register(
        'serviceWorker.js');
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      await fetch(`serviceworker-write?secret=${key}`);
    }

read: async () => {
      console.log("trying to register the serviceworker now...");
      const registration = await Promise.race([
        navigator.serviceWorker.register('serviceWorker.js'),
        sleepMs(500)
      ]);
      if (registration === undefined) {
        // We timed out or otherwise failed.
        throw new Error("ServiceWorker registration failed");
      }
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      let response = await fetch("serviceworker-read");
      return await response.text();
    }

result, same first party: 
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->


result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      if (!navigator.serviceWorker) {
        throw new Error("Unsupported");
      }
      let registration = await navigator.serviceWorker.register(
        'serviceWorker.js');
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      await fetch(`serviceworker-write?secret=${key}`);
    }

read: async () => {
      console.log("trying to register the serviceworker now...");
      const registration = await Promise.race([
        navigator.serviceWorker.register('serviceWorker.js'),
        sleepMs(500)
      ]);
      if (registration === undefined) {
        // We timed out or otherwise failed.
        throw new Error("ServiceWorker registration failed");
      }
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      let response = await fetch("serviceworker-read");
      return await response.text();
    }

result, same first party: 
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->


result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!navigator.serviceWorker) {
        throw new Error("Unsupported");
      }
      let registration = await navigator.serviceWorker.register(
        'serviceWorker.js');
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      await fetch(`serviceworker-write?secret=${key}`);
    }

read: async () => {
      console.log("trying to register the serviceworker now...");
      const registration = await Promise.race([
        navigator.serviceWorker.register('serviceWorker.js'),
        sleepMs(500)
      ]);
      if (registration === undefined) {
        // We timed out or otherwise failed.
        throw new Error("ServiceWorker registration failed");
      }
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      let response = await fetch("serviceworker-read");
      return await response.text();
    }

result, same first party: 
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined

result, different first party: 
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!navigator.serviceWorker) {
        throw new Error("Unsupported");
      }
      let registration = await navigator.serviceWorker.register(
        'serviceWorker.js');
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      await fetch(`serviceworker-write?secret=${key}`);
    }

read: async () => {
      console.log("trying to register the serviceworker now...");
      const registration = await Promise.race([
        navigator.serviceWorker.register('serviceWorker.js'),
        sleepMs(500)
      ]);
      if (registration === undefined) {
        // We timed out or otherwise failed.
        throw new Error("ServiceWorker registration failed");
      }
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      let response = await fetch("serviceworker-read");
      return await response.text();
    }

result, same first party: 
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined

result, different first party: 
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!navigator.serviceWorker) {
        throw new Error("Unsupported");
      }
      let registration = await navigator.serviceWorker.register(
        'serviceWorker.js');
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      await fetch(`serviceworker-write?secret=${key}`);
    }

read: async () => {
      console.log("trying to register the serviceworker now...");
      const registration = await Promise.race([
        navigator.serviceWorker.register('serviceWorker.js'),
        sleepMs(500)
      ]);
      if (registration === undefined) {
        // We timed out or otherwise failed.
        throw new Error("ServiceWorker registration failed");
      }
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      let response = await fetch("serviceworker-read");
      return await response.text();
    }

result, same first party: 
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined

result, different first party: 
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: async (key) => {
      if (!navigator.serviceWorker) {
        throw new Error("Unsupported");
      }
      let registration = await navigator.serviceWorker.register(
        'serviceWorker.js');
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      await fetch(`serviceworker-write?secret=${key}`);
    }

read: async () => {
      console.log("trying to register the serviceworker now...");
      const registration = await Promise.race([
        navigator.serviceWorker.register('serviceWorker.js'),
        sleepMs(500)
      ]);
      if (registration === undefined) {
        // We timed out or otherwise failed.
        throw new Error("ServiceWorker registration failed");
      }
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      let response = await fetch("serviceworker-read");
      return await response.text();
    }

result, same first party: 
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->


result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      if (!navigator.serviceWorker) {
        throw new Error("Unsupported");
      }
      let registration = await navigator.serviceWorker.register(
        'serviceWorker.js');
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      await fetch(`serviceworker-write?secret=${key}`);
    }

read: async () => {
      console.log("trying to register the serviceworker now...");
      const registration = await Promise.race([
        navigator.serviceWorker.register('serviceWorker.js'),
        sleepMs(500)
      ]);
      if (registration === undefined) {
        // We timed out or otherwise failed.
        throw new Error("ServiceWorker registration failed");
      }
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      let response = await fetch("serviceworker-read");
      return await response.text();
    }

result, same first party: 

result, different first party: 

unsupported: false

passed: undefined

test failed: true
Unsupported
write: async (key) => {
      if (!navigator.serviceWorker) {
        throw new Error("Unsupported");
      }
      let registration = await navigator.serviceWorker.register(
        'serviceWorker.js');
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      await fetch(`serviceworker-write?secret=${key}`);
    }

read: async () => {
      console.log("trying to register the serviceworker now...");
      const registration = await Promise.race([
        navigator.serviceWorker.register('serviceWorker.js'),
        sleepMs(500)
      ]);
      if (registration === undefined) {
        // We timed out or otherwise failed.
        throw new Error("ServiceWorker registration failed");
      }
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      let response = await fetch("serviceworker-read");
      return await response.text();
    }

result, same first party: 
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined

result, different first party: 
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined,
Error: navigator.serviceWorker is undefined

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Passed
write: async (key) => {
      if (!navigator.serviceWorker) {
        throw new Error("Unsupported");
      }
      let registration = await navigator.serviceWorker.register(
        'serviceWorker.js');
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      await fetch(`serviceworker-write?secret=${key}`);
    }

read: async () => {
      console.log("trying to register the serviceworker now...");
      const registration = await Promise.race([
        navigator.serviceWorker.register('serviceWorker.js'),
        sleepMs(500)
      ]);
      if (registration === undefined) {
        // We timed out or otherwise failed.
        throw new Error("ServiceWorker registration failed");
      }
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      let response = await fetch("serviceworker-read");
      return await response.text();
    }

result, same first party: 
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->


result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      if (!navigator.serviceWorker) {
        throw new Error("Unsupported");
      }
      let registration = await navigator.serviceWorker.register(
        'serviceWorker.js');
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      await fetch(`serviceworker-write?secret=${key}`);
    }

read: async () => {
      console.log("trying to register the serviceworker now...");
      const registration = await Promise.race([
        navigator.serviceWorker.register('serviceWorker.js'),
        sleepMs(500)
      ]);
      if (registration === undefined) {
        // We timed out or otherwise failed.
        throw new Error("ServiceWorker registration failed");
      }
      console.log(registration);
      await navigator.serviceWorker.ready;
      console.log("service worker ready");
      await sleepMs(100);
      let response = await fetch("serviceworker-read");
      return await response.text();
    }

result, same first party: 
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
,
<html>
<head><title>404 Not Found</title></head>
<body>
<center><h1>404 Not Found</h1></center>
<hr><center>nginx/1.18.0 (Ubuntu)</center>
</body>
</html>
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->
<!-- a padding to disable MSIE and Chrome friendly error page -->


result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
SharedWorker
The SharedWorker API allows scripts from multiple tabs to share a background thread of computation. If SharedWorker is not partitioned, then it can be abused to shared data between websites in your browser.
              
Passed
write: async (secret) => {
      try {
        let worker = new SharedWorker("supercookies_sharedworker.js");
        worker.port.start();
//        console.log("worker", worker);
        const messagePromise = new Promise((resolve) => {
          worker.port.onmessage = (e) => resolve(e.data);
        });
        worker.port.postMessage(secret);
        await messagePromise;
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let worker = new SharedWorker("supercookies_sharedworker.js");
      worker.port.start();
      const messagePromise = new Promise((resolve, reject) => {
        worker.port.onmessage = (e) => resolve(e.data);
        setTimeout(() => reject(new Error("no SharedWorker message received")), 200);
      });
      worker.port.postMessage("request");
      const message = await messagePromise;
      if (message === "none") {
        throw new Error("Unsupported");
      }
      return message;
    }

result, same first party: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c,
6e0d0559-9ac2-4abc-9025-582a29e22a8f,
066762a4-4e2b-4344-82a6-b11540d563ab,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2

result, different first party: 
Error: no SharedWorker message received,
Error: no SharedWorker message received,
Error: no SharedWorker message received,
Error: no SharedWorker message received,
Error: no SharedWorker message received

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        let worker = new SharedWorker("supercookies_sharedworker.js");
        worker.port.start();
//        console.log("worker", worker);
        const messagePromise = new Promise((resolve) => {
          worker.port.onmessage = (e) => resolve(e.data);
        });
        worker.port.postMessage(secret);
        await messagePromise;
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let worker = new SharedWorker("supercookies_sharedworker.js");
      worker.port.start();
      const messagePromise = new Promise((resolve, reject) => {
        worker.port.onmessage = (e) => resolve(e.data);
        setTimeout(() => reject(new Error("no SharedWorker message received")), 200);
      });
      worker.port.postMessage("request");
      const message = await messagePromise;
      if (message === "none") {
        throw new Error("Unsupported");
      }
      return message;
    }

result, same first party: 
ead45dd4-f0d1-4ab5-af71-11d423c35903,
139bd1c9-0bf8-414e-809a-d12f37b34aab,
33d68d0c-5918-4ede-a519-aff107d2ce1c,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5,
fd597d68-f9be-48a0-9656-b7d0b2e140be

result, different first party: 
Error: no SharedWorker message received,
Error: no SharedWorker message received,
Error: no SharedWorker message received,
Error: no SharedWorker message received,
Error: no SharedWorker message received

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        let worker = new SharedWorker("supercookies_sharedworker.js");
        worker.port.start();
//        console.log("worker", worker);
        const messagePromise = new Promise((resolve) => {
          worker.port.onmessage = (e) => resolve(e.data);
        });
        worker.port.postMessage(secret);
        await messagePromise;
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let worker = new SharedWorker("supercookies_sharedworker.js");
      worker.port.start();
      const messagePromise = new Promise((resolve, reject) => {
        worker.port.onmessage = (e) => resolve(e.data);
        setTimeout(() => reject(new Error("no SharedWorker message received")), 200);
      });
      worker.port.postMessage("request");
      const message = await messagePromise;
      if (message === "none") {
        throw new Error("Unsupported");
      }
      return message;
    }

result, same first party: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb,
42c2adb9-619b-4b8f-a843-84ede2df77e1,
8b52f9c3-d53f-4dad-91bb-5e162d085e46,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f,
57161958-02cf-461a-8b94-db8e32150e06

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        let worker = new SharedWorker("supercookies_sharedworker.js");
        worker.port.start();
//        console.log("worker", worker);
        const messagePromise = new Promise((resolve) => {
          worker.port.onmessage = (e) => resolve(e.data);
        });
        worker.port.postMessage(secret);
        await messagePromise;
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let worker = new SharedWorker("supercookies_sharedworker.js");
      worker.port.start();
      const messagePromise = new Promise((resolve, reject) => {
        worker.port.onmessage = (e) => resolve(e.data);
        setTimeout(() => reject(new Error("no SharedWorker message received")), 200);
      });
      worker.port.postMessage("request");
      const message = await messagePromise;
      if (message === "none") {
        throw new Error("Unsupported");
      }
      return message;
    }

result, same first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40,
142f7096-d2d9-4c89-ba22-d0a3b8055308,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        let worker = new SharedWorker("supercookies_sharedworker.js");
        worker.port.start();
//        console.log("worker", worker);
        const messagePromise = new Promise((resolve) => {
          worker.port.onmessage = (e) => resolve(e.data);
        });
        worker.port.postMessage(secret);
        await messagePromise;
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let worker = new SharedWorker("supercookies_sharedworker.js");
      worker.port.start();
      const messagePromise = new Promise((resolve, reject) => {
        worker.port.onmessage = (e) => resolve(e.data);
        setTimeout(() => reject(new Error("no SharedWorker message received")), 200);
      });
      worker.port.postMessage("request");
      const message = await messagePromise;
      if (message === "none") {
        throw new Error("Unsupported");
      }
      return message;
    }

result, same first party: 
784004e0-9aeb-47e5-9c5f-12b883e353ac,
eab30b13-9ecc-48af-9a1a-cae23815efbc,
a652f390-6821-4125-9fd0-c118d2a1e7a2,
09045837-46dd-45c8-ad94-7b7c171aeef0,
7cb67be0-5127-4489-aeb1-7164e3b5af92

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        let worker = new SharedWorker("supercookies_sharedworker.js");
        worker.port.start();
//        console.log("worker", worker);
        const messagePromise = new Promise((resolve) => {
          worker.port.onmessage = (e) => resolve(e.data);
        });
        worker.port.postMessage(secret);
        await messagePromise;
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let worker = new SharedWorker("supercookies_sharedworker.js");
      worker.port.start();
      const messagePromise = new Promise((resolve, reject) => {
        worker.port.onmessage = (e) => resolve(e.data);
        setTimeout(() => reject(new Error("no SharedWorker message received")), 200);
      });
      worker.port.postMessage("request");
      const message = await messagePromise;
      if (message === "none") {
        throw new Error("Unsupported");
      }
      return message;
    }

result, same first party: 
79152685-f997-47b4-aef1-9e391daef62f,
3863988a-95b2-4b54-a63c-ce0727dc2084,
218debd1-a05b-48c0-b827-e4249b90263c,
6d855fa7-5396-4b31-ac62-f63859f5a3ef,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        let worker = new SharedWorker("supercookies_sharedworker.js");
        worker.port.start();
//        console.log("worker", worker);
        const messagePromise = new Promise((resolve) => {
          worker.port.onmessage = (e) => resolve(e.data);
        });
        worker.port.postMessage(secret);
        await messagePromise;
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let worker = new SharedWorker("supercookies_sharedworker.js");
      worker.port.start();
      const messagePromise = new Promise((resolve, reject) => {
        worker.port.onmessage = (e) => resolve(e.data);
        setTimeout(() => reject(new Error("no SharedWorker message received")), 200);
      });
      worker.port.postMessage("request");
      const message = await messagePromise;
      if (message === "none") {
        throw new Error("Unsupported");
      }
      return message;
    }

result, same first party: 
0a2e161c-e0f2-4394-8989-cbcb8cc210b9,
51502f82-4cca-4007-a857-3c34c3db9bf8,
37efbb8c-8f64-4994-8d72-bf40bf93d2bb,
3b85b9c3-ce06-4354-ab1d-998c750dab4a,
70cc2ae2-5861-4824-85db-4f905ce0c785

result, different first party: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        let worker = new SharedWorker("supercookies_sharedworker.js");
        worker.port.start();
//        console.log("worker", worker);
        const messagePromise = new Promise((resolve) => {
          worker.port.onmessage = (e) => resolve(e.data);
        });
        worker.port.postMessage(secret);
        await messagePromise;
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let worker = new SharedWorker("supercookies_sharedworker.js");
      worker.port.start();
      const messagePromise = new Promise((resolve, reject) => {
        worker.port.onmessage = (e) => resolve(e.data);
        setTimeout(() => reject(new Error("no SharedWorker message received")), 200);
      });
      worker.port.postMessage("request");
      const message = await messagePromise;
      if (message === "none") {
        throw new Error("Unsupported");
      }
      return message;
    }

result, same first party: 
92ee92f4-0792-4f9a-ba56-45989cd80572,
606b3a57-c3cf-45fe-a079-3b1566f6abd1,
d40bde18-63e0-419d-b233-1dc09c4c5f94,
df73d3b8-d807-46cf-9f77-65d30bc06fa4,
ba872d52-1a40-47cf-9575-e78f927ff5b3

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      try {
        let worker = new SharedWorker("supercookies_sharedworker.js");
        worker.port.start();
//        console.log("worker", worker);
        const messagePromise = new Promise((resolve) => {
          worker.port.onmessage = (e) => resolve(e.data);
        });
        worker.port.postMessage(secret);
        await messagePromise;
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let worker = new SharedWorker("supercookies_sharedworker.js");
      worker.port.start();
      const messagePromise = new Promise((resolve, reject) => {
        worker.port.onmessage = (e) => resolve(e.data);
        setTimeout(() => reject(new Error("no SharedWorker message received")), 200);
      });
      worker.port.postMessage("request");
      const message = await messagePromise;
      if (message === "none") {
        throw new Error("Unsupported");
      }
      return message;
    }

result, same first party: Error: Unsupported

result, different first party: Error: Unsupported

unsupported: true

passed: undefined

test failed: false
Passed
write: async (secret) => {
      try {
        let worker = new SharedWorker("supercookies_sharedworker.js");
        worker.port.start();
//        console.log("worker", worker);
        const messagePromise = new Promise((resolve) => {
          worker.port.onmessage = (e) => resolve(e.data);
        });
        worker.port.postMessage(secret);
        await messagePromise;
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let worker = new SharedWorker("supercookies_sharedworker.js");
      worker.port.start();
      const messagePromise = new Promise((resolve, reject) => {
        worker.port.onmessage = (e) => resolve(e.data);
        setTimeout(() => reject(new Error("no SharedWorker message received")), 200);
      });
      worker.port.postMessage("request");
      const message = await messagePromise;
      if (message === "none") {
        throw new Error("Unsupported");
      }
      return message;
    }

result, same first party: 
37c72365-5852-42a4-9c27-47b4aa879b24,
323291dd-ef8a-484c-aaf9-641497dea358,
bd429ec2-5e93-4afc-8a05-5781fb0abad6

result, different first party: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (secret) => {
      try {
        let worker = new SharedWorker("supercookies_sharedworker.js");
        worker.port.start();
//        console.log("worker", worker);
        const messagePromise = new Promise((resolve) => {
          worker.port.onmessage = (e) => resolve(e.data);
        });
        worker.port.postMessage(secret);
        await messagePromise;
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let worker = new SharedWorker("supercookies_sharedworker.js");
      worker.port.start();
      const messagePromise = new Promise((resolve, reject) => {
        worker.port.onmessage = (e) => resolve(e.data);
        setTimeout(() => reject(new Error("no SharedWorker message received")), 200);
      });
      worker.port.postMessage("request");
      const message = await messagePromise;
      if (message === "none") {
        throw new Error("Unsupported");
      }
      return message;
    }

result, same first party: 
3a254b81-9486-4572-a38a-b4bd82252b3e,
c0ee6f27-d50e-4975-9a0a-523696e05cbf,
89d605d6-80b6-488e-8454-c4de9c5dfe6b,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6,
d0752b25-d241-400f-a69f-21c3ff242a56

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        let worker = new SharedWorker("supercookies_sharedworker.js");
        worker.port.start();
//        console.log("worker", worker);
        const messagePromise = new Promise((resolve) => {
          worker.port.onmessage = (e) => resolve(e.data);
        });
        worker.port.postMessage(secret);
        await messagePromise;
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let worker = new SharedWorker("supercookies_sharedworker.js");
      worker.port.start();
      const messagePromise = new Promise((resolve, reject) => {
        worker.port.onmessage = (e) => resolve(e.data);
        setTimeout(() => reject(new Error("no SharedWorker message received")), 200);
      });
      worker.port.postMessage("request");
      const message = await messagePromise;
      if (message === "none") {
        throw new Error("Unsupported");
      }
      return message;
    }

result, same first party: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078,
fad257e3-bcc3-4aa7-b541-2b4c42aee488,
3da92bf0-3ab6-4d33-9452-fac73e80ebba,
d276eb0f-b849-45c6-97b9-57596abf3678,
b64d49aa-de07-400a-9631-2935db968706

result, different first party: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
TLS Session ID
The TLS protocol is used by HTTPS to make connections secure. If the browser were to re-use a TLS session, then the session ID could be used to track users across websites.
              
Passed
write: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

read: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

result, same first party: 
ab3a73fde83549f516bcf1066eada28ff27566347ccd200c3e4e702090f407ac,
18d915897cecc640bbe32626af6fcab4499786089b6a375823bff0763001b20f,
239f5aeac5981c0adb6af2ae5f7ef8241186a92e10f8a8de791e28194270d3ae,
5ab9a36c70683b39dc9dcf5275c4405f6b044e68c2543f9e19df3406b47e4de9,
b38ac5783c1a9496424ad8134939acca5cf01d831e3dc5407a86866765937522

result, different first party: 
16beca756a872f5f659e282371b674226d21749df3320c792b9f1cab4b23c9c6,
3066a10bf12a7f57d7fafc790b6f51a519a49cd9798dfb3eb10a0cebaf25e28b,
04ea4e2fedac57bfa80289cfc8e79b1dc7b1ace2949802d23a98d5cdc108acbc,
39ffb7f6126172e0ae6afb79b771ba5f93f84673e236aee71705bfc7aa00eff4,
6447cc771d1d9e6ed37cf651e6fc7b754080a8a282fd8f332967c5c831c59600

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

read: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

result, same first party: 
3bd9cd0127a4f55a580f166baf4d2ee0cdde51b11f422f45817d009d5a6cee32,
1f1dae7236ee7aff1ae8cd55b13972485b54c8939b27621a1415caafbbbc3901,
e9a416f1533b8b14a3219a7aba0ef7fc5b992a92db68390ed5282590d27cd953,
b441c84aa61b66e210d9ec042af325f04274b32c0cb3ff9cc532792e6d3b7628,
238337c2a640e9d04c829707f795952dd507df3bf1bf70bf763ecb43a6a056fe

result, different first party: 
73e407b23e6c680c28e379754747e9a7ad7440b0bf5ee9d855fb3b1752761915,
59015b3dc610584c8710906d0c35d0367de3a19d5e45f43778956346582f9956,
008cfa25d8c2ad2cfb4a6523f49ebdb1d99fa2736299503b2091e45fc68f2269,
1d7dd152b5500c397f212542488be7ab34e7342668f03c3930211d18efce20e3,
51d4ca62537098c9c719ca98d211633cf89f741df57fd08f84ea2fe1d6d61064

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

read: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

result, same first party: 
3801d968117b842db14376885432c5f558eb66823550b19e3d5f508c1682025c,
fa07684eddecf6f1d6a5451d4fc3387b8709c312e6f9bab96c287044fee3c7d0,
c929198f1ec771f6d105e98953c2b749bb94a6bda40ee13ca6930d6ac5cfcc6a,
eb30b2a12b1b39bd17a59f508f2db4c2fa77b0900a5be80ef3be31478babc0cc,
ef6780bb02577643826b2ee7316d5a7c39b145b3f548d92f6cb397705954dde5

result, different first party: 
7c2365be46206dc184f144854df188e398d86fc25073f82fa7624a6e894183c7,
edc34449da3200b3dfd21a1b58503dc2c560f21ef7600f84c7c6f8be686176fd,
c4e7be2b610afe451a0b23af408f507e381be272ed5fde74cb5a68a915455a73,
8d4b415b49a475684fb74337519730f14ca8aef843cd15351ec8653d8657506e,
9a5f183ced398f8d9b681bb47c49b1e8a110bc718d3d03e695477b54291fa494

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

read: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

result, same first party: 
992b97b2df17fe1c6efe284c150cbdfcbe17c6cc50f68f9f846df82aec98a633,
0a727d4c2cb1bacbe78848b998100cca6ab81c9187495cf5f9458503b96993c1,
4e278d0904068efae0657c624495a9182f2fff4509b1fb24f4748d59b9ad3d3f,
dd26a82552b7b1709aebd3191beba5713736e4ec32063e12ef937ffeca2a72e8,
409fa03235f7e2d00dacf7aed943e29544bbe55765d1d701219083935d194028

result, different first party: 
b70adc1b065c4a0454617fbbb5c316365663d50e27a7339023cc152b472b791c,
5e64fc2cba78c313cf86a8eb563a5fd1321fcc7ddb5d1e0efffcd345a3924ed6,
99e6f5e2106fe57aa0ba4f74826768ddb9be6795c8bd48f531e4bd9f735af2ba,
eea52a112527b8bc320576dc6f777b47ff68019e504d1330cdb534f588f5473f,
ec0ccfb1b8799c6feff097c391eabecbfc149f18fea8559545cde82bf8ad1a07

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

read: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

result, same first party: 
dc1de356b1012b7b24b0eceab0e14754e074de5ec57e92e1282dd54f26c5a9e9,
0159262bdf36a3bbd2a2355a97ee25ff1189724456a076f604c1cf5e7ebfcdef,
4239215146ab6137176b2febcca19c41bb24d2eb758623f133b7db032c5e244b,
4692130a34e73c8649f1178fcfa2d7512ea5a6cab65fc78b95f272612de5ed89,
435245b095d57bae1a06c72b01020e43dd7ea8c5c34f89adc63a41b1bb140a18

result, different first party: 
60c3770dfb14b5204343e051a352a3763750c88ef085aa0eeb144fc550279949,
7468b59ec76916dbcc05c19824e8fb3df21f5ac84da4595d48279cb218b0243b,
509e70ead9ebc23e8a6af04a9f709ade863d770218a1ea7c6ee2279216bd9d32,
c5d52d98f315e2cee91459017620d2223c5ad6ae53b4aa5fdb2f062d073ca8c3,
dda398cab755ca92dbb8706c9bcdad1d88b2b6ea359760411a7e52830bb524fc

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

read: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

result, same first party: 
bb32fa849e5f339aedd278ebb5a51e1f759ffc9c0814a9062998485a3de82082,
9624e91de9507159724bb5b53ce054bb6615dbbb7be33b2686a6940c6a0c6464,
8371a2062eac382a783e4552bd96da1b2f014c8530dbcd3597fc4016e5945aa3,
26a20be77d545c9fa36d4b314a1dad294e37f693be8615f62f2bdbe300ddb9bd,
abf6eecd9f3f6b482564328af4b227240f726e914d4abb4390eb66f79961ebbc

result, different first party: 
31e85e5b545c94e4954d82f52d32a12bcd558935db9289bc93f9aa207266b659,
9719b4044aad26828fd04331f7d21e727707ea99f6f6243ac427430bee0d7d8b,
7eae3417b4b9a0344d4caee320a5532b9f3b80c24b82f33a12547685324af228,
4d2f899ed64ce0e9e14933f6ae5f278677d6acae7b6b711ad7b8bfc2d3302179,
cac0cb030e012a11c0f60effe1b55d8ae49c12a9ccda2585a6538246113b1569

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

read: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

result, same first party: 
6f3850a0ac6d86d97a1e5fc5b4760232f7ac87d2dcc91f70abe55fceaa74dfb7,
c4829d0cec043fa2eb41aa73eebdb55ad98cc406892405311b54538b5a889945,
34156eb2ae21766d8968eea0f686dcb4ae1fcb8cc8487c154abb10956f63511c,
ed25ed0fb5cffa1e50afb3e8276d03d551f3de6673b8318e5ef514e518825201,
f10d76d17680b090c52dee53a8a6a5754bc9eb0b35ea354a8779c71e63b34441

result, different first party: 
11f793a9c831f6763b9164d460bf62f331e76027169920fd64945067d5c39fc0,
124a2a618f871ad91c261198904dcf3720a0db46e5da56270fd7e99f081b3a0c,
ddb28c21b929880ab6374aa1486194bf5d7a4cb02261d6b1c0bac0b10bc0239e,
e3f5c78f215648aeb7284aae614ed773a93b00d4531503dd718ad2657329069c,
71b39b2a8aa8bdf86af4d624cb3f3729c0e75f11db1af25b6332a5f5460bd84c

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

read: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

result, same first party: 
b7530c714e7e6fda47b0dc78f6269b1d64ffb77c299c97be93e6dbc8c675943c,
ab0ae9423bae36081a0797572ff798743cc64ebc1232bdcf5f9062cbd1047d3f,
3918fc058235f5a3e1f562da3d7cf1a73c936f5fc9011e98cc3357aa420258ba,
2e7bfb60e572e90ce2347d9352419d83b5235677bda5da0df317fc17e22587a8,
91cd416197dd1697dd703509dea3e822abd91801de8ff4e3a37497cc4eb05587

result, different first party: 
bb5a2e2c0ed080c5b704dad8466590e8d3132b459699d786d74436bf33d1db4c,
f8b11f6c76fe5369c89dd83706a6894c8b659c7680e716713f959fee0c3f1ec9,
81236f52ee9513579985086b8638e32d1ab3f359308a48e918c6a9952eb75415,
8d4c8b43c9086f12cd4e7b9e6294e19cb1ad0523e030767acb7519021e520e89,
50e82ad0068ea2b99be11108aca5ac7961334d3a6caed8c26569db48b31e7737

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

read: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

result, same first party: 8e836357b9d65ee464e8c4bc01358b09f537be1d28192ea511a17802cd79688f

result, different first party: b720f9a1d1ce9a171b2d8720dd0dbf5f365e5484c365f68bf0f4d1787f366acf

unsupported: false

passed: true

test failed: false
Passed
write: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

read: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

result, same first party: 
759560e57d61d8ab2762f9558cec0f11b30e4113db22f9d63e541c3175814c74,
7d2f9a9bffa511c946e4cbd4c2fa1cd993b2c2d48346ca44cb3741c0cfb25b94,
843139bb9ad43b7ee0e7825dd747f6221d69c68e43bed6c0593809c262e14109

result, different first party: 
07d8187cb08baf4e7cddc5af2581c5f84eec60400e2124eb9177dc8f66bff1c9,
3a164cac7e4966180e8eca6b2522e93413f5dbbc79ff5cf878e7a2b21dfb394b,
a9f3ecf7fd4563371a09f1d2cbf46e76980b6c58197f6c2581c060c36b33a5af

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

read: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

result, same first party: 
986b105602ed74c286b58002b29ec758c33f773742046dd86b5d2b2ef017ec66,
cd2774c0745bcc56202eb15343e96470b6e822cb90c0bc67c05f74245420576e,
f57a5c8aa1201fd9bd3ede946d238741e71d06c7704a7f285c46a128ba19a041,
deeb56bd29037703d9f49b272660d1ba1c92c68585d3d04673661b209e2407ae,
ead60b79c6968f53198ec6470b500713373546a3e688817b32f0ef74148876bc

result, different first party: 
2447dfdae79443cb7c4cdc61986d4ea4f7aab8d0f5e902b0fd9fa08f431281cd,
64c69f2d150375471690291c6d56ba7924a1e9d4a16941fda03d4f6c87e00216,
43eba3b1abd1d4fb0d13b505c036115019d2e2aae2cab77ad3c772d68f32408c,
4949449bd62fe48c87f61755b071f16e1a8017a314298b03bc2eb7d8d9341329,
289c68c86679a7bc67e0e9060f00898b5b50b32eda87f0d28d360678422a390a

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

read: async () => {
      let results = await fetch("https://tls.privacytests2.org:8900/");
      return (await results.json()).sessionId;
    }

result, same first party: 
02009f7d6d1687bcd3c57689200410cfe7abb8b7581671e362862c4a4dfaf774,
a81f3cf21758992dd8d840559f72d4817068513127e4ccff901496b0213116a4,
d45b4fb0804f41092dca256f7859aa6c52cdd5f596f15ba6dd015f6aa61a45c7,
5b116593c8cdd00ac413d4cd36cdfc58fb62dddf12dfa4e4826dfb4e59d53559,
74ce885e5aa6f3bc7dc5a77cb0a19d4c41f53c927533506152bdf3e27be1e7a2

result, different first party: 
013a8226f4c687fee85aac1ec5cf45a792fd19ecfef9042bb8259b3c3c83cec1,
63a840b7d8b3835a13d153624ada623519536785fba847551aab8cda6b514935,
428a0b3efd80d60dc56776e25dcb2c5adcec508b618211965ba3a6469659716e,
27f73d57d79586bdca3ce4097ffb212fbd69b5269e5e2908b8c1b522dd747c11,
c1188d5daa94faea874041327245cc7c929a8f57764a2ee73db18d71e3c0ce7f

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Web SQL Database
The Web SQL Database is a deprecated web API for storing data in an SQL database.
              
Passed
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: false, false, false, false, false

passed: undefined

test failed: true, true, true, true, true
Passed
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: false, false, false, false, false

passed: undefined

test failed: true, true, true, true, true
Passed
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: false, false, false, false, false

passed: undefined

test failed: true, true, true, true, true
Passed
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: false, false, false, false, false

passed: undefined

test failed: true, true, true, true, true
Passed
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: false, false, false, false, false

passed: undefined

test failed: true, true, true, true, true
Passed
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: false, false, false, false, false

passed: undefined

test failed: true, true, true, true, true
Passed
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: false, false, false, false, false

passed: undefined

test failed: true, true, true, true, true
Passed
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: false, false, false, false, false

passed: undefined

test failed: true, true, true, true, true
Passed
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same first party: Error: Web SQL is deprecated

result, different first party: Error: Web SQL is deprecated

unsupported: false

passed: undefined

test failed: true
Passed
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: false, false, false

passed: undefined

test failed: true, true, true
Passed
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: false, false, false, false, false

passed: undefined

test failed: true, true, true, true, true
Passed
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different first party: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: false, false, false, false, false

passed: undefined

test failed: true, true, true, true, true
XMLHttpRequest cache
Similar to the newer Fetch API, any resource received may be cached by the browser. The cache is potentially vulnerable to cross-site tracking attack.
              
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 2

result, different first party: 3

unsupported: false

passed: true

test failed: false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1

result, different first party: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same first party: 1, 1, 1, 1, 1

result, different first party: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Navigation testsWhich browsers prevent websites from sharing tracking data when you click on a link?
When you click a hyperlink to navigate your browser from one site to another, certain browser APIs allow the first site to communicate to the second site. These privacy vulnerabilities can be fixed by introducing new limits on how much data is transfered between sites.',
        
document.referrer
The Referer [sic] request header is a mechanism used by browsers to let a website know where the user is visiting from. This header is inherently tracking users across websites. In recent times, browsers have switched to a policy of trimming a referrer to convey less tracking information, but Referer continues to convey cross-site tracking data by default.
              
Failed
write: (secret) => { /* do nothing */ }

read: () => document.referrer

result, same first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

result, different first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Failed
write: (secret) => { /* do nothing */ }

read: () => document.referrer

result, same first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

result, different first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Failed
write: (secret) => { /* do nothing */ }

read: () => document.referrer

result, same first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

result, different first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Failed
write: (secret) => { /* do nothing */ }

read: () => document.referrer

result, same first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

result, different first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Failed
write: (secret) => { /* do nothing */ }

read: () => document.referrer

result, same first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

result, different first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Failed
write: (secret) => { /* do nothing */ }

read: () => document.referrer

result, same first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

result, different first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Failed
write: (secret) => { /* do nothing */ }

read: () => document.referrer

result, same first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

result, different first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Failed
write: (secret) => { /* do nothing */ }

read: () => document.referrer

result, same first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

result, different first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Passed
write: (secret) => { /* do nothing */ }

read: () => document.referrer

result, same first party: https://test-pages.privacytests2.org/

result, different first party: 

unsupported: false

passed: true

test failed: false
Failed
write: (secret) => { /* do nothing */ }

read: () => document.referrer

result, same first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

result, different first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

unsupported: false, false, false

passed: false, false, false

test failed: false, false, false
Failed
write: (secret) => { /* do nothing */ }

read: () => document.referrer

result, same first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

result, different first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Failed
write: (secret) => { /* do nothing */ }

read: () => document.referrer

result, same first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

result, different first party: 
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/,
https://test-pages.privacytests2.org/

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
sessionStorage
The sessionStorage API is similar to the localStorage API, but it does not persist across tabs or across browser sessions. Nonetheless, it can be used to track users if they navigate from one website to another. This tracking can be thwarted by partitioning sessionStorage between websites.
              
Passed
write: (secret) => sessionStorage.setItem("secret", secret)

read: () => sessionStorage.getItem("secret")

result, same first party: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c,
6e0d0559-9ac2-4abc-9025-582a29e22a8f,
066762a4-4e2b-4344-82a6-b11540d563ab,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => sessionStorage.setItem("secret", secret)

read: () => sessionStorage.getItem("secret")

result, same first party: 
ead45dd4-f0d1-4ab5-af71-11d423c35903,
139bd1c9-0bf8-414e-809a-d12f37b34aab,
33d68d0c-5918-4ede-a519-aff107d2ce1c,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5,
fd597d68-f9be-48a0-9656-b7d0b2e140be

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => sessionStorage.setItem("secret", secret)

read: () => sessionStorage.getItem("secret")

result, same first party: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb,
42c2adb9-619b-4b8f-a843-84ede2df77e1,
8b52f9c3-d53f-4dad-91bb-5e162d085e46,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f,
57161958-02cf-461a-8b94-db8e32150e06

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => sessionStorage.setItem("secret", secret)

read: () => sessionStorage.getItem("secret")

result, same first party: 
7d86a137-2f75-4c34-aade-b50776a7fd40,
142f7096-d2d9-4c89-ba22-d0a3b8055308,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => sessionStorage.setItem("secret", secret)

read: () => sessionStorage.getItem("secret")

result, same first party: 
784004e0-9aeb-47e5-9c5f-12b883e353ac,
eab30b13-9ecc-48af-9a1a-cae23815efbc,
a652f390-6821-4125-9fd0-c118d2a1e7a2,
09045837-46dd-45c8-ad94-7b7c171aeef0,
7cb67be0-5127-4489-aeb1-7164e3b5af92

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => sessionStorage.setItem("secret", secret)

read: () => sessionStorage.getItem("secret")

result, same first party: 
79152685-f997-47b4-aef1-9e391daef62f,
3863988a-95b2-4b54-a63c-ce0727dc2084,
218debd1-a05b-48c0-b827-e4249b90263c,
6d855fa7-5396-4b31-ac62-f63859f5a3ef,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => sessionStorage.setItem("secret", secret)

read: () => sessionStorage.getItem("secret")

result, same first party: 
0a2e161c-e0f2-4394-8989-cbcb8cc210b9,
51502f82-4cca-4007-a857-3c34c3db9bf8,
37efbb8c-8f64-4994-8d72-bf40bf93d2bb,
3b85b9c3-ce06-4354-ab1d-998c750dab4a,
70cc2ae2-5861-4824-85db-4f905ce0c785

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => sessionStorage.setItem("secret", secret)

read: () => sessionStorage.getItem("secret")

result, same first party: 
92ee92f4-0792-4f9a-ba56-45989cd80572,
606b3a57-c3cf-45fe-a079-3b1566f6abd1,
d40bde18-63e0-419d-b233-1dc09c4c5f94,
df73d3b8-d807-46cf-9f77-65d30bc06fa4,
ba872d52-1a40-47cf-9575-e78f927ff5b3

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => sessionStorage.setItem("secret", secret)

read: () => sessionStorage.getItem("secret")

result, same first party: a2053d10-495f-40b1-89b7-6a0c8ad71604

result, different first party: null

unsupported: false

passed: true

test failed: false
Passed
write: (secret) => sessionStorage.setItem("secret", secret)

read: () => sessionStorage.getItem("secret")

result, same first party: 
37c72365-5852-42a4-9c27-47b4aa879b24,
323291dd-ef8a-484c-aaf9-641497dea358,
bd429ec2-5e93-4afc-8a05-5781fb0abad6

result, different first party: , , 

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: (secret) => sessionStorage.setItem("secret", secret)

read: () => sessionStorage.getItem("secret")

result, same first party: 
3a254b81-9486-4572-a38a-b4bd82252b3e,
c0ee6f27-d50e-4975-9a0a-523696e05cbf,
89d605d6-80b6-488e-8454-c4de9c5dfe6b,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6,
d0752b25-d241-400f-a69f-21c3ff242a56

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => sessionStorage.setItem("secret", secret)

read: () => sessionStorage.getItem("secret")

result, same first party: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078,
fad257e3-bcc3-4aa7-b541-2b4c42aee488,
3da92bf0-3ab6-4d33-9452-fac73e80ebba,
d276eb0f-b849-45c6-97b9-57596abf3678,
b64d49aa-de07-400a-9631-2935db968706

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
window.name
The window.name API allows websites to store data that will persist after the user has navigated the tab to a different website. This mechanism could be partitioned so that data is not allowed to persist between websites.
              
Passed
write: (secret) => window.name = "name_" + secret

read: () => window.name

result, same first party: 
name_31218ce0-f3be-4668-9c98-1b15e3ad8b6c,
name_6e0d0559-9ac2-4abc-9025-582a29e22a8f,
name_066762a4-4e2b-4344-82a6-b11540d563ab,
name_78743dea-6f8e-447a-b696-c3ce5c4cd9ee,
name_9a7622ad-f032-49a0-b1bc-fc695ef8e8a2

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => window.name = "name_" + secret

read: () => window.name

result, same first party: 
name_ead45dd4-f0d1-4ab5-af71-11d423c35903,
name_139bd1c9-0bf8-414e-809a-d12f37b34aab,
name_33d68d0c-5918-4ede-a519-aff107d2ce1c,
name_f4e0fc78-5add-4e91-9d10-0c5ddea287f5,
name_fd597d68-f9be-48a0-9656-b7d0b2e140be

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Failed
write: (secret) => window.name = "name_" + secret

read: () => window.name

result, same first party: 
name_acd0081b-f37c-4bfc-a842-4b74df0c5efb,
name_42c2adb9-619b-4b8f-a843-84ede2df77e1,
name_8b52f9c3-d53f-4dad-91bb-5e162d085e46,
name_d3f4e3d0-8cc1-4a45-aebb-941d62aa805f,
name_57161958-02cf-461a-8b94-db8e32150e06

result, different first party: 
name_acd0081b-f37c-4bfc-a842-4b74df0c5efb,
name_42c2adb9-619b-4b8f-a843-84ede2df77e1,
name_8b52f9c3-d53f-4dad-91bb-5e162d085e46,
name_d3f4e3d0-8cc1-4a45-aebb-941d62aa805f,
name_57161958-02cf-461a-8b94-db8e32150e06

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Failed
write: (secret) => window.name = "name_" + secret

read: () => window.name

result, same first party: 
name_7d86a137-2f75-4c34-aade-b50776a7fd40,
name_142f7096-d2d9-4c89-ba22-d0a3b8055308,
name_02a358c3-f14e-4ab4-9828-1f93ac1a8c6c,
name_ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd,
name_2b01d48f-2808-4d2b-b5e0-fa0ef53e7346

result, different first party: 
name_7d86a137-2f75-4c34-aade-b50776a7fd40,
name_142f7096-d2d9-4c89-ba22-d0a3b8055308,
name_02a358c3-f14e-4ab4-9828-1f93ac1a8c6c,
name_ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd,
name_2b01d48f-2808-4d2b-b5e0-fa0ef53e7346

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Passed
write: (secret) => window.name = "name_" + secret

read: () => window.name

result, same first party: 
name_784004e0-9aeb-47e5-9c5f-12b883e353ac,
name_eab30b13-9ecc-48af-9a1a-cae23815efbc,
name_a652f390-6821-4125-9fd0-c118d2a1e7a2,
name_09045837-46dd-45c8-ad94-7b7c171aeef0,
name_7cb67be0-5127-4489-aeb1-7164e3b5af92

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => window.name = "name_" + secret

read: () => window.name

result, same first party: 
name_79152685-f997-47b4-aef1-9e391daef62f,
name_3863988a-95b2-4b54-a63c-ce0727dc2084,
name_218debd1-a05b-48c0-b827-e4249b90263c,
name_6d855fa7-5396-4b31-ac62-f63859f5a3ef,
name_dfb73e15-8f36-4d94-b21e-ec65fc5d2f75

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => window.name = "name_" + secret

read: () => window.name

result, same first party: 
name_0a2e161c-e0f2-4394-8989-cbcb8cc210b9,
name_51502f82-4cca-4007-a857-3c34c3db9bf8,
name_37efbb8c-8f64-4994-8d72-bf40bf93d2bb,
name_3b85b9c3-ce06-4354-ab1d-998c750dab4a,
name_70cc2ae2-5861-4824-85db-4f905ce0c785

result, different first party: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Failed
write: (secret) => window.name = "name_" + secret

read: () => window.name

result, same first party: 
name_92ee92f4-0792-4f9a-ba56-45989cd80572,
name_606b3a57-c3cf-45fe-a079-3b1566f6abd1,
name_d40bde18-63e0-419d-b233-1dc09c4c5f94,
name_df73d3b8-d807-46cf-9f77-65d30bc06fa4,
name_ba872d52-1a40-47cf-9575-e78f927ff5b3

result, different first party: 
name_92ee92f4-0792-4f9a-ba56-45989cd80572,
name_606b3a57-c3cf-45fe-a079-3b1566f6abd1,
name_d40bde18-63e0-419d-b233-1dc09c4c5f94,
name_df73d3b8-d807-46cf-9f77-65d30bc06fa4,
name_ba872d52-1a40-47cf-9575-e78f927ff5b3

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Passed
write: (secret) => window.name = "name_" + secret

read: () => window.name

result, same first party: name_a2053d10-495f-40b1-89b7-6a0c8ad71604

result, different first party: 

unsupported: false

passed: true

test failed: false
Passed
write: (secret) => window.name = "name_" + secret

read: () => window.name

result, same first party: 
name_37c72365-5852-42a4-9c27-47b4aa879b24,
name_323291dd-ef8a-484c-aaf9-641497dea358,
name_bd429ec2-5e93-4afc-8a05-5781fb0abad6

result, different first party: , , 

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Failed
write: (secret) => window.name = "name_" + secret

read: () => window.name

result, same first party: 
name_3a254b81-9486-4572-a38a-b4bd82252b3e,
name_c0ee6f27-d50e-4975-9a0a-523696e05cbf,
name_89d605d6-80b6-488e-8454-c4de9c5dfe6b,
name_c190dea2-f7a7-4c44-aac0-2cfb817ddac6,
name_d0752b25-d241-400f-a69f-21c3ff242a56

result, different first party: 
name_3a254b81-9486-4572-a38a-b4bd82252b3e,
name_c0ee6f27-d50e-4975-9a0a-523696e05cbf,
name_89d605d6-80b6-488e-8454-c4de9c5dfe6b,
name_c190dea2-f7a7-4c44-aac0-2cfb817ddac6,
name_d0752b25-d241-400f-a69f-21c3ff242a56

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Failed
write: (secret) => window.name = "name_" + secret

read: () => window.name

result, same first party: 
name_c82dabdd-cc1d-4366-ab2f-6e65bea8b078,
name_fad257e3-bcc3-4aa7-b541-2b4c42aee488,
name_3da92bf0-3ab6-4d33-9452-fac73e80ebba,
name_d276eb0f-b849-45c6-97b9-57596abf3678,
name_b64d49aa-de07-400a-9631-2935db968706

result, different first party: 
name_c82dabdd-cc1d-4366-ab2f-6e65bea8b078,
name_fad257e3-bcc3-4aa7-b541-2b4c42aee488,
name_3da92bf0-3ab6-4d33-9452-fac73e80ebba,
name_d276eb0f-b849-45c6-97b9-57596abf3678,
name_b64d49aa-de07-400a-9631-2935db968706

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
HTTPS testsWhich browsers prevent unencrypted network connections?
HTTPS is the protocol that web browsers use to connect securely to websites. When HTTPS is being used, the connection is encrypted so that third parties on the network cannot read content being sent between the server and your browser. In the past, insecure connections were the default and websites would need to actively request that a browser use HTTPS. Now the status quo is shifting, and browser makers are moving toward a world where HTTPS is the default protocol.`
        
Insecure website warning
Checks to see if the browser stops loading an insecure website and warns the user before giving them the option to continue. Known as HTTPS-Only Mode in some browsers.
              
Failed
passed: false,false,false,false,false
result: Insecure website loaded,Insecure website loaded,Insecure website loaded,Insecure website loaded,Insecure website loaded
Passed
passed: true,true,true,true,true
result: Insecure website never loaded,Insecure website never loaded,Insecure website never loaded,Insecure website never loaded,Insecure website never loaded
Failed
passed: false,false,false,false,false
result: Insecure website loaded,Insecure website loaded,Insecure website loaded,Insecure website loaded,Insecure website loaded
Failed
passed: false,false,false,false,false
result: Insecure website loaded,Insecure website loaded,Insecure website loaded,Insecure website loaded,Insecure website loaded
Failed
passed: false,false,false,false,false
result: Insecure website loaded,Insecure website loaded,Insecure website loaded,Insecure website loaded,Insecure website loaded
Passed
passed: true,true,true,true,true
result: Insecure website never loaded,Insecure website never loaded,Insecure website never loaded,Insecure website never loaded,Insecure website never loaded
Passed
passed: true,true,true,true,true
result: Insecure website never loaded,Insecure website never loaded,Insecure website never loaded,Insecure website never loaded,Insecure website never loaded
Failed
passed: false,false,false,false,false
result: Insecure website loaded,Insecure website loaded,Insecure website loaded,Insecure website loaded,Insecure website loaded
Failed
passed: false
result: Insecure website loaded
Passed
passed: true,true,true
result: Insecure website never loaded,Insecure website never loaded,Insecure website never loaded
Failed
passed: false,false,false,false,false
result: Insecure website loaded,Insecure website loaded,Insecure website loaded,Insecure website loaded,Insecure website loaded
Failed
passed: false,false,false,false,false
result: Insecure website loaded,Insecure website loaded,Insecure website loaded,Insecure website loaded,Insecure website loaded
Upgradable address
Checks to see if an insecure address entered into the browser's address bar is upgraded to HTTPS whenever possible.
              
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Failed
upgraded: false
passed: false
Passed
upgraded: true,true,true
passed: true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Upgradable hyperlink
Checks to see if the user has clicked on a hyperlink to an insecure address, if the browser upgrades that address to HTTPS whenever possible.
              
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Failed
upgraded: false
passed: false
Passed
upgraded: true,true,true
passed: true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Passed
upgraded: true,true,true,true,true
passed: true,true,true,true,true
Upgradable image
Checks to see if the browser attempts to upgrade an insecure address for an image to HTTPS whenever possible.
              
Passed
passed: true,true,true,true,true
result: upgraded,upgraded,upgraded,upgraded,upgraded
Passed
passed: true,true,true,true,true
result: upgraded,upgraded,upgraded,upgraded,upgraded
Passed
passed: true,true,true,true,true
result: upgraded,upgraded,upgraded,upgraded,upgraded
Passed
passed: true,true,true,true,true
result: upgraded,upgraded,upgraded,upgraded,upgraded
Failed
passed: false,false,false,false,false
result: loaded insecurely,loaded insecurely,loaded insecurely,loaded insecurely,loaded insecurely
Passed
passed: true,true,true,true,true
result: upgraded,upgraded,upgraded,upgraded,upgraded
Passed
passed: true,true,true,true,true
result: upgraded,upgraded,upgraded,upgraded,upgraded
Passed
passed: true,true,true,true,true
result: upgraded,upgraded,upgraded,upgraded,upgraded
Failed
passed: false
result: loaded insecurely
Passed
passed: true,true,true
result: upgraded,upgraded,upgraded
Passed
passed: true,true,true,true,true
result: upgraded,upgraded,upgraded,upgraded,upgraded
Passed
passed: true,true,true,true,true
result: upgraded,upgraded,upgraded,upgraded,upgraded
Upgradable script
Checks to see if the browser attempts to upgrade an insecure address for an script to HTTPS whenever possible.
              
Passed
passed: true,true,true,true,true
result: blocked,blocked,blocked,blocked,blocked
Passed
passed: true,true,true,true,true
result: blocked,blocked,blocked,blocked,blocked
Passed
passed: true,true,true,true,true
result: blocked,blocked,blocked,blocked,blocked
Passed
passed: true,true,true,true,true
result: blocked,blocked,blocked,blocked,blocked
Passed
passed: true,true,true,true,true
result: blocked,blocked,blocked,blocked,blocked
Passed
passed: true,true,true,true,true
result: upgraded,upgraded,upgraded,upgraded,upgraded
Passed
passed: true,true,true,true,true
result: upgraded,upgraded,upgraded,upgraded,upgraded
Passed
passed: true,true,true,true,true
result: blocked,blocked,blocked,blocked,blocked
Passed
passed: true
result: blocked
Passed
passed: true,true,true
result: upgraded,upgraded,upgraded
Passed
passed: true,true,true,true,true
result: blocked,blocked,blocked,blocked,blocked
Passed
passed: true,true,true,true,true
result: blocked,blocked,blocked,blocked,blocked
Misc testsWhich browsers provide additional assorted privacy protections?
This category includes tests for the presence of miscellaneous privacy features
        
ECH enabled
Encrypted Client Hello (ECH) is a new protocol that hides the website you are visiting from third-party network eavesdroppers.
              
Passed
SNI_status: encrypted
passed: true,true,true,true,true
Failed
SNI_status: plaintext
passed: false,false,false,false,false
Passed
SNI_status: encrypted
passed: true,true,true,true,true
Passed
SNI_status: encrypted
passed: true,true,true,true,true
Failed
SNI_status: plaintext
passed: false,false,false,false,false
Failed
SNI_status: plaintext
passed: false,false,false,false,false
Failed
SNI_status: plaintext
passed: false,false,false,false,false
Failed
SNI_status: encrypted
passed: true,false,true,true,true
Failed
SNI_status: plaintext
passed: false
Failed
SNI_status: plaintext
passed: false,false,false
Passed
SNI_status: encrypted
passed: true,true,true,true,true
Passed
SNI_status: encrypted
passed: true,true,true,true,true
GPC enabled first-party
The Global Privacy Control is an HTTP header that can be sent by a browser to instruct a website not to sell the user's personal data to third parties. This test checks to see if the GPC header is sent by default to the top-level website.
              
Passed
header value: 1
passed: true,true,true,true,true
Passed
header value: 1
passed: true,true,true,true,true
Failed
header value: undefined
passed: false,false,false,false,false
Failed
header value: undefined
passed: false,false,false,false,false
Passed
header value: 1
passed: true,true,true,true,true
Passed
header value: 1
passed: true,true,true,true,true
Failed
header value: undefined
passed: false,false,false,false,false
Failed
header value: undefined
passed: false,false,false,false,false
Failed
header value: undefined
passed: false
Failed
header value: undefined
passed: false,false,false
Failed
header value: undefined
passed: false,false,false,false,false
Failed
header value: undefined
passed: false,false,false,false,false
GPC enabled third-party
The Global Privacy Control is an HTTP header that can be sent by a browser to instruct a visited website not to sell the user's personal data to other parties. This test checks to see if the GPC header is sent to third-party elements on the web page.
              
Passed
sec-gpc: 1
passed: true,true,true,true,true
Passed
sec-gpc: 1
passed: true,true,true,true,true
Failed
passed: false,false,false,false,false
Failed
passed: false,false,false,false,false
Passed
sec-gpc: 1
passed: true,true,true,true,true
Passed
sec-gpc: 1
passed: true,true,true,true,true
Failed
passed: false,false,false,false,false
Failed
passed: false,false,false,false,false
Failed
passed: false
Failed
passed: false,false,false
Failed
passed: false,false,false,false,false
Failed
passed: false,false,false,false,false
IP address leak
IP addresses can be used to uniquely identify a large percentage of users. A proxy, VPN, or Tor can mask a user's IP address.
              
Failed
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Failed
passed: false,false,false,false,false
Failed
passed: false,false,false,false,false
Failed
passed: false,false,false,false,false
Failed
passed: false,false,false,false,false
Failed
passed: false,false,false,false,false
Failed
passed: false,false,false,false,false
Failed
passed: false
Passed
passed: true,true,true
Failed
passed: false,false,false,false,false
Failed
passed: false,false,false,false,false
Stream isolation
Browsers that use Tor can use a different Tor circuit per top-level website.
              
Unsupported
write: () => {
      if (!usingTor) {
        throw new Error("Unsupported");
      }
    }
read: async () => {
      if (usingTor) {
        return ipAddress;
      } else {
        throw new Error("Unsupported");
      }
    }
unsupported: true,true,true,true,true
readSameFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
readDifferentFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
testFailed: false,false,false,false,false
Failed
write: () => {
      if (!usingTor) {
        throw new Error("Unsupported");
      }
    }
read: async () => {
      if (usingTor) {
        return ipAddress;
      } else {
        throw new Error("Unsupported");
      }
    }
unsupported: false,false,false,false,false
readSameFirstParty: 185.220.101.10,185.220.101.26,165.73.242.163,185.220.101.7,2a04:52c0:106:394::1
readDifferentFirstParty: 185.220.101.10,185.220.101.26,165.73.242.163,185.220.101.7,2a04:52c0:106:394::1
passed: false,false,false,false,false
testFailed: false,false,false,false,false
Unsupported
write: () => {
      if (!usingTor) {
        throw new Error("Unsupported");
      }
    }
read: async () => {
      if (usingTor) {
        return ipAddress;
      } else {
        throw new Error("Unsupported");
      }
    }
unsupported: true,true,true,true,true
readSameFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
readDifferentFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
testFailed: false,false,false,false,false
Unsupported
write: () => {
      if (!usingTor) {
        throw new Error("Unsupported");
      }
    }
read: async () => {
      if (usingTor) {
        return ipAddress;
      } else {
        throw new Error("Unsupported");
      }
    }
unsupported: true,true,true,true,true
readSameFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
readDifferentFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
testFailed: false,false,false,false,false
Unsupported
write: () => {
      if (!usingTor) {
        throw new Error("Unsupported");
      }
    }
read: async () => {
      if (usingTor) {
        return ipAddress;
      } else {
        throw new Error("Unsupported");
      }
    }
unsupported: true,true,true,true,true
readSameFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
readDifferentFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
testFailed: false,false,false,false,false
Unsupported
write: () => {
      if (!usingTor) {
        throw new Error("Unsupported");
      }
    }
read: async () => {
      if (usingTor) {
        return ipAddress;
      } else {
        throw new Error("Unsupported");
      }
    }
unsupported: true,true,true,true,true
readSameFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
readDifferentFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
testFailed: false,false,false,false,false
Unsupported
write: () => {
      if (!usingTor) {
        throw new Error("Unsupported");
      }
    }
read: async () => {
      if (usingTor) {
        return ipAddress;
      } else {
        throw new Error("Unsupported");
      }
    }
unsupported: true,true,true,true,true
readSameFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
readDifferentFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
testFailed: false,false,false,false,false
Unsupported
write: () => {
      if (!usingTor) {
        throw new Error("Unsupported");
      }
    }
read: async () => {
      if (usingTor) {
        return ipAddress;
      } else {
        throw new Error("Unsupported");
      }
    }
unsupported: true,true,true,true,true
readSameFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
readDifferentFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
testFailed: false,false,false,false,false
Unsupported
write: () => {
      if (!usingTor) {
        throw new Error("Unsupported");
      }
    }
read: async () => {
      if (usingTor) {
        return ipAddress;
      } else {
        throw new Error("Unsupported");
      }
    }
unsupported: true
readSameFirstParty: Error: Unsupported
readDifferentFirstParty: Error: Unsupported
testFailed: false
Passed
write: () => {
      if (!usingTor) {
        throw new Error("Unsupported");
      }
    }
read: async () => {
      if (usingTor) {
        return ipAddress;
      } else {
        throw new Error("Unsupported");
      }
    }
unsupported: false,false,false
readSameFirstParty: 2a0b:f4c2:3::64,2605:6400:30:f858:2704:73e1:7085:12ef,2a0e:e701:1198::1
readDifferentFirstParty: 2a03:4000:37:3:e842:2ff:feb9:c49c,2001:67c:6ec:203:192:42:116:217,2a03:4000:66:d88:6469:50ff:fe0b:98a8
passed: true,true,true
testFailed: false,false,false
Unsupported
write: () => {
      if (!usingTor) {
        throw new Error("Unsupported");
      }
    }
read: async () => {
      if (usingTor) {
        return ipAddress;
      } else {
        throw new Error("Unsupported");
      }
    }
unsupported: true,true,true,true,true
readSameFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
readDifferentFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
testFailed: false,false,false,false,false
Unsupported
write: () => {
      if (!usingTor) {
        throw new Error("Unsupported");
      }
    }
read: async () => {
      if (usingTor) {
        return ipAddress;
      } else {
        throw new Error("Unsupported");
      }
    }
unsupported: true,true,true,true,true
readSameFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
readDifferentFirstParty: Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported,Error: Unsupported
testFailed: false,false,false,false,false
Tor enabled
The Tor network sends the browser's web requests through a series of relays to hide a user's IP address, thereby helping to mask their identity and location. This test checks to see if the Tor network is being used by default.
              
Failed
IsTorExit: false,false,false,false,false
passed: false,false,false,false,false
Passed
IsTorExit: true,true,true,true,true
passed: true,true,true,true,true
Failed
IsTorExit: false,false,false,false,false
passed: false,false,false,false,false
Failed
IsTorExit: false,false,false,false,false
passed: false,false,false,false,false
Failed
IsTorExit: false,false,false,false,false
passed: false,false,false,false,false
Failed
IsTorExit: false,false,false,false,false
passed: false,false,false,false,false
Failed
IsTorExit: false,false,false,false,false
passed: false,false,false,false,false
Failed
IsTorExit: false,false,false,false,false
passed: false,false,false,false,false
Failed
IsTorExit: false
passed: false
Passed
IsTorExit: true,true,true
passed: true,true,true
Failed
IsTorExit: false,false,false,false,false
passed: false,false,false,false,false
Failed
IsTorExit: false,false,false,false,false
passed: false,false,false,false,false
Fingerprinting resistance testsWhich browsers hide what's unique about your device?
Fingerprinting is a technique trackers use to uniquely identify you as you browse the web. A fingerprinting script will measure several characteristics of your browser and, combining this data, will build a fingerprint that may uniquely identify you among web users. Browsers can introduce countermeasures, such as minimizing the distinguishing information disclosed by certain web APIs so your browser is harder to pick out from the crowd (so-called 'fingerprinting resistance').`,
        
Media query screen height
Height of the user's screen in pixels.
              
Passed
expression: undefined
desired expression: undefined
actual value: 1302,1300,1298,1301,1301
desired value: undefined
passed: true,true,true,true,true
Passed
expression: undefined
desired expression: undefined
actual value: 1302,1301,1304,1297,1300
desired value: undefined
passed: true,true,true,true,true
Failed
expression: undefined
desired expression: undefined
actual value: 1440,1440,1440,1440,1440
desired value: undefined
passed: false,false,false,false,false
Failed
expression: undefined
desired expression: undefined
actual value: 1440,1440,1440,1440,1440
desired value: undefined
passed: false,false,false,false,false
Failed
expression: undefined
desired expression: undefined
actual value: 1440,1440,1440,1440,1440
desired value: undefined
passed: false,false,false,false,false
Passed
expression: undefined
desired expression: undefined
actual value: 900,900,900,900,900
desired value: undefined
passed: true,true,true,true,true
Passed
expression: undefined
desired expression: undefined
actual value: 900,900,900,900,900
desired value: undefined
passed: true,true,true,true,true
Failed
expression: undefined
desired expression: undefined
actual value: 1440,1440,1440,1440,1440
desired value: undefined
passed: false,false,false,false,false
Passed
expression: undefined
desired expression: undefined
actual value: 1255
desired value: undefined
passed: true
Passed
expression: undefined
desired expression: undefined
actual value: 900,900,900
desired value: undefined
passed: true,true,true
Failed
expression: undefined
desired expression: undefined
actual value: 1440,1440,1440,1440,1440
desired value: undefined
passed: false,false,false,false,false
Failed
expression: undefined
desired expression: undefined
actual value: 1440,1440,1440,1440,1440
desired value: undefined
passed: false,false,false,false,false
Media query screen width
Width of the user's screen in pixels.
              
Passed
expression: undefined
desired expression: undefined
actual value: 1207,1207,1204,1201,1206
desired value: undefined
passed: true,true,true,true,true
Passed
expression: undefined
desired expression: undefined
actual value: 1206,1203,1203,1206,1208
desired value: undefined
passed: true,true,true,true,true
Failed
expression: undefined
desired expression: undefined
actual value: 2560,2560,2560,2560,2560
desired value: undefined
passed: false,false,false,false,false
Failed
expression: undefined
desired expression: undefined
actual value: 2560,2560,2560,2560,2560
desired value: undefined
passed: false,false,false,false,false
Failed
expression: undefined
desired expression: undefined
actual value: 2560,2560,2560,2560,2560
desired value: undefined
passed: false,false,false,false,false
Passed
expression: undefined
desired expression: undefined
actual value: 1600,1600,1600,1600,1600
desired value: undefined
passed: true,true,true,true,true
Passed
expression: undefined
desired expression: undefined
actual value: 1400,1400,1400,1400,1400
desired value: undefined
passed: true,true,true,true,true
Failed
expression: undefined
desired expression: undefined
actual value: 2560,2560,2560,2560,2560
desired value: undefined
passed: false,false,false,false,false
Passed
expression: undefined
desired expression: undefined
actual value: 1180
desired value: undefined
passed: true
Passed
expression: undefined
desired expression: undefined
actual value: 1400,1400,1400
desired value: undefined
passed: true,true,true
Failed
expression: undefined
desired expression: undefined
actual value: 2560,2560,2560,2560,2560
desired value: undefined
passed: false,false,false,false,false
Failed
expression: undefined
desired expression: undefined
actual value: 2560,2560,2560,2560,2560
desired value: undefined
passed: false,false,false,false,false
outerHeight
Height of the browser window in pixels, including browser chrome.
              
Passed
expression: outerHeight
desired expression: undefined
actual value: 1302,1300,1298,1301,1301
desired value: undefined
passed: true,true,true,true,true
Passed
expression: outerHeight
desired expression: undefined
actual value: 1302,1301,1304,1297,1300
desired value: undefined
passed: true,true,true,true,true
Failed
expression: outerHeight
desired expression: undefined
actual value: 1371,1371,1371,1371,1371
desired value: undefined
passed: false,false,false,false,false
Failed
expression: outerHeight
desired expression: undefined
actual value: 1371,1371,1371,1371,1371
desired value: undefined
passed: false,false,false,false,false
Failed
expression: outerHeight
desired expression: undefined
actual value: 1040,1040,1040,1040,1040
desired value: undefined
passed: false,false,false,false,false
Passed
expression: outerHeight
desired expression: undefined
actual value: 900,900,900,900,900
desired value: undefined
passed: true,true,true,true,true
Passed
expression: outerHeight
desired expression: undefined
actual value: 900,900,900,900,900
desired value: undefined
passed: true,true,true,true,true
Failed
expression: outerHeight
desired expression: undefined
actual value: 1061,1061,1061,1061,1061
desired value: undefined
passed: false,false,false,false,false
Passed
expression: outerHeight
desired expression: undefined
actual value: 1255
desired value: undefined
passed: true
Passed
expression: outerHeight
desired expression: undefined
actual value: 900,900,900
desired value: undefined
passed: true,true,true
Failed
expression: outerHeight
desired expression: undefined
actual value: 1371,1371,1371,1371,1371
desired value: undefined
passed: false,false,false,false,false
Failed
expression: outerHeight
desired expression: undefined
actual value: 1371,1371,1371,1371,1371
desired value: undefined
passed: false,false,false,false,false
screen.height
Height of the user's screen, in pixels.
              
Passed
expression: screen.height
desired expression: undefined
actual value: 1302,1300,1298,1301,1301
desired value: undefined
passed: true,true,true,true,true
Passed
expression: screen.height
desired expression: undefined
actual value: 1302,1301,1304,1297,1300
desired value: undefined
passed: true,true,true,true,true
Failed
expression: screen.height
desired expression: undefined
actual value: 1440,1440,1440,1440,1440
desired value: undefined
passed: false,false,false,false,false
Failed
expression: screen.height
desired expression: undefined
actual value: 1440,1440,1440,1440,1440
desired value: undefined
passed: false,false,false,false,false
Failed
expression: screen.height
desired expression: undefined
actual value: 1440,1440,1440,1440,1440
desired value: undefined
passed: false,false,false,false,false
Passed
expression: screen.height
desired expression: undefined
actual value: 900,900,900,900,900
desired value: undefined
passed: true,true,true,true,true
Passed
expression: screen.height
desired expression: undefined
actual value: 900,900,900,900,900
desired value: undefined
passed: true,true,true,true,true
Failed
expression: screen.height
desired expression: undefined
actual value: 1440,1440,1440,1440,1440
desired value: undefined
passed: false,false,false,false,false
Passed
expression: screen.height
desired expression: undefined
actual value: 1255
desired value: undefined
passed: true
Passed
expression: screen.height
desired expression: undefined
actual value: 900,900,900
desired value: undefined
passed: true,true,true
Failed
expression: screen.height
desired expression: undefined
actual value: 1440,1440,1440,1440,1440
desired value: undefined
passed: false,false,false,false,false
Failed
expression: screen.height
desired expression: undefined
actual value: 1440,1440,1440,1440,1440
desired value: undefined
passed: false,false,false,false,false
screen.width
Width of the user's screen, in pixels.
              
Passed
expression: screen.width
desired expression: undefined
actual value: 1207,1207,1204,1201,1206
desired value: undefined
passed: true,true,true,true,true
Passed
expression: screen.width
desired expression: undefined
actual value: 1206,1203,1203,1206,1208
desired value: undefined
passed: true,true,true,true,true
Failed
expression: screen.width
desired expression: undefined
actual value: 2560,2560,2560,2560,2560
desired value: undefined
passed: false,false,false,false,false
Failed
expression: screen.width
desired expression: undefined
actual value: 2560,2560,2560,2560,2560
desired value: undefined
passed: false,false,false,false,false
Failed
expression: screen.width
desired expression: undefined
actual value: 2560,2560,2560,2560,2560
desired value: undefined
passed: false,false,false,false,false
Passed
expression: screen.width
desired expression: undefined
actual value: 1600,1600,1600,1600,1600
desired value: undefined
passed: true,true,true,true,true
Passed
expression: screen.width
desired expression: undefined
actual value: 1400,1400,1400,1400,1400
desired value: undefined
passed: true,true,true,true,true
Failed
expression: screen.width
desired expression: undefined
actual value: 2560,2560,2560,2560,2560
desired value: undefined
passed: false,false,false,false,false
Passed
expression: screen.width
desired expression: undefined
actual value: 1180
desired value: undefined
passed: true
Passed
expression: screen.width
desired expression: undefined
actual value: 1400,1400,1400
desired value: undefined
passed: true,true,true
Failed
expression: screen.width
desired expression: undefined
actual value: 2560,2560,2560,2560,2560
desired value: undefined
passed: false,false,false,false,false
Failed
expression: screen.width
desired expression: undefined
actual value: 2560,2560,2560,2560,2560
desired value: undefined
passed: false,false,false,false,false
screenX
Position, in pixels, of the left edge of the browser window on screen.
              
Passed
expression: screenX
desired expression: undefined
actual value: 6,6,1,2,3
desired value: undefined
passed: true,true,true,true,true
Passed
expression: screenX
desired expression: undefined
actual value: 7,7,3,5,1
desired value: undefined
passed: true,true,true,true,true
Failed
expression: screenX
desired expression: undefined
actual value: 22,22,22,22,22
desired value: undefined
passed: false,false,false,false,false
Failed
expression: screenX
desired expression: undefined
actual value: 22,22,22,22,22
desired value: undefined
passed: false,false,false,false,false
Passed
expression: screenX
desired expression: undefined
actual value: 4,4,4,4,4
desired value: undefined
passed: true,true,true,true,true
Passed
expression: screenX
desired expression: undefined
actual value: 0,0,0,0,0
desired value: undefined
passed: true,true,true,true,true
Passed
expression: screenX
desired expression: undefined
actual value: 0,0,0,0,0
desired value: undefined
passed: true,true,true,true,true
Failed
expression: screenX
desired expression: undefined
actual value: 320,320,320,320,320
desired value: undefined
passed: false,false,false,false,false
Passed
expression: screenX
desired expression: undefined
actual value: 0
desired value: undefined
passed: true
Passed
expression: screenX
desired expression: undefined
actual value: 0,0,0
desired value: undefined
passed: true,true,true
Failed
expression: screenX
desired expression: undefined
actual value: 22,22,22,22,22
desired value: undefined
passed: false,false,false,false,false
Failed
expression: screenX
desired expression: undefined
actual value: 22,22,22,22,22
desired value: undefined
passed: false,false,false,false,false
screenY
Position, in pixels, of the top edge of the browser window on screen.
              
Passed
expression: screenY
desired expression: undefined
actual value: 2,3,6,5,6
desired value: undefined
passed: true,true,true,true,true
Passed
expression: screenY
desired expression: undefined
actual value: 3,7,5,4,6
desired value: undefined
passed: true,true,true,true,true
Failed
expression: screenY
desired expression: undefined
actual value: 47,47,47,47,47
desired value: undefined
passed: false,false,false,false,false
Failed
expression: screenY
desired expression: undefined
actual value: 47,47,47,47,47
desired value: undefined
passed: false,false,false,false,false
Failed
expression: screenY
desired expression: undefined
actual value: 25,25,25,25,25
desired value: undefined
passed: false,false,false,false,false
Passed
expression: screenY
desired expression: undefined
actual value: 0,0,0,0,0
desired value: undefined
passed: true,true,true,true,true
Passed
expression: screenY
desired expression: undefined
actual value: 0,0,0,0,0
desired value: undefined
passed: true,true,true,true,true
Failed
expression: screenY
desired expression: undefined
actual value: 202,202,202,202,202
desired value: undefined
passed: false,false,false,false,false
Passed
expression: screenY
desired expression: undefined
actual value: 0
desired value: undefined
passed: true
Passed
expression: screenY
desired expression: undefined
actual value: 0,0,0
desired value: undefined
passed: true,true,true
Failed
expression: screenY
desired expression: undefined
actual value: 47,47,47,47,47
desired value: undefined
passed: false,false,false,false,false
Failed
expression: screenY
desired expression: undefined
actual value: 47,47,47,47,47
desired value: undefined
passed: false,false,false,false,false
System font detection
Web pages can detect the presence of a font installed on the user's system. The presence or absence of various fonts is commonly used to fingerprint users.
              
Failed
expression: undefined
desired expression: undefined
actual value: undefined
desired value: undefined
passed: true,true,true,true,false
Failed
expression: undefined
desired expression: undefined
actual value: undefined
desired value: undefined
passed: true,true,true,false,true
Failed
expression: undefined
desired expression: undefined
actual value: undefined
desired value: undefined
passed: false,false,false,false,false
Failed
expression: undefined
desired expression: undefined
actual value: undefined
desired value: undefined
passed: false,false,false,false,false
Passed
expression: undefined
desired expression: undefined
actual value: undefined
desired value: undefined
passed: true,true,true,true,true
Passed
expression: undefined
desired expression: undefined
actual value: undefined
desired value: undefined
passed: true,true,true,true,true
Passed
expression: undefined
desired expression: undefined
actual value: undefined
desired value: undefined
passed: true,true,true,true,true
Failed
expression: undefined
desired expression: undefined
actual value: undefined
desired value: undefined
passed: false,false,false,false,false
Passed
expression: undefined
desired expression: undefined
actual value: undefined
desired value: undefined
passed: true
Passed
expression: undefined
desired expression: undefined
actual value: undefined
desired value: undefined
passed: true,true,true
Failed
expression: undefined
desired expression: undefined
actual value: undefined
desired value: undefined
passed: false,false,false,false,false
Failed
expression: undefined
desired expression: undefined
actual value: undefined
desired value: undefined
passed: false,false,false,false,false
Tracking query parameter testsWhich browsers remove URL parameters that can track you?
When you browse from one web page to another, tracking companies will frequently attach a 'tracking query parameter' to the address of the second web page. That query parameter may contain a unique identifier that tracks you individually as you browse the web. And these query parameters are frequently synchronized with cookies, making them a powerful tracking vector. Web browsers can protect you from known tracking query parameters by stripping them from web addresses before your browser sends them. (The set of tracking query parameters tested here was largely borrowed from Brave.)`
        
__hsfp
HubSpot tracking parameter
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
__hssc
HubSpot tracking parameter
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
__hstc
HubSpot tracking parameter
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
__s
Drip.com email address tracking parameter
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
_hsenc
HubSpot tracking parameter
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
_openstat
Yandex tracking parameter
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
dclid
DoubleClick Click ID (Google)
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
fbclid
Facebook Click Identifier
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
gclid
Google Click Identifier
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
hsCtaTracking
HubSpot tracking parameter
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
mc_eid
Mailchimp Email ID (email recipient's address)
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
mkt_tok
Adobe Marketo tracking parameter
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
ml_subscriber
MailerLite email tracking
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Failed
value: 6220409046356266
passed: false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
ml_subscriber_hash
MailerLite email tracking
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Failed
value: 6220409046356266
passed: false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
msclkid
Microsoft Click ID
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
oly_anon_id
Omeda marketing 'anonymous' customer id
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
oly_enc_id
Omeda marketing 'known' customer id
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
rb_clickid
Unknown high-entropy tracking parameter
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Failed
value: 6220409046356266
passed: false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
s_cid
Adobe Site Catalyst tracking parameter
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Failed
value: 6220409046356266
passed: false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
vero_conv
Vero tracking parameter
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Failed
value: 6220409046356266
passed: false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
vero_id
Vero tracking parameter
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
wickedid
Wicked Reports e-commerce tracking
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
yclid
Yandex Click ID
              
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true,true,true,true,true
Passed
passed: true,true,true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Passed
passed: true
Passed
passed: true,true,true
Failed
value: 6220409046356266
passed: false,false,false,false,false
Failed
value: 6220409046356266
passed: false,false,false,false,false
Tracker content blocking testsWhich browsers block important known tracking scripts and pixels?
When you visit a web page, it frequently has third-party embedded tracking content, such as scripts and tracking pixels. These embedded components spy on you. Some browsers and browser extensions maintain list of tracking companies and block their content from being loaded. This section checks to see if a browser blocks 20 of the largest trackers listed by https://whotracks.me.`
        
Adobe
Tests whether the browser blocks the page from loading the tracker at https://munchkin.marketo.net/munchkin.js
              
Passed
url: https://munchkin.marketo.net/munchkin.js
passed: true,true,true,true,true
Passed
url: https://munchkin.marketo.net/munchkin.js
passed: true,true,true,true,true
Failed
url: https://munchkin.marketo.net/munchkin.js
passed: false,false,false,false,false
Failed
url: https://munchkin.marketo.net/munchkin.js
passed: false,false,false,false,false
Passed
url: https://munchkin.marketo.net/munchkin.js
passed: true,true,true,true,true
Passed
url: https://munchkin.marketo.net/munchkin.js
passed: true,true,true,true,true
Passed
url: https://munchkin.marketo.net/munchkin.js
passed: true,true,true,true,true
Failed
url: https://munchkin.marketo.net/munchkin.js
passed: false,false,false,false,false
Failed
url: https://munchkin.marketo.net/munchkin.js
passed: false
Failed
url: https://munchkin.marketo.net/munchkin.js
passed: false,false,false
Failed
url: https://munchkin.marketo.net/munchkin.js
passed: false,false,false,false,false
Failed
url: https://munchkin.marketo.net/munchkin.js
passed: false,false,false,false,false
Adobe Audience Manager
Tests whether the browser blocks the page from loading the tracker at https://dpm.demdex.net/ibs
              
Passed
url: https://dpm.demdex.net/ibs
passed: true,true,true,true,true
Passed
url: https://dpm.demdex.net/ibs
passed: true,true,true,true,true
Failed
url: https://dpm.demdex.net/ibs
passed: false,false,false,false,false
Failed
url: https://dpm.demdex.net/ibs
passed: false,false,false,false,false
Passed
url: https://dpm.demdex.net/ibs
passed: true,true,true,true,true
Passed
url: https://dpm.demdex.net/ibs
passed: true,true,true,true,true
Passed
url: https://dpm.demdex.net/ibs
passed: true,true,true,true,true
Failed
url: https://dpm.demdex.net/ibs
passed: false,false,false,false,false
Failed
url: https://dpm.demdex.net/ibs
passed: false
Failed
url: https://dpm.demdex.net/ibs
passed: false,false,false
Failed
url: https://dpm.demdex.net/ibs
passed: false,false,false,false,false
Failed
url: https://dpm.demdex.net/ibs
passed: false,false,false,false,false
Amazon adsystem
Tests whether the browser blocks the page from loading the tracker at https://s.amazon-adsystem.com/dcm
              
Passed
url: https://s.amazon-adsystem.com/dcm
passed: true,true,true,true,true
Passed
url: https://s.amazon-adsystem.com/dcm
passed: true,true,true,true,true
Failed
url: https://s.amazon-adsystem.com/dcm
passed: false,false,false,false,false
Failed
url: https://s.amazon-adsystem.com/dcm
passed: false,false,false,false,false
Passed
url: https://s.amazon-adsystem.com/dcm
passed: true,true,true,true,true
Passed
url: https://s.amazon-adsystem.com/dcm
passed: true,true,true,true,true
Passed
url: https://s.amazon-adsystem.com/dcm
passed: true,true,true,true,true
Failed
url: https://s.amazon-adsystem.com/dcm
passed: false,false,false,false,false
Failed
url: https://s.amazon-adsystem.com/dcm
passed: false
Failed
url: https://s.amazon-adsystem.com/dcm
passed: false,false,false
Failed
url: https://s.amazon-adsystem.com/dcm
passed: false,false,false,false,false
Failed
url: https://s.amazon-adsystem.com/dcm
passed: false,false,false,false,false
AppNexus
Tests whether the browser blocks the page from loading the tracker at https://ib.adnxs.com/px?id=178248&t=1
              
Passed
url: https://ib.adnxs.com/px?id=178248&t=1
passed: true,true,true,true,true
Passed
url: https://ib.adnxs.com/px?id=178248&t=1
passed: true,true,true,true,true
Failed
url: https://ib.adnxs.com/px?id=178248&t=1
passed: false,false,false,false,false
Failed
url: https://ib.adnxs.com/px?id=178248&t=1
passed: false,false,false,false,false
Passed
url: https://ib.adnxs.com/px?id=178248&t=1
passed: true,true,true,true,true
Passed
url: https://ib.adnxs.com/px?id=178248&t=1
passed: true,true,true,true,true
Passed
url: https://ib.adnxs.com/px?id=178248&t=1
passed: true,true,true,true,true
Failed
url: https://ib.adnxs.com/px?id=178248&t=1
passed: false,false,false,false,false
Failed
url: https://ib.adnxs.com/px?id=178248&t=1
passed: false
Failed
url: https://ib.adnxs.com/px?id=178248&t=1
passed: false,false,false
Failed
url: https://ib.adnxs.com/px?id=178248&t=1
passed: false,false,false,false,false
Failed
url: https://ib.adnxs.com/px?id=178248&t=1
passed: false,false,false,false,false
Bing Ads
Tests whether the browser blocks the page from loading the tracker at https://bat.bing.com/bat.js
              
Passed
url: https://bat.bing.com/bat.js
passed: true,true,true,true,true
Passed
url: https://bat.bing.com/bat.js
passed: true,true,true,true,true
Failed
url: https://bat.bing.com/bat.js
passed: false,false,false,false,false
Failed
url: https://bat.bing.com/bat.js
passed: false,false,false,false,false
Passed
url: https://bat.bing.com/bat.js
passed: true,true,true,true,true
Passed
url: https://bat.bing.com/bat.js
passed: true,true,true,true,true
Passed
url: https://bat.bing.com/bat.js
passed: true,true,true,true,true
Failed
url: https://bat.bing.com/bat.js
passed: false,false,false,false,false
Failed
url: https://bat.bing.com/bat.js
passed: false
Failed
url: https://bat.bing.com/bat.js
passed: false,false,false
Failed
url: https://bat.bing.com/bat.js
passed: false,false,false,false,false
Failed
url: https://bat.bing.com/bat.js
passed: false,false,false,false,false
Chartbeat
Tests whether the browser blocks the page from loading the tracker at https://static.chartbeat.com/js/chartbeat.js
              
Passed
url: https://static.chartbeat.com/js/chartbeat.js
passed: true,true,true,true,true
Passed
url: https://static.chartbeat.com/js/chartbeat.js
passed: true,true,true,true,true
Failed
url: https://static.chartbeat.com/js/chartbeat.js
passed: false,false,false,false,false
Failed
url: https://static.chartbeat.com/js/chartbeat.js
passed: false,false,false,false,false
Passed
url: https://static.chartbeat.com/js/chartbeat.js
passed: true,true,true,true,true
Passed
url: https://static.chartbeat.com/js/chartbeat.js
passed: true,true,true,true,true
Passed
url: https://static.chartbeat.com/js/chartbeat.js
passed: true,true,true,true,true
Failed
url: https://static.chartbeat.com/js/chartbeat.js
passed: false,false,false,false,false
Passed
url: https://static.chartbeat.com/js/chartbeat.js
passed: true
Failed
url: https://static.chartbeat.com/js/chartbeat.js
passed: false,false,false
Failed
url: https://static.chartbeat.com/js/chartbeat.js
passed: false,false,false,false,false
Failed
url: https://static.chartbeat.com/js/chartbeat.js
passed: false,false,false,false,false
Criteo
Tests whether the browser blocks the page from loading the tracker at https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
              
Passed
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
passed: true,true,true,true,true
Passed
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
passed: true,true,true,true,true
Failed
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
passed: false,false,false,false,false
Failed
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
passed: false,false,false,false,false
Passed
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
passed: true,true,true,true,true
Passed
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
passed: true,true,true,true,true
Passed
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
passed: true,true,true,true,true
Failed
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
passed: false,false,false,false,false
Failed
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
passed: false
Failed
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
passed: false,false,false
Failed
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
passed: false,false,false,false,false
Failed
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
passed: false,false,false,false,false
DoubleClick (Google)
Tests whether the browser blocks the page from loading the tracker at https://securepubads.g.doubleclick.net/static/glade.js
              
Passed
url: https://securepubads.g.doubleclick.net/static/glade.js
passed: true,true,true,true,true
Passed
url: https://securepubads.g.doubleclick.net/static/glade.js
passed: true,true,true,true,true
Failed
url: https://securepubads.g.doubleclick.net/static/glade.js
passed: false,false,false,false,false
Failed
url: https://securepubads.g.doubleclick.net/static/glade.js
passed: false,false,false,false,false
Passed
url: https://securepubads.g.doubleclick.net/static/glade.js
passed: true,true,true,true,true
Passed
url: https://securepubads.g.doubleclick.net/static/glade.js
passed: true,true,true,true,true
Passed
url: https://securepubads.g.doubleclick.net/static/glade.js
passed: true,true,true,true,true
Failed
url: https://securepubads.g.doubleclick.net/static/glade.js
passed: false,false,false,false,false
Failed
url: https://securepubads.g.doubleclick.net/static/glade.js
passed: false
Failed
url: https://securepubads.g.doubleclick.net/static/glade.js
passed: false,false,false
Failed
url: https://securepubads.g.doubleclick.net/static/glade.js
passed: false,false,false,false,false
Failed
url: https://securepubads.g.doubleclick.net/static/glade.js
passed: false,false,false,false,false
Facebook tracking
Tests whether the browser blocks the page from loading the tracker at https://connect.facebook.net/en_US/fbevents.js
              
Passed
url: https://connect.facebook.net/en_US/fbevents.js
passed: true,true,true,true,true
Passed
url: https://connect.facebook.net/en_US/fbevents.js
passed: true,true,true,true,true
Failed
url: https://connect.facebook.net/en_US/fbevents.js
passed: false,false,false,false,false
Failed
url: https://connect.facebook.net/en_US/fbevents.js
passed: false,false,false,false,false
Passed
url: https://connect.facebook.net/en_US/fbevents.js
passed: true,true,true,true,true
Passed
url: https://connect.facebook.net/en_US/fbevents.js
passed: true,true,true,true,true
Passed
url: https://connect.facebook.net/en_US/fbevents.js
passed: true,true,true,true,true
Failed
url: https://connect.facebook.net/en_US/fbevents.js
passed: false,false,false,false,false
Passed
url: https://connect.facebook.net/en_US/fbevents.js
passed: true
Failed
url: https://connect.facebook.net/en_US/fbevents.js
passed: false,false,false
Failed
url: https://connect.facebook.net/en_US/fbevents.js
passed: false,false,false,false,false
Failed
url: https://connect.facebook.net/en_US/fbevents.js
passed: false,false,false,false,false
Google (third-party ad pixel)
Tests whether the browser blocks the page from loading the tracker at https://www.google.com/pagead/1p-user-list/
              
Passed
url: https://www.google.com/pagead/1p-user-list/
passed: true,true,true,true,true
Passed
url: https://www.google.com/pagead/1p-user-list/
passed: true,true,true,true,true
Failed
url: https://www.google.com/pagead/1p-user-list/
passed: false,false,false,false,false
Failed
url: https://www.google.com/pagead/1p-user-list/
passed: false,false,false,false,false
Failed
url: https://www.google.com/pagead/1p-user-list/
passed: false,false,false,false,false
Passed
url: https://www.google.com/pagead/1p-user-list/
passed: true,true,true,true,true
Passed
url: https://www.google.com/pagead/1p-user-list/
passed: true,true,true,true,true
Failed
url: https://www.google.com/pagead/1p-user-list/
passed: false,false,false,false,false
Failed
url: https://www.google.com/pagead/1p-user-list/
passed: false
Failed
url: https://www.google.com/pagead/1p-user-list/
passed: false,false,false
Failed
url: https://www.google.com/pagead/1p-user-list/
passed: false,false,false,false,false
Failed
url: https://www.google.com/pagead/1p-user-list/
passed: false,false,false,false,false
Google Analytics
Tests whether the browser blocks the page from loading the tracker at https://google-analytics.com/urchin.js
              
Passed
url: https://google-analytics.com/urchin.js
passed: true,true,true,true,true
Passed
url: https://google-analytics.com/urchin.js
passed: true,true,true,true,true
Failed
url: https://google-analytics.com/urchin.js
passed: false,false,false,false,false
Failed
url: https://google-analytics.com/urchin.js
passed: false,false,false,false,false
Passed
url: https://google-analytics.com/urchin.js
passed: true,true,true,true,true
Passed
url: https://google-analytics.com/urchin.js
passed: true,true,true,true,true
Passed
url: https://google-analytics.com/urchin.js
passed: true,true,true,true,true
Failed
url: https://google-analytics.com/urchin.js
passed: false,false,false,false,false
Passed
url: https://google-analytics.com/urchin.js
passed: true
Failed
url: https://google-analytics.com/urchin.js
passed: false,false,false
Failed
url: https://google-analytics.com/urchin.js
passed: false,false,false,false,false
Failed
url: https://google-analytics.com/urchin.js
passed: false,false,false,false,false
Google Tag Manager
Tests whether the browser blocks the page from loading the tracker at https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
              
Passed
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
passed: true,true,true,true,true
Passed
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
passed: true,true,true,true,true
Failed
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
passed: false,false,false,false,false
Failed
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
passed: false,false,false,false,false
Failed
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
passed: false,false,false,false,false
Passed
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
passed: true,true,true,true,true
Passed
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
passed: true,true,true,true,true
Failed
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
passed: false,false,false,false,false
Passed
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
passed: true
Failed
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
passed: false,false,false
Failed
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
passed: false,false,false,false,false
Failed
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
passed: false,false,false,false,false
Index Exchange
Tests whether the browser blocks the page from loading the tracker at https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
              
Passed
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
passed: true,true,true,true,true
Passed
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
passed: true,true,true,true,true
Failed
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
passed: false,false,false,false,false
Failed
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
passed: false,false,false,false,false
Passed
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
passed: true,true,true,true,true
Passed
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
passed: true,true,true,true,true
Passed
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
passed: true,true,true,true,true
Failed
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
passed: false,false,false,false,false
Passed
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
passed: true
Failed
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
passed: false,false,false
Failed
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
passed: false,false,false,false,false
Failed
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
passed: false,false,false,false,false
New Relic
Tests whether the browser blocks the page from loading the tracker at https://js-agent.newrelic.com/nr-1212.min.js
              
Passed
url: https://js-agent.newrelic.com/nr-1212.min.js
passed: true,true,true,true,true
Passed
url: https://js-agent.newrelic.com/nr-1212.min.js
passed: true,true,true,true,true
Failed
url: https://js-agent.newrelic.com/nr-1212.min.js
passed: false,false,false,false,false
Failed
url: https://js-agent.newrelic.com/nr-1212.min.js
passed: false,false,false,false,false
Passed
url: https://js-agent.newrelic.com/nr-1212.min.js
passed: true,true,true,true,true
Passed
url: https://js-agent.newrelic.com/nr-1212.min.js
passed: true,true,true,true,true
Passed
url: https://js-agent.newrelic.com/nr-1212.min.js
passed: true,true,true,true,true
Failed
url: https://js-agent.newrelic.com/nr-1212.min.js
passed: false,false,false,false,false
Failed
url: https://js-agent.newrelic.com/nr-1212.min.js
passed: false
Failed
url: https://js-agent.newrelic.com/nr-1212.min.js
passed: false,false,false
Failed
url: https://js-agent.newrelic.com/nr-1212.min.js
passed: false,false,false,false,false
Failed
url: https://js-agent.newrelic.com/nr-1212.min.js
passed: false,false,false,false,false
Quantcast
Tests whether the browser blocks the page from loading the tracker at https://pixel.quantserve.com/pixel
              
Passed
url: https://pixel.quantserve.com/pixel
passed: true,true,true,true,true
Passed
url: https://pixel.quantserve.com/pixel
passed: true,true,true,true,true
Failed
url: https://pixel.quantserve.com/pixel
passed: false,false,false,false,false
Failed
url: https://pixel.quantserve.com/pixel
passed: false,false,false,false,false
Passed
url: https://pixel.quantserve.com/pixel
passed: true,true,true,true,true
Passed
url: https://pixel.quantserve.com/pixel
passed: true,true,true,true,true
Passed
url: https://pixel.quantserve.com/pixel
passed: true,true,true,true,true
Failed
url: https://pixel.quantserve.com/pixel
passed: false,false,false,false,false
Passed
url: https://pixel.quantserve.com/pixel
passed: true
Failed
url: https://pixel.quantserve.com/pixel
passed: false,false,false
Failed
url: https://pixel.quantserve.com/pixel
passed: false,false,false,false,false
Failed
url: https://pixel.quantserve.com/pixel
passed: false,false,false,false,false
Scorecard Research Beacon
Tests whether the browser blocks the page from loading the tracker at https://sb.scorecardresearch.com/internal-c2/default/cs.js
              
Passed
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
passed: true,true,true,true,true
Passed
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
passed: true,true,true,true,true
Failed
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
passed: false,false,false,false,false
Failed
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
passed: false,false,false,false,false
Passed
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
passed: true,true,true,true,true
Passed
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
passed: true,true,true,true,true
Passed
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
passed: true,true,true,true,true
Failed
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
passed: false,false,false,false,false
Failed
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
passed: false
Failed
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
passed: false,false,false
Failed
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
passed: false,false,false,false,false
Failed
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
passed: false,false,false,false,false
Taboola
Tests whether the browser blocks the page from loading the tracker at https://trc.taboola.com/futureplc-tomsguide/trc/3/json
              
Passed
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
passed: true,true,true,true,true
Passed
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
passed: true,true,true,true,true
Failed
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
passed: false,false,false,false,false
Failed
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
passed: false,false,false,false,false
Passed
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
passed: true,true,true,true,true
Passed
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
passed: true,true,true,true,true
Passed
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
passed: true,true,true,true,true
Failed
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
passed: false,false,false,false,false
Failed
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
passed: false
Failed
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
passed: false,false,false
Failed
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
passed: false,false,false,false,false
Failed
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
passed: false,false,false,false,false
Twitter pixel
Tests whether the browser blocks the page from loading the tracker at https://t.co/i/adsct
              
Passed
url: https://t.co/i/adsct
passed: true,true,true,true,true
Passed
url: https://t.co/i/adsct
passed: true,true,true,true,true
Failed
url: https://t.co/i/adsct
passed: false,false,false,false,false
Failed
url: https://t.co/i/adsct
passed: false,false,false,false,false
Passed
url: https://t.co/i/adsct
passed: true,true,true,true,true
Passed
url: https://t.co/i/adsct
passed: true,true,true,true,true
Passed
url: https://t.co/i/adsct
passed: true,true,true,true,true
Failed
url: https://t.co/i/adsct
passed: false,false,false,false,false
Failed
url: https://t.co/i/adsct
passed: false
Failed
url: https://t.co/i/adsct
passed: false,false,false
Failed
url: https://t.co/i/adsct
passed: false,false,false,false,false
Failed
url: https://t.co/i/adsct
passed: false,false,false,false,false
Yandex Ads
Tests whether the browser blocks the page from loading the tracker at https://yandex.ru/ads/system/header-bidding.js
              
Passed
url: https://yandex.ru/ads/system/header-bidding.js
passed: true,true,true,true,true
Passed
url: https://yandex.ru/ads/system/header-bidding.js
passed: true,true,true,true,true
Failed
url: https://yandex.ru/ads/system/header-bidding.js
passed: false,false,false,false,false
Failed
url: https://yandex.ru/ads/system/header-bidding.js
passed: false,false,false,false,false
Failed
url: https://yandex.ru/ads/system/header-bidding.js
passed: false,false,false,false,false
Passed
url: https://yandex.ru/ads/system/header-bidding.js
passed: true,true,true,true,true
Passed
url: https://yandex.ru/ads/system/header-bidding.js
passed: true,true,true,true,true
Failed
url: https://yandex.ru/ads/system/header-bidding.js
passed: false,false,false,false,false
Failed
url: https://yandex.ru/ads/system/header-bidding.js
passed: false
Failed
url: https://yandex.ru/ads/system/header-bidding.js
passed: false,false,false
Failed
url: https://yandex.ru/ads/system/header-bidding.js
passed: false,false,false,false,false
Failed
url: https://yandex.ru/ads/system/header-bidding.js
passed: false,false,false,false,false
Tracking cookie protection testsWhich browsers block important known tracking cookies?
A large fraction of web pages on the web have hidden third-party trackers that read and write cookies in your browser. These cookies can be used to track your browsing across websites. This section checks to see if a browser stops cross-site tracking by cookies from 20 of the largest trackers listed by https://whotracks.me.`,
        
Adobe
Tests whether the browser stops cookies from munchkin.marketo.net from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://munchkin.marketo.net/munchkin.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://munchkin.marketo.net/munchkin.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://munchkin.marketo.net/munchkin.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://munchkin.marketo.net/munchkin.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://munchkin.marketo.net/munchkin.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://munchkin.marketo.net/munchkin.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://munchkin.marketo.net/munchkin.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://munchkin.marketo.net/munchkin.js
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://munchkin.marketo.net/munchkin.js
cookieFound: false
Passed
passed: true,true,true
url: https://munchkin.marketo.net/munchkin.js
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://munchkin.marketo.net/munchkin.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://munchkin.marketo.net/munchkin.js
cookieFound: false,false,false,false,false
Adobe Audience Manager
Tests whether the browser stops cookies from dpm.demdex.net from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://dpm.demdex.net/ibs
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dpm.demdex.net/ibs
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dpm.demdex.net/ibs
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dpm.demdex.net/ibs
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dpm.demdex.net/ibs
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dpm.demdex.net/ibs
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dpm.demdex.net/ibs
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dpm.demdex.net/ibs
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://dpm.demdex.net/ibs
cookieFound: false
Passed
passed: true,true,true
url: https://dpm.demdex.net/ibs
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://dpm.demdex.net/ibs
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dpm.demdex.net/ibs
cookieFound: false,false,false,false,false
Amazon adsystem
Tests whether the browser stops cookies from s.amazon-adsystem.com from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://s.amazon-adsystem.com/dcm
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://s.amazon-adsystem.com/dcm
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://s.amazon-adsystem.com/dcm
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://s.amazon-adsystem.com/dcm
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://s.amazon-adsystem.com/dcm
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://s.amazon-adsystem.com/dcm
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://s.amazon-adsystem.com/dcm
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://s.amazon-adsystem.com/dcm
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://s.amazon-adsystem.com/dcm
cookieFound: false
Passed
passed: true,true,true
url: https://s.amazon-adsystem.com/dcm
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://s.amazon-adsystem.com/dcm
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://s.amazon-adsystem.com/dcm
cookieFound: false,false,false,false,false
AppNexus
Tests whether the browser stops cookies from ib.adnxs.com from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://ib.adnxs.com/px?id=178248&t=1
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://ib.adnxs.com/px?id=178248&t=1
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://ib.adnxs.com/px?id=178248&t=1
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://ib.adnxs.com/px?id=178248&t=1
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://ib.adnxs.com/px?id=178248&t=1
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://ib.adnxs.com/px?id=178248&t=1
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://ib.adnxs.com/px?id=178248&t=1
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://ib.adnxs.com/px?id=178248&t=1
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://ib.adnxs.com/px?id=178248&t=1
cookieFound: false
Passed
passed: true,true,true
url: https://ib.adnxs.com/px?id=178248&t=1
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://ib.adnxs.com/px?id=178248&t=1
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://ib.adnxs.com/px?id=178248&t=1
cookieFound: false,false,false,false,false
Bing Ads
Tests whether the browser stops cookies from bat.bing.com from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://bat.bing.com/bat.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://bat.bing.com/bat.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://bat.bing.com/bat.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://bat.bing.com/bat.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://bat.bing.com/bat.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://bat.bing.com/bat.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://bat.bing.com/bat.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://bat.bing.com/bat.js
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://bat.bing.com/bat.js
cookieFound: false
Passed
passed: true,true,true
url: https://bat.bing.com/bat.js
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://bat.bing.com/bat.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://bat.bing.com/bat.js
cookieFound: false,false,false,false,false
Chartbeat
Tests whether the browser stops cookies from static.chartbeat.com from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://static.chartbeat.com/js/chartbeat.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://static.chartbeat.com/js/chartbeat.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://static.chartbeat.com/js/chartbeat.js
cookieFound: false,false,false,false,false
Failed
passed: false,false,false,false,false
url: https://static.chartbeat.com/js/chartbeat.js
cookieFound: true,true,true,true,true
Passed
passed: true,true,true,true,true
url: https://static.chartbeat.com/js/chartbeat.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://static.chartbeat.com/js/chartbeat.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://static.chartbeat.com/js/chartbeat.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://static.chartbeat.com/js/chartbeat.js
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://static.chartbeat.com/js/chartbeat.js
cookieFound: false
Passed
passed: true,true,true
url: https://static.chartbeat.com/js/chartbeat.js
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://static.chartbeat.com/js/chartbeat.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://static.chartbeat.com/js/chartbeat.js
cookieFound: false,false,false,false,false
Criteo
Tests whether the browser stops cookies from dis.criteo.com from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
cookieFound: false
Passed
passed: true,true,true
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dis.criteo.com/dis/rtb/appnexus/cookiematch.aspx
cookieFound: false,false,false,false,false
DoubleClick (Google)
Tests whether the browser stops cookies from securepubads.g.doubleclick.net from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://securepubads.g.doubleclick.net/static/glade.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://securepubads.g.doubleclick.net/static/glade.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://securepubads.g.doubleclick.net/static/glade.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://securepubads.g.doubleclick.net/static/glade.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://securepubads.g.doubleclick.net/static/glade.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://securepubads.g.doubleclick.net/static/glade.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://securepubads.g.doubleclick.net/static/glade.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://securepubads.g.doubleclick.net/static/glade.js
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://securepubads.g.doubleclick.net/static/glade.js
cookieFound: false
Passed
passed: true,true,true
url: https://securepubads.g.doubleclick.net/static/glade.js
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://securepubads.g.doubleclick.net/static/glade.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://securepubads.g.doubleclick.net/static/glade.js
cookieFound: false,false,false,false,false
Facebook tracking
Tests whether the browser stops cookies from connect.facebook.net from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://connect.facebook.net/en_US/fbevents.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://connect.facebook.net/en_US/fbevents.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://connect.facebook.net/en_US/fbevents.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://connect.facebook.net/en_US/fbevents.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://connect.facebook.net/en_US/fbevents.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://connect.facebook.net/en_US/fbevents.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://connect.facebook.net/en_US/fbevents.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://connect.facebook.net/en_US/fbevents.js
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://connect.facebook.net/en_US/fbevents.js
cookieFound: false
Passed
passed: true,true,true
url: https://connect.facebook.net/en_US/fbevents.js
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://connect.facebook.net/en_US/fbevents.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://connect.facebook.net/en_US/fbevents.js
cookieFound: false,false,false,false,false
Google (third-party ad pixel)
Tests whether the browser stops cookies from www.google.com from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://www.google.com/pagead/1p-user-list/
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://www.google.com/pagead/1p-user-list/
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://www.google.com/pagead/1p-user-list/
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://www.google.com/pagead/1p-user-list/
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://www.google.com/pagead/1p-user-list/
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://www.google.com/pagead/1p-user-list/
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://www.google.com/pagead/1p-user-list/
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://www.google.com/pagead/1p-user-list/
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://www.google.com/pagead/1p-user-list/
cookieFound: false
Passed
passed: true,true,true
url: https://www.google.com/pagead/1p-user-list/
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://www.google.com/pagead/1p-user-list/
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://www.google.com/pagead/1p-user-list/
cookieFound: false,false,false,false,false
Google Analytics
Tests whether the browser stops cookies from google-analytics.com from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://google-analytics.com/urchin.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://google-analytics.com/urchin.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://google-analytics.com/urchin.js
cookieFound: false,false,false,false,false
Failed
passed: false,false,false,false,false
url: https://google-analytics.com/urchin.js
cookieFound: true,true,true,true,true
Passed
passed: true,true,true,true,true
url: https://google-analytics.com/urchin.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://google-analytics.com/urchin.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://google-analytics.com/urchin.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://google-analytics.com/urchin.js
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://google-analytics.com/urchin.js
cookieFound: false
Passed
passed: true,true,true
url: https://google-analytics.com/urchin.js
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://google-analytics.com/urchin.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://google-analytics.com/urchin.js
cookieFound: false,false,false,false,false
Google Tag Manager
Tests whether the browser stops cookies from www.googletagmanager.com from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
cookieFound: false,false,false,false,false
Failed
passed: false,false,false,false,false
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
cookieFound: true,true,true,true,true
Passed
passed: true,true,true,true,true
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
cookieFound: false
Passed
passed: true,true,true
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://www.googletagmanager.com/gtag.js?id=GTM-NX4SMZL
cookieFound: false,false,false,false,false
Index Exchange
Tests whether the browser stops cookies from dsum-sec.casalemedia.com from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
cookieFound: false
Passed
passed: true,true,true
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://dsum-sec.casalemedia.com/crum?cm_dsp_id=10&external_user_id=629685505537&C=1
cookieFound: false,false,false,false,false
New Relic
Tests whether the browser stops cookies from js-agent.newrelic.com from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://js-agent.newrelic.com/nr-1212.min.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://js-agent.newrelic.com/nr-1212.min.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://js-agent.newrelic.com/nr-1212.min.js
cookieFound: false,false,false,false,false
Failed
passed: false,false,false,false,false
url: https://js-agent.newrelic.com/nr-1212.min.js
cookieFound: true,true,true,true,true
Passed
passed: true,true,true,true,true
url: https://js-agent.newrelic.com/nr-1212.min.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://js-agent.newrelic.com/nr-1212.min.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://js-agent.newrelic.com/nr-1212.min.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://js-agent.newrelic.com/nr-1212.min.js
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://js-agent.newrelic.com/nr-1212.min.js
cookieFound: false
Passed
passed: true,true,true
url: https://js-agent.newrelic.com/nr-1212.min.js
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://js-agent.newrelic.com/nr-1212.min.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://js-agent.newrelic.com/nr-1212.min.js
cookieFound: false,false,false,false,false
Quantcast
Tests whether the browser stops cookies from pixel.quantserve.com from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://pixel.quantserve.com/pixel
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://pixel.quantserve.com/pixel
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://pixel.quantserve.com/pixel
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://pixel.quantserve.com/pixel
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://pixel.quantserve.com/pixel
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://pixel.quantserve.com/pixel
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://pixel.quantserve.com/pixel
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://pixel.quantserve.com/pixel
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://pixel.quantserve.com/pixel
cookieFound: false
Passed
passed: true,true,true
url: https://pixel.quantserve.com/pixel
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://pixel.quantserve.com/pixel
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://pixel.quantserve.com/pixel
cookieFound: false,false,false,false,false
Scorecard Research Beacon
Tests whether the browser stops cookies from sb.scorecardresearch.com from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
cookieFound: false,false,false,false,false
Failed
passed: false,false,false,false,false
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
cookieFound: true,true,true,true,true
Passed
passed: true,true,true,true,true
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
cookieFound: false
Passed
passed: true,true,true
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://sb.scorecardresearch.com/internal-c2/default/cs.js
cookieFound: false,false,false,false,false
Taboola
Tests whether the browser stops cookies from trc.taboola.com from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
cookieFound: false
Passed
passed: true,true,true
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://trc.taboola.com/futureplc-tomsguide/trc/3/json
cookieFound: false,false,false,false,false
Twitter pixel
Tests whether the browser stops cookies from t.co from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://t.co/i/adsct
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://t.co/i/adsct
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://t.co/i/adsct
cookieFound: false,false,false,false,false
Failed
passed: false,false,false,true,false
url: https://t.co/i/adsct
cookieFound: true,true,true,false,true
Passed
passed: true,true,true,true,true
url: https://t.co/i/adsct
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://t.co/i/adsct
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://t.co/i/adsct
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://t.co/i/adsct
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://t.co/i/adsct
cookieFound: false
Passed
passed: true,true,true
url: https://t.co/i/adsct
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://t.co/i/adsct
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://t.co/i/adsct
cookieFound: false,false,false,false,false
Yandex Ads
Tests whether the browser stops cookies from yandex.ru from tracking users across websites.
              
Passed
passed: true,true,true,true,true
url: https://yandex.ru/ads/system/header-bidding.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://yandex.ru/ads/system/header-bidding.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://yandex.ru/ads/system/header-bidding.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://yandex.ru/ads/system/header-bidding.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://yandex.ru/ads/system/header-bidding.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://yandex.ru/ads/system/header-bidding.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://yandex.ru/ads/system/header-bidding.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://yandex.ru/ads/system/header-bidding.js
cookieFound: false,false,false,false,false
Passed
passed: true
url: https://yandex.ru/ads/system/header-bidding.js
cookieFound: false
Passed
passed: true,true,true
url: https://yandex.ru/ads/system/header-bidding.js
cookieFound: false,false,false
Passed
passed: true,true,true,true,true
url: https://yandex.ru/ads/system/header-bidding.js
cookieFound: false,false,false,false,false
Passed
passed: true,true,true,true,true
url: https://yandex.ru/ads/system/header-bidding.js
cookieFound: false,false,false,false,false
Cross-session first-party tracking testsWhich browsers prevent websites from tracking you across browser sessions?
A common vulnerability of web browsers is that they allow websites ("first parties") to 'tag' your browser with some tracking data. This tag can be used to re-identify you when you return to a website you visited before. This category of leaks can be prevented by browser if they clean or isolate data between browser sessions. (In cases where a user has logged into a website or entered detailed information, it may be justifiable for a browser to retain information across sessions. These tests check when no such justification exists: when you have entered no significant information into a website, will the browser still retain data that allows you to be tracked across sessions?)
        
Alt-Svc
Alt-Svc allows the server to indicate to the web browser that a resource should be loaded on a different server. Because this is a persistent setting, it could be used to track users across websites if it is not correctly partitioned.
              
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h2, h2, h2, h2, h2

result, different session: h2, h2, h2, h2, h2

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h2

result, different session: h2

unsupported: true

passed: undefined

test failed: false
Unsupported
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h2, h2, h2

result, different session: h2, h2, h2

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
CacheStorage
The Cache API is a content storage mechanism originally introduced to support ServiceWorkers. If the same Cache object is accessible to multiple websites, it can be abused to track users.
              
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c_1p,
6e0d0559-9ac2-4abc-9025-582a29e22a8f_1p,
066762a4-4e2b-4344-82a6-b11540d563ab_1p,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee_1p,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2_1p

result, different session: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
ead45dd4-f0d1-4ab5-af71-11d423c35903_1p,
139bd1c9-0bf8-414e-809a-d12f37b34aab_1p,
33d68d0c-5918-4ede-a519-aff107d2ce1c_1p,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5_1p,
fd597d68-f9be-48a0-9656-b7d0b2e140be_1p

result, different session: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb_1p,
42c2adb9-619b-4b8f-a843-84ede2df77e1_1p,
8b52f9c3-d53f-4dad-91bb-5e162d085e46_1p,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f_1p,
57161958-02cf-461a-8b94-db8e32150e06_1p

result, different session: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
7d86a137-2f75-4c34-aade-b50776a7fd40_1p,
142f7096-d2d9-4c89-ba22-d0a3b8055308_1p,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c_1p,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd_1p,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346_1p

result, different session: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

result, different session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

result, different session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

result, different session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
92ee92f4-0792-4f9a-ba56-45989cd80572_1p,
606b3a57-c3cf-45fe-a079-3b1566f6abd1_1p,
d40bde18-63e0-419d-b233-1dc09c4c5f94_1p,
df73d3b8-d807-46cf-9f77-65d30bc06fa4_1p,
ba872d52-1a40-47cf-9575-e78f927ff5b3_1p

result, different session: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: Error: undefined is not an object (evaluating 'cacheKeys[0].url')

result, different session: Error: undefined is not an object (evaluating 'cacheKeys[0].url')

unsupported: true

passed: undefined

test failed: false
Unsupported
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

result, different session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
3a254b81-9486-4572-a38a-b4bd82252b3e_1p,
c0ee6f27-d50e-4975-9a0a-523696e05cbf_1p,
89d605d6-80b6-488e-8454-c4de9c5dfe6b_1p,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6_1p,
d0752b25-d241-400f-a69f-21c3ff242a56_1p

result, different session: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078_1p,
fad257e3-bcc3-4aa7-b541-2b4c42aee488_1p,
3da92bf0-3ab6-4d33-9452-fac73e80ebba_1p,
d276eb0f-b849-45c6-97b9-57596abf3678_1p,
b64d49aa-de07-400a-9631-2935db968706_1p

result, different session: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
cookie (HTTP)
The cookie, first introduced by Netscape in 1994, is a small amount of data stored by your browser on a website's behalf. It has legitimate uses, but it is also the classic cross-site tracking mechanism, and today still the most popular method of tracking users across websites. Browsers can stop cookies from being used for cross-site tracking by either blocking or partitioning them.
              
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c_1p_http,
6e0d0559-9ac2-4abc-9025-582a29e22a8f_1p_http,
066762a4-4e2b-4344-82a6-b11540d563ab_1p_http,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee_1p_http,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2_1p_http

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: 
ead45dd4-f0d1-4ab5-af71-11d423c35903_1p_http,
139bd1c9-0bf8-414e-809a-d12f37b34aab_1p_http,
33d68d0c-5918-4ede-a519-aff107d2ce1c_1p_http,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5_1p_http,
fd597d68-f9be-48a0-9656-b7d0b2e140be_1p_http

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb_1p_http,
42c2adb9-619b-4b8f-a843-84ede2df77e1_1p_http,
8b52f9c3-d53f-4dad-91bb-5e162d085e46_1p_http,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f_1p_http,
57161958-02cf-461a-8b94-db8e32150e06_1p_http

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: 
7d86a137-2f75-4c34-aade-b50776a7fd40_1p_http,
142f7096-d2d9-4c89-ba22-d0a3b8055308_1p_http,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c_1p_http,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd_1p_http,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346_1p_http

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: 
784004e0-9aeb-47e5-9c5f-12b883e353ac_1p_http,
eab30b13-9ecc-48af-9a1a-cae23815efbc_1p_http,
a652f390-6821-4125-9fd0-c118d2a1e7a2_1p_http,
09045837-46dd-45c8-ad94-7b7c171aeef0_1p_http,
7cb67be0-5127-4489-aeb1-7164e3b5af92_1p_http

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: 
79152685-f997-47b4-aef1-9e391daef62f_1p_http,
3863988a-95b2-4b54-a63c-ce0727dc2084_1p_http,
218debd1-a05b-48c0-b827-e4249b90263c_1p_http,
6d855fa7-5396-4b31-ac62-f63859f5a3ef_1p_http,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75_1p_http

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: 
0a2e161c-e0f2-4394-8989-cbcb8cc210b9_1p_http,
51502f82-4cca-4007-a857-3c34c3db9bf8_1p_http,
37efbb8c-8f64-4994-8d72-bf40bf93d2bb_1p_http,
3b85b9c3-ce06-4354-ab1d-998c750dab4a_1p_http,
70cc2ae2-5861-4824-85db-4f905ce0c785_1p_http

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: 
92ee92f4-0792-4f9a-ba56-45989cd80572_1p_http,
606b3a57-c3cf-45fe-a079-3b1566f6abd1_1p_http,
d40bde18-63e0-419d-b233-1dc09c4c5f94_1p_http,
df73d3b8-d807-46cf-9f77-65d30bc06fa4_1p_http,
ba872d52-1a40-47cf-9575-e78f927ff5b3_1p_http

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: null

result, different session: null

unsupported: true

passed: undefined

test failed: false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: 
37c72365-5852-42a4-9c27-47b4aa879b24_1p_http,
323291dd-ef8a-484c-aaf9-641497dea358_1p_http,
bd429ec2-5e93-4afc-8a05-5781fb0abad6_1p_http

result, different session: , , 

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: 
3a254b81-9486-4572-a38a-b4bd82252b3e_1p_http,
c0ee6f27-d50e-4975-9a0a-523696e05cbf_1p_http,
89d605d6-80b6-488e-8454-c4de9c5dfe6b_1p_http,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6_1p_http,
d0752b25-d241-400f-a69f-21c3ff242a56_1p_http

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078_1p_http,
fad257e3-bcc3-4aa7-b541-2b4c42aee488_1p_http,
3da92bf0-3ab6-4d33-9452-fac73e80ebba_1p_http,
d276eb0f-b849-45c6-97b9-57596abf3678_1p_http,
b64d49aa-de07-400a-9631-2935db968706_1p_http

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
cookie (JS)
The cookie, first introduced by Netscape in 1994, is a small amount of data stored by your browser on a website's behalf. It has legitimate uses, but it is also the classic cross-site tracking mechanism, and today still the most popular method of tracking users across websites. Browsers can stop cookies from being used for cross-site tracking by either blocking or partitioning them.
              
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c_1p_js,
6e0d0559-9ac2-4abc-9025-582a29e22a8f_1p_js,
066762a4-4e2b-4344-82a6-b11540d563ab_1p_js,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee_1p_js,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2_1p_js

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: 
ead45dd4-f0d1-4ab5-af71-11d423c35903_1p_js,
139bd1c9-0bf8-414e-809a-d12f37b34aab_1p_js,
33d68d0c-5918-4ede-a519-aff107d2ce1c_1p_js,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5_1p_js,
fd597d68-f9be-48a0-9656-b7d0b2e140be_1p_js

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb_1p_js,
42c2adb9-619b-4b8f-a843-84ede2df77e1_1p_js,
8b52f9c3-d53f-4dad-91bb-5e162d085e46_1p_js,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f_1p_js,
57161958-02cf-461a-8b94-db8e32150e06_1p_js

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: 
7d86a137-2f75-4c34-aade-b50776a7fd40_1p_js,
142f7096-d2d9-4c89-ba22-d0a3b8055308_1p_js,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c_1p_js,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd_1p_js,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346_1p_js

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: 
784004e0-9aeb-47e5-9c5f-12b883e353ac_1p_js,
eab30b13-9ecc-48af-9a1a-cae23815efbc_1p_js,
a652f390-6821-4125-9fd0-c118d2a1e7a2_1p_js,
09045837-46dd-45c8-ad94-7b7c171aeef0_1p_js,
7cb67be0-5127-4489-aeb1-7164e3b5af92_1p_js

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: 
79152685-f997-47b4-aef1-9e391daef62f_1p_js,
3863988a-95b2-4b54-a63c-ce0727dc2084_1p_js,
218debd1-a05b-48c0-b827-e4249b90263c_1p_js,
6d855fa7-5396-4b31-ac62-f63859f5a3ef_1p_js,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75_1p_js

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: 
0a2e161c-e0f2-4394-8989-cbcb8cc210b9_1p_js,
51502f82-4cca-4007-a857-3c34c3db9bf8_1p_js,
37efbb8c-8f64-4994-8d72-bf40bf93d2bb_1p_js,
3b85b9c3-ce06-4354-ab1d-998c750dab4a_1p_js,
70cc2ae2-5861-4824-85db-4f905ce0c785_1p_js

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: 
92ee92f4-0792-4f9a-ba56-45989cd80572_1p_js,
606b3a57-c3cf-45fe-a079-3b1566f6abd1_1p_js,
d40bde18-63e0-419d-b233-1dc09c4c5f94_1p_js,
df73d3b8-d807-46cf-9f77-65d30bc06fa4_1p_js,
ba872d52-1a40-47cf-9575-e78f927ff5b3_1p_js

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: null

result, different session: null

unsupported: true

passed: undefined

test failed: false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: 
37c72365-5852-42a4-9c27-47b4aa879b24_1p_js,
323291dd-ef8a-484c-aaf9-641497dea358_1p_js,
bd429ec2-5e93-4afc-8a05-5781fb0abad6_1p_js

result, different session: , , 

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: 
3a254b81-9486-4572-a38a-b4bd82252b3e_1p_js,
c0ee6f27-d50e-4975-9a0a-523696e05cbf_1p_js,
89d605d6-80b6-488e-8454-c4de9c5dfe6b_1p_js,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6_1p_js,
d0752b25-d241-400f-a69f-21c3ff242a56_1p_js

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078_1p_js,
fad257e3-bcc3-4aa7-b541-2b4c42aee488_1p_js,
3da92bf0-3ab6-4d33-9452-fac73e80ebba_1p_js,
d276eb0f-b849-45c6-97b9-57596abf3678_1p_js,
b64d49aa-de07-400a-9631-2935db968706_1p_js

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
CookieStore
The Cookie Store API is an alternative asynchronous API for managing cookies, supported by some browsers.
              
Passed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c_1p,
6e0d0559-9ac2-4abc-9025-582a29e22a8f_1p,
066762a4-4e2b-4344-82a6-b11540d563ab_1p,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee_1p,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
ead45dd4-f0d1-4ab5-af71-11d423c35903_1p,
139bd1c9-0bf8-414e-809a-d12f37b34aab_1p,
33d68d0c-5918-4ede-a519-aff107d2ce1c_1p,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5_1p,
fd597d68-f9be-48a0-9656-b7d0b2e140be_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb_1p,
42c2adb9-619b-4b8f-a843-84ede2df77e1_1p,
8b52f9c3-d53f-4dad-91bb-5e162d085e46_1p,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f_1p,
57161958-02cf-461a-8b94-db8e32150e06_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
7d86a137-2f75-4c34-aade-b50776a7fd40_1p,
142f7096-d2d9-4c89-ba22-d0a3b8055308_1p,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c_1p,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd_1p,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different session: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different session: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different session: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
92ee92f4-0792-4f9a-ba56-45989cd80572_1p,
606b3a57-c3cf-45fe-a079-3b1566f6abd1_1p,
d40bde18-63e0-419d-b233-1dc09c4c5f94_1p,
df73d3b8-d807-46cf-9f77-65d30bc06fa4_1p,
ba872d52-1a40-47cf-9575-e78f927ff5b3_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: Error: Unsupported

result, different session: Error: Unsupported

unsupported: true

passed: undefined

test failed: false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different session: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Passed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
3a254b81-9486-4572-a38a-b4bd82252b3e_1p,
c0ee6f27-d50e-4975-9a0a-523696e05cbf_1p,
89d605d6-80b6-488e-8454-c4de9c5dfe6b_1p,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6_1p,
d0752b25-d241-400f-a69f-21c3ff242a56_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078_1p,
fad257e3-bcc3-4aa7-b541-2b4c42aee488_1p,
3da92bf0-3ab6-4d33-9452-fac73e80ebba_1p,
d276eb0f-b849-45c6-97b9-57596abf3678_1p,
b64d49aa-de07-400a-9631-2935db968706_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
CSS cache
CSS stylesheets are cached, and if that cache is shared between websites, it can be used to track users across sites.
              
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_7540660980953051,
fake_6185958997376233,
fake_4720572742680742,
fake_2063090027373531,
fake_8668874630975729

result, different session: 
fake_11332853396787423,
fake_9995231001210816,
fake_8784122809772732,
fake_32178962927840016,
fake_3447120822403251

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_9225924692445988,
fake_045094383073702504,
fake_24498165250095982,
fake_10919949505840965,
fake_676765106206342

result, different session: 
fake_0016347013885753547,
fake_17718263598485784,
fake_2937895990504511,
fake_8828775235261859,
fake_6927154229196599

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_5584044775876096,
fake_20922638121754877,
fake_638614905288434,
fake_4397355182469782,
fake_6001007132978664

result, different session: 
fake_9403422454614481,
fake_5162540144017276,
fake_5169283151374635,
fake_7353029775033506,
fake_1771663374365695

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_21216792879789037,
fake_83589987081755,
fake_41345125025268503,
fake_5620390606798547,
fake_3745806695217824

result, different session: 
fake_39404426075136323,
fake_030930696118337986,
fake_16678980636696172,
fake_6206039983713179,
fake_06092157091357242

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_4889636828221493,
fake_5584194277899299,
fake_8146062992202778,
fake_11203680032159058,
fake_10526802195654006

result, different session: 
fake_5706586236913096,
fake_17640659525219937,
fake_5732658111949502,
fake_3898077900958079,
fake_7447705403773417

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_8320180611020433,
fake_7347279999008682,
fake_7509704621232911,
fake_7967376782621247,
fake_8006008017856916

result, different session: 
fake_768892352926835,
fake_3057851048313127,
fake_5071488891927276,
fake_896913164614942,
fake_42506477921977015

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_0019589319625246127,
fake_6211570844274388,
fake_7473430068427458,
fake_09674917951547202,
fake_9913628438933066

result, different session: 
fake_8411016649394036,
fake_4009087824563635,
fake_20173529561694914,
fake_42538054514094625,
fake_07002726490263522

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_7015490329051348,
fake_26735654307163337,
fake_8147848966735838,
fake_11551418836193306,
fake_6160731674949274

result, different session: 
fake_8327334534590911,
fake_19353624934862323,
fake_8875214679115044,
fake_40182437703098217,
fake_8135815931506727

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: fake_9256307750801749

result, different session: fake_9410986574987643

unsupported: false

passed: true

test failed: false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_8553411731715832,
fake_4123892120278143,
fake_39174878462036467

result, different session: 
fake_4791951932225298,
fake_5961697616445887,
fake_7081991048858112

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_06423105909610083,
fake_9691063397206081,
fake_7402662405807223,
fake_24408420983996404,
fake_6284639732671113

result, different session: 
fake_6558571187282802,
fake_009874853137581852,
fake_7004454245832032,
fake_4514595439105371,
fake_08767326760220295

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_5813940034943081,
fake_11065158550436927,
fake_7701104266596324,
fake_8628514523768189,
fake_04327017492928209

result, different session: 
fake_5798538837452569,
fake_03336017890079579,
fake_5154519427795463,
fake_4532335209155054,
fake_9473657684460726

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
favicon cache
A favicon is an icon that represents a website, typically shown in browser tab and bookmarks menu. If the favicon cache is not partitioned, it can be used to track users across websites.
              
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1

result, different session: 2

unsupported: false

passed: true

test failed: false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1

result, different session: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 2, 2, 2, 2, 2

result, different session: 4, 4, 4, 4, 4

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
fetch cache
When a resource is received via the Fetch API, it is frequently cached. That cache can potentially be abused for cross-site tracking.
              
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 2

result, different session: 3

unsupported: false

passed: true

test failed: false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1

result, different session: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
font cache
Web fonts are sometimes stored in their own cache, which is vulnerable to being abused for cross-site tracking.
              
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 2

result, different session: 3

unsupported: false

passed: true

test failed: false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1

result, different session: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
iframe cache
An iframe is an element in a web page than allows websites to embed a second web page. Caching of this web page could be abused for cross-site tracking.
              
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 2

result, different session: 3

unsupported: false

passed: true

test failed: false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1

result, different session: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
image cache
Caching of images in web browsers is a standard behavior. But if that cache leaks between websites, it can be abused for cross-site tracking.
              
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 2

result, different session: 3

unsupported: false

passed: true

test failed: false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1

result, different session: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
indexedDB
The IndexedDB API exposes a transactional database to web pages. That database can be used to track users across websites, unless it is partitioned.
              
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c_1p,
6e0d0559-9ac2-4abc-9025-582a29e22a8f_1p,
066762a4-4e2b-4344-82a6-b11540d563ab_1p,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee_1p,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2_1p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
ead45dd4-f0d1-4ab5-af71-11d423c35903_1p,
139bd1c9-0bf8-414e-809a-d12f37b34aab_1p,
33d68d0c-5918-4ede-a519-aff107d2ce1c_1p,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5_1p,
fd597d68-f9be-48a0-9656-b7d0b2e140be_1p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb_1p,
42c2adb9-619b-4b8f-a843-84ede2df77e1_1p,
8b52f9c3-d53f-4dad-91bb-5e162d085e46_1p,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f_1p,
57161958-02cf-461a-8b94-db8e32150e06_1p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
7d86a137-2f75-4c34-aade-b50776a7fd40_1p,
142f7096-d2d9-4c89-ba22-d0a3b8055308_1p,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c_1p,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd_1p,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346_1p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
784004e0-9aeb-47e5-9c5f-12b883e353ac_1p,
eab30b13-9ecc-48af-9a1a-cae23815efbc_1p,
a652f390-6821-4125-9fd0-c118d2a1e7a2_1p,
09045837-46dd-45c8-ad94-7b7c171aeef0_1p,
7cb67be0-5127-4489-aeb1-7164e3b5af92_1p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
79152685-f997-47b4-aef1-9e391daef62f_1p,
3863988a-95b2-4b54-a63c-ce0727dc2084_1p,
218debd1-a05b-48c0-b827-e4249b90263c_1p,
6d855fa7-5396-4b31-ac62-f63859f5a3ef_1p,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75_1p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
0a2e161c-e0f2-4394-8989-cbcb8cc210b9_1p,
51502f82-4cca-4007-a857-3c34c3db9bf8_1p,
37efbb8c-8f64-4994-8d72-bf40bf93d2bb_1p,
3b85b9c3-ce06-4354-ab1d-998c750dab4a_1p,
70cc2ae2-5861-4824-85db-4f905ce0c785_1p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
92ee92f4-0792-4f9a-ba56-45989cd80572_1p,
606b3a57-c3cf-45fe-a079-3b1566f6abd1_1p,
d40bde18-63e0-419d-b233-1dc09c4c5f94_1p,
df73d3b8-d807-46cf-9f77-65d30bc06fa4_1p,
ba872d52-1a40-47cf-9575-e78f927ff5b3_1p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: undefined

result, different session: undefined

unsupported: true

passed: undefined

test failed: false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
37c72365-5852-42a4-9c27-47b4aa879b24_1p,
323291dd-ef8a-484c-aaf9-641497dea358_1p,
bd429ec2-5e93-4afc-8a05-5781fb0abad6_1p

result, different session: undefined

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
3a254b81-9486-4572-a38a-b4bd82252b3e_1p,
c0ee6f27-d50e-4975-9a0a-523696e05cbf_1p,
89d605d6-80b6-488e-8454-c4de9c5dfe6b_1p,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6_1p,
d0752b25-d241-400f-a69f-21c3ff242a56_1p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078_1p,
fad257e3-bcc3-4aa7-b541-2b4c42aee488_1p,
3da92bf0-3ab6-4d33-9452-fac73e80ebba_1p,
d276eb0f-b849-45c6-97b9-57596abf3678_1p,
b64d49aa-de07-400a-9631-2935db968706_1p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
localStorage
The localStorage API gives websites access to a key-value database that will remain available across visits. If the localStorage API is not partitioned or blocked, it can also be used to track users across websites.
              
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c_1p,
6e0d0559-9ac2-4abc-9025-582a29e22a8f_1p,
066762a4-4e2b-4344-82a6-b11540d563ab_1p,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee_1p,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
ead45dd4-f0d1-4ab5-af71-11d423c35903_1p,
139bd1c9-0bf8-414e-809a-d12f37b34aab_1p,
33d68d0c-5918-4ede-a519-aff107d2ce1c_1p,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5_1p,
fd597d68-f9be-48a0-9656-b7d0b2e140be_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb_1p,
42c2adb9-619b-4b8f-a843-84ede2df77e1_1p,
8b52f9c3-d53f-4dad-91bb-5e162d085e46_1p,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f_1p,
57161958-02cf-461a-8b94-db8e32150e06_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
7d86a137-2f75-4c34-aade-b50776a7fd40_1p,
142f7096-d2d9-4c89-ba22-d0a3b8055308_1p,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c_1p,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd_1p,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
784004e0-9aeb-47e5-9c5f-12b883e353ac_1p,
eab30b13-9ecc-48af-9a1a-cae23815efbc_1p,
a652f390-6821-4125-9fd0-c118d2a1e7a2_1p,
09045837-46dd-45c8-ad94-7b7c171aeef0_1p,
7cb67be0-5127-4489-aeb1-7164e3b5af92_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
79152685-f997-47b4-aef1-9e391daef62f_1p,
3863988a-95b2-4b54-a63c-ce0727dc2084_1p,
218debd1-a05b-48c0-b827-e4249b90263c_1p,
6d855fa7-5396-4b31-ac62-f63859f5a3ef_1p,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
0a2e161c-e0f2-4394-8989-cbcb8cc210b9_1p,
51502f82-4cca-4007-a857-3c34c3db9bf8_1p,
37efbb8c-8f64-4994-8d72-bf40bf93d2bb_1p,
3b85b9c3-ce06-4354-ab1d-998c750dab4a_1p,
70cc2ae2-5861-4824-85db-4f905ce0c785_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
92ee92f4-0792-4f9a-ba56-45989cd80572_1p,
606b3a57-c3cf-45fe-a079-3b1566f6abd1_1p,
d40bde18-63e0-419d-b233-1dc09c4c5f94_1p,
df73d3b8-d807-46cf-9f77-65d30bc06fa4_1p,
ba872d52-1a40-47cf-9575-e78f927ff5b3_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: null

result, different session: null

unsupported: true

passed: undefined

test failed: false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
37c72365-5852-42a4-9c27-47b4aa879b24_1p,
323291dd-ef8a-484c-aaf9-641497dea358_1p,
bd429ec2-5e93-4afc-8a05-5781fb0abad6_1p

result, different session: , , 

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
3a254b81-9486-4572-a38a-b4bd82252b3e_1p,
c0ee6f27-d50e-4975-9a0a-523696e05cbf_1p,
89d605d6-80b6-488e-8454-c4de9c5dfe6b_1p,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6_1p,
d0752b25-d241-400f-a69f-21c3ff242a56_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078_1p,
fad257e3-bcc3-4aa7-b541-2b4c42aee488_1p,
3da92bf0-3ab6-4d33-9452-fac73e80ebba_1p,
d276eb0f-b849-45c6-97b9-57596abf3678_1p,
b64d49aa-de07-400a-9631-2935db968706_1p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
prefetch cache
A <link rel='prefetch'...> suggests to browsers they should fetch a resource ahead of time and cache it. But if browsers don't partition this cache, it can be used to track users across websites.
              
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received

result, different session: 
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received

result, different session: 
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: Error: No requests received

result, different session: Error: No requests received

unsupported: true

passed: undefined

test failed: false
Unsupported
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 
Error: No requests received,
Error: No requests received,
Error: No requests received

result, different session: 
Error: No requests received,
Error: No requests received,
Error: No requests received

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Web SQL Database
The Web SQL Database is a deprecated web API for storing data in an SQL database.
              
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: Error: Web SQL is deprecated

result, different session: Error: Web SQL is deprecated

unsupported: true

passed: undefined

test failed: false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
XMLHttpRequest cache
Similar to the newer Fetch API, any resource received may be cached by the browser. The cache is potentially vulnerable to cross-site tracking attack.
              
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 2

result, different session: 3

unsupported: false

passed: true

test failed: false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1

result, different session: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Cross-session third-party tracking testsWhich browsers prevent third-party trackers from tracking you across browser sessions?
A common vulnerability of web browsers is that they allow third-party trackers to 'tag' your browser with some tracking data. This tag can be used to re-identify you when you return to a website you visited before. This category of leaks can be prevented by browser if they clean or isolate data between browser sessions. (In cases where a user has logged into a website or entered detailed information, it may be justifiable for a browser to retain information across sessions. These tests check when no such justification exists: when you have entered no significant information into a website, will the browser still retain data that allows you to be tracked across sessions?)
        
Alt-Svc
Alt-Svc allows the server to indicate to the web browser that a resource should be loaded on a different server. Because this is a persistent setting, it could be used to track users across websites if it is not correctly partitioned.
              
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h2, h2, h2, h2, h2

result, different session: h2, h2, h2, h2, h2

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h2

result, different session: h2

unsupported: true

passed: undefined

test failed: false
Unsupported
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h2, h2, h2

result, different session: h2, h2, h2

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async () => {
      // Clear Alt-Svc caching first.
      let responseText = "";
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/clear");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after clear:", responseText);
      // Store "h3" state in Alt-Svc cache
      for (let i = 0; i < 3; ++i) {
        await fetch(altSvcOrigin + "/set");
        await sleepMs(100);
      }
      responseText = await fetchText(altSvcOrigin + "/protocol");
      console.log("after set:", responseText);
    }

read: async () => {
      const protocol = await fetchText(altSvcOrigin + "/protocol");
      if ((new URL(location)).searchParams.get("thirdparty") === "same") {
        if (protocol !== "h3") {
          throw new Error("Unsupported");
        }
      }
      return protocol;
    }

result, same session: h3, h3, h3, h3, h3

result, different session: h2, h2, h2, h2, h2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
CacheStorage
The Cache API is a content storage mechanism originally introduced to support ServiceWorkers. If the same Cache object is accessible to multiple websites, it can be abused to track users.
              
Unsupported
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.

result, different session: 
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.

result, different session: 
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.,
Error: Failed to execute 'open' on 'CacheStorage': An attempt was made to break through the security policy of the user agent.

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb_3p,
42c2adb9-619b-4b8f-a843-84ede2df77e1_3p,
8b52f9c3-d53f-4dad-91bb-5e162d085e46_3p,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f_3p,
57161958-02cf-461a-8b94-db8e32150e06_3p

result, different session: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
7d86a137-2f75-4c34-aade-b50776a7fd40_3p,
142f7096-d2d9-4c89-ba22-d0a3b8055308_3p,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c_3p,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd_3p,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346_3p

result, different session: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

result, different session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

result, different session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

result, different session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
92ee92f4-0792-4f9a-ba56-45989cd80572_3p,
606b3a57-c3cf-45fe-a079-3b1566f6abd1_3p,
d40bde18-63e0-419d-b233-1dc09c4c5f94_3p,
df73d3b8-d807-46cf-9f77-65d30bc06fa4_3p,
ba872d52-1a40-47cf-9575-e78f927ff5b3_3p

result, different session: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: Error: undefined is not an object (evaluating 'cacheKeys[0].url')

result, different session: Error: undefined is not an object (evaluating 'cacheKeys[0].url')

unsupported: true

passed: undefined

test failed: false
Unsupported
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

result, different session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
3a254b81-9486-4572-a38a-b4bd82252b3e_3p,
c0ee6f27-d50e-4975-9a0a-523696e05cbf_3p,
89d605d6-80b6-488e-8454-c4de9c5dfe6b_3p,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6_3p,
d0752b25-d241-400f-a69f-21c3ff242a56_3p

result, different session: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      try {
        let cache = await caches.open("supercookies");
        cache.addAll([`test.css?key=${key}`]);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: async () => {
      let cache = await caches.open("supercookies");
      let cacheKeys = await cache.keys();
      let url = cacheKeys[0].url;
      return (new URL(url)).searchParams.get("key");
    }

result, same session: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078_3p,
fad257e3-bcc3-4aa7-b541-2b4c42aee488_3p,
3da92bf0-3ab6-4d33-9452-fac73e80ebba_3p,
d276eb0f-b849-45c6-97b9-57596abf3678_3p,
b64d49aa-de07-400a-9631-2935db968706_3p

result, different session: 
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url'),
Error: Cannot read properties of undefined (reading 'url')

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
cookie (HTTP)
The cookie, first introduced by Netscape in 1994, is a small amount of data stored by your browser on a website's behalf. It has legitimate uses, but it is also the classic cross-site tracking mechanism, and today still the most popular method of tracking users across websites. Browsers can stop cookies from being used for cross-site tracking by either blocking or partitioning them.
              
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c_3p_http,
6e0d0559-9ac2-4abc-9025-582a29e22a8f_3p_http,
066762a4-4e2b-4344-82a6-b11540d563ab_3p_http,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee_3p_http,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2_3p_http

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: 
ead45dd4-f0d1-4ab5-af71-11d423c35903_3p_http,
139bd1c9-0bf8-414e-809a-d12f37b34aab_3p_http,
33d68d0c-5918-4ede-a519-aff107d2ce1c_3p_http,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5_3p_http,
fd597d68-f9be-48a0-9656-b7d0b2e140be_3p_http

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: , , , , 

result, different session: , , , , 

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: 
7d86a137-2f75-4c34-aade-b50776a7fd40_3p_http,
142f7096-d2d9-4c89-ba22-d0a3b8055308_3p_http,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c_3p_http,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd_3p_http,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346_3p_http

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: 
784004e0-9aeb-47e5-9c5f-12b883e353ac_3p_http,
eab30b13-9ecc-48af-9a1a-cae23815efbc_3p_http,
a652f390-6821-4125-9fd0-c118d2a1e7a2_3p_http,
09045837-46dd-45c8-ad94-7b7c171aeef0_3p_http,
7cb67be0-5127-4489-aeb1-7164e3b5af92_3p_http

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: 
79152685-f997-47b4-aef1-9e391daef62f_3p_http,
3863988a-95b2-4b54-a63c-ce0727dc2084_3p_http,
218debd1-a05b-48c0-b827-e4249b90263c_3p_http,
6d855fa7-5396-4b31-ac62-f63859f5a3ef_3p_http,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75_3p_http

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: , , , , 

result, different session: , , , , 

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: , , , , 

result, different session: , , , , 

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: null

result, different session: null

unsupported: true

passed: undefined

test failed: false
Unsupported
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: , , 

result, different session: , , 

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Unsupported
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: , , , , 

result, different session: , , , , 

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      // Request a page that will send an HTTPOnly 'set-cookie' response header with secret value.
      await fetch(`${baseURI}cookie?secret=${secret}_http`);
    }

read: async () => {
      // Test if we now send a requests with a 'cookie' header containing the secret.
      let response = await fetch(`${baseURI}headers`);
      let cookie = (await response.json())["cookie"];
      return cookie ? cookie.match(/secret=([\w-]+)/)[1]: null;
    }

result, same session: , , , , 

result, different session: , , , , 

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
cookie (JS)
The cookie, first introduced by Netscape in 1994, is a small amount of data stored by your browser on a website's behalf. It has legitimate uses, but it is also the classic cross-site tracking mechanism, and today still the most popular method of tracking users across websites. Browsers can stop cookies from being used for cross-site tracking by either blocking or partitioning them.
              
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c_3p_js,
6e0d0559-9ac2-4abc-9025-582a29e22a8f_3p_js,
066762a4-4e2b-4344-82a6-b11540d563ab_3p_js,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee_3p_js,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2_3p_js

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: 
ead45dd4-f0d1-4ab5-af71-11d423c35903_3p_js,
139bd1c9-0bf8-414e-809a-d12f37b34aab_3p_js,
33d68d0c-5918-4ede-a519-aff107d2ce1c_3p_js,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5_3p_js,
fd597d68-f9be-48a0-9656-b7d0b2e140be_3p_js

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: , , , , 

result, different session: , , , , 

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: 
7d86a137-2f75-4c34-aade-b50776a7fd40_3p_js,
142f7096-d2d9-4c89-ba22-d0a3b8055308_3p_js,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c_3p_js,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd_3p_js,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346_3p_js

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: 
784004e0-9aeb-47e5-9c5f-12b883e353ac_3p_js,
eab30b13-9ecc-48af-9a1a-cae23815efbc_3p_js,
a652f390-6821-4125-9fd0-c118d2a1e7a2_3p_js,
09045837-46dd-45c8-ad94-7b7c171aeef0_3p_js,
7cb67be0-5127-4489-aeb1-7164e3b5af92_3p_js

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: 
79152685-f997-47b4-aef1-9e391daef62f_3p_js,
3863988a-95b2-4b54-a63c-ce0727dc2084_3p_js,
218debd1-a05b-48c0-b827-e4249b90263c_3p_js,
6d855fa7-5396-4b31-ac62-f63859f5a3ef_3p_js,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75_3p_js

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: , , , , 

result, different session: , , , , 

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: , , , , 

result, different session: , , , , 

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: null

result, different session: null

unsupported: true

passed: undefined

test failed: false
Unsupported
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: , , 

result, different session: , , 

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Unsupported
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: , , , , 

result, different session: , , , , 

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: (secret) => {
      document.cookie = `secret=${secret}_js; max-age=3600; SameSite=None; Secure`;
    }

read: () => document.cookie ? document.cookie.match(/secret=([\w-]+)/)[1] : null

result, same session: , , , , 

result, different session: , , , , 

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
CookieStore
The Cookie Store API is an alternative asynchronous API for managing cookies, supported by some browsers.
              
Passed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c_3p,
6e0d0559-9ac2-4abc-9025-582a29e22a8f_3p,
066762a4-4e2b-4344-82a6-b11540d563ab_3p,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee_3p,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2_3p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
ead45dd4-f0d1-4ab5-af71-11d423c35903_3p,
139bd1c9-0bf8-414e-809a-d12f37b34aab_3p,
33d68d0c-5918-4ede-a519-aff107d2ce1c_3p,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5_3p,
fd597d68-f9be-48a0-9656-b7d0b2e140be_3p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: , , , , 

result, different session: , , , , 

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
7d86a137-2f75-4c34-aade-b50776a7fd40_3p,
142f7096-d2d9-4c89-ba22-d0a3b8055308_3p,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c_3p,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd_3p,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346_3p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different session: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different session: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different session: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: , , , , 

result, different session: , , , , 

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: Error: Unsupported

result, different session: Error: Unsupported

unsupported: true

passed: undefined

test failed: false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

result, different session: 
Error: Unsupported,
Error: Unsupported,
Error: Unsupported

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: , , , , 

result, different session: , , , , 

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: (data) => {
      const msPerHour = 60 * 60 * 1000;
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      window.cookieStore.set({
        name: "partition_test",
        value: data,
        expires: Date.now() + msPerHour,
        sameSite: "none"
      });
    }

read: async () => {
      if (!window.cookieStore) {
        throw new Error("Unsupported");
      }
      const cookie = await window.cookieStore.get("partition_test");
      if (!cookie) {
        return null;
      }
      return cookie.value;
    }

result, same session: , , , , 

result, different session: , , , , 

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
CSS cache
CSS stylesheets are cached, and if that cache is shared between websites, it can be used to track users across sites.
              
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_6930649439685761,
fake_1644147482076117,
fake_9681088364591528,
fake_6931026820164943,
fake_08070763789937652

result, different session: 
fake_7712973420579898,
fake_5544662649528862,
fake_4374883458776446,
fake_9796102986607615,
fake_25367359150042224

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_6436083281982434,
fake_552034892527353,
fake_7875234768167523,
fake_22870977602364784,
fake_9683303543873862

result, different session: 
fake_19830494847109326,
fake_5456001821245442,
fake_6969507063169718,
fake_5508724062186672,
fake_24795584653861913

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_5552006602469317,
fake_13029231005512543,
fake_5803572726640933,
fake_9753242809602021,
fake_500215030783536

result, different session: 
fake_3818389921172036,
fake_6987015925027875,
fake_44239730405873234,
fake_009325197576034894,
fake_09114838993932639

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_704475416513842,
fake_032334099840567454,
fake_0026012940127988493,
fake_9873368291179148,
fake_35815576447478525

result, different session: 
fake_8084741536014928,
fake_02635438052618655,
fake_5946917540059997,
fake_585628310105351,
fake_9767380980920481

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_24870293329616588,
fake_8445048433088169,
fake_6753559497168116,
fake_4973801446156152,
fake_80816281964552

result, different session: 
fake_5981112692551835,
fake_031143861755742908,
fake_2203820967062764,
fake_15325024412238308,
fake_8066540824850774

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_5368907610195199,
fake_056892500808908064,
fake_25459551662833646,
fake_675492039774616,
fake_48175904091646204

result, different session: 
fake_4221198412903786,
fake_739692100899322,
fake_26012269209524463,
fake_2013681625442758,
fake_6808493759420595

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_9102177000784333,
fake_3487120030033297,
fake_01769780552454181,
fake_3058905632043081,
fake_8306789773387067

result, different session: 
fake_1756999684613325,
fake_31126290215304087,
fake_7850202416741467,
fake_22762657535836417,
fake_8113646028591399

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_1527225059975017,
fake_6028136745327568,
fake_4403150865753682,
fake_46453565863627877,
fake_9519245911495038

result, different session: 
fake_6616719122729553,
fake_5680220888649292,
fake_9068762849819347,
fake_23800184560340965,
fake_5847956049280039

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: fake_4706790223020445

result, different session: fake_6494132516526245

unsupported: false

passed: true

test failed: false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_4608854242674769,
fake_10917935182949456,
fake_11223182582334568

result, different session: 
fake_4404767760974764,
fake_6140113584600722,
fake_030052179030011894

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_5838445113214317,
fake_20435238299497915,
fake_9624957154532661,
fake_1378219404457106,
fake_9048579568475847

result, different session: 
fake_5129606921739998,
fake_10335270299322108,
fake_19397164634861097,
fake_06206155230412724,
fake_6174165719546765

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return key;
    }

read: async (key) => {
      const href = testURI("resource", "css", key);
      const head = document.getElementsByTagName("head")[0];
      head.innerHTML += `<link type="text/css" rel="stylesheet" href="${href}">`;
      const testElement = document.querySelector("#css");
      let fontFamily;
      while (true) {
        await sleepMs(100);
        fontFamily = getComputedStyle(testElement).fontFamily;
        if (fontFamily.startsWith("fake")) {
          break;
        }
      }
      console.log(fontFamily);
      return fontFamily;
    }

result, same session: 
fake_13820042010988587,
fake_2070041320849585,
fake_2979918935986263,
fake_08472755554814992,
fake_016529746638011655

result, different session: 
fake_7938695732370176,
fake_9025413955415189,
fake_7937750495627542,
fake_334867091007341,
fake_9192885652711715

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
favicon cache
A favicon is an icon that represents a website, typically shown in browser tab and bookmarks menu. If the favicon cache is not partitioned, it can be used to track users across websites.
              
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 2

result, different session: 3

unsupported: false

passed: true

test failed: false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1

result, different session: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => key

read: async (key) => {
      // Wait for the favicon to load (defined in supercookies.html)
      await sleepMs(2000);
      let response = await fetch(
        testURI("ctr", "favicon", key), {"cache": "reload"});
      let count = (await response.text()).trim();
      if (count === "0") {
        throw new Error("No requests received");
      }
      return count;
    }

result, same session: 2, 2, 2, 2, 2

result, different session: 4, 4, 4, 4, 4

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
fetch cache
When a resource is received via the Fetch API, it is frequently cached. That cache can potentially be abused for cross-site tracking.
              
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 2

result, different session: 3

unsupported: false

passed: true

test failed: false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1

result, different session: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      return key;
    }

read: async (key) => {
      let response = await fetch(testURI("resource", "fetch", key),
                                 {cache: "force-cache"});
      let countResponse = await fetch(testURI("ctr", "fetch", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
font cache
Web fonts are sometimes stored in their own cache, which is vulnerable to being abused for cross-site tracking.
              
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 2

result, different session: 3

unsupported: false

passed: true

test failed: false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1

result, different session: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } body { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      return key;
    }

read: async (key) => {
      const text = document.createElement("span");
      text.id = "text";
      text.innerText = "test";
      document.body.appendChild(text);
      const originalWidth = text.getBoundingClientRect().width;
      let style = document.createElement("style");
      style.type='text/css';
      let fontURI = testURI("resource", "font", key);
      style.innerHTML = `@font-face {font-family: "myFont"; src: url("${fontURI}"); } #text { font-family: "myFont" }`;
      document.getElementsByTagName("head")[0].appendChild(style);
      let newWidth;
      do {
        await sleepMs(100);
        newWidth = text.getBoundingClientRect().width;
      } while (newWidth < 0 || newWidth === originalWidth)
      let response = await fetch(
        testURI("ctr", "font", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
iframe cache
An iframe is an element in a web page than allows websites to embed a second web page. Caching of this web page could be abused for cross-site tracking.
              
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 2

result, different session: 3

unsupported: false

passed: true

test failed: false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1

result, different session: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      iframe.addEventListener("load", () => resolve(key), {once: true});
      iframe.src = testURI("resource", "page", key);
    })

read: async (key) => {
      let iframe = document.createElement("iframe");
      document.body.appendChild(iframe);
      let iframeLoadPromise = new Promise((resolve, reject) => {
        iframe.addEventListener("load", resolve, {once: true});
      });
      let address = testURI("resource", "page", key);
      iframe.src = address;
      await iframeLoadPromise;
      let response = await fetch(
        testURI("ctr", "page", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
image cache
Caching of images in web browsers is a standard behavior. But if that cache leaks between websites, it can be abused for cross-site tracking.
              
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 2

result, different session: 3

unsupported: false

passed: true

test failed: false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1

result, different session: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (key) => new Promise((resolve, reject) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      img.addEventListener("load", () => resolve(key), {once: true});
      img.src = testURI("resource", "image", key);
    })

read: async (key) => {
      let img = document.createElement("img");
      document.body.appendChild(img);
      let imgLoadPromise = new Promise((resolve, reject) => {
        img.addEventListener("load", resolve, {once: true});
      });
      img.src = testURI("resource", "image", key);
      await imgLoadPromise;
      let response = await fetch(
        testURI("ctr", "image", key), {"cache": "reload"});
      return (await response.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
indexedDB
The IndexedDB API exposes a transactional database to web pages. That database can be used to track users across websites, unless it is partitioned.
              
Unsupported
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.

result, different session: 
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.

result, different session: 
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.,
Error: The user denied permission to access the database.

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb_3p,
42c2adb9-619b-4b8f-a843-84ede2df77e1_3p,
8b52f9c3-d53f-4dad-91bb-5e162d085e46_3p,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f_3p,
57161958-02cf-461a-8b94-db8e32150e06_3p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
7d86a137-2f75-4c34-aade-b50776a7fd40_3p,
142f7096-d2d9-4c89-ba22-d0a3b8055308_3p,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c_3p,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd_3p,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346_3p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
784004e0-9aeb-47e5-9c5f-12b883e353ac_3p,
eab30b13-9ecc-48af-9a1a-cae23815efbc_3p,
a652f390-6821-4125-9fd0-c118d2a1e7a2_3p,
09045837-46dd-45c8-ad94-7b7c171aeef0_3p,
7cb67be0-5127-4489-aeb1-7164e3b5af92_3p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
79152685-f997-47b4-aef1-9e391daef62f_3p,
3863988a-95b2-4b54-a63c-ce0727dc2084_3p,
218debd1-a05b-48c0-b827-e4249b90263c_3p,
6d855fa7-5396-4b31-ac62-f63859f5a3ef_3p,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75_3p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

result, different session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
92ee92f4-0792-4f9a-ba56-45989cd80572_3p,
606b3a57-c3cf-45fe-a079-3b1566f6abd1_3p,
d40bde18-63e0-419d-b233-1dc09c4c5f94_3p,
df73d3b8-d807-46cf-9f77-65d30bc06fa4_3p,
ba872d52-1a40-47cf-9575-e78f927ff5b3_3p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: undefined

result, different session: undefined

unsupported: true

passed: undefined

test failed: false
Unsupported
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

result, different session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
3a254b81-9486-4572-a38a-b4bd82252b3e_3p,
c0ee6f27-d50e-4975-9a0a-523696e05cbf_3p,
89d605d6-80b6-488e-8454-c4de9c5dfe6b_3p,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6_3p,
d0752b25-d241-400f-a69f-21c3ff242a56_3p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (secret) => {
      try {
        return await IdbKeyVal.set("secret", secret);
      } catch (e) {
        throw new Error("Unsupported");
      }
    }

read: () => IdbKeyVal.get("secret")

result, same session: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078_3p,
fad257e3-bcc3-4aa7-b541-2b4c42aee488_3p,
3da92bf0-3ab6-4d33-9452-fac73e80ebba_3p,
d276eb0f-b849-45c6-97b9-57596abf3678_3p,
b64d49aa-de07-400a-9631-2935db968706_3p

result, different session: undefined

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
localStorage
The localStorage API gives websites access to a key-value database that will remain available across visits. If the localStorage API is not partitioned or blocked, it can also be used to track users across websites.
              
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
31218ce0-f3be-4668-9c98-1b15e3ad8b6c_3p,
6e0d0559-9ac2-4abc-9025-582a29e22a8f_3p,
066762a4-4e2b-4344-82a6-b11540d563ab_3p,
78743dea-6f8e-447a-b696-c3ce5c4cd9ee_3p,
9a7622ad-f032-49a0-b1bc-fc695ef8e8a2_3p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
ead45dd4-f0d1-4ab5-af71-11d423c35903_3p,
139bd1c9-0bf8-414e-809a-d12f37b34aab_3p,
33d68d0c-5918-4ede-a519-aff107d2ce1c_3p,
f4e0fc78-5add-4e91-9d10-0c5ddea287f5_3p,
fd597d68-f9be-48a0-9656-b7d0b2e140be_3p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
acd0081b-f37c-4bfc-a842-4b74df0c5efb_3p,
42c2adb9-619b-4b8f-a843-84ede2df77e1_3p,
8b52f9c3-d53f-4dad-91bb-5e162d085e46_3p,
d3f4e3d0-8cc1-4a45-aebb-941d62aa805f_3p,
57161958-02cf-461a-8b94-db8e32150e06_3p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
7d86a137-2f75-4c34-aade-b50776a7fd40_3p,
142f7096-d2d9-4c89-ba22-d0a3b8055308_3p,
02a358c3-f14e-4ab4-9828-1f93ac1a8c6c_3p,
ae71cdf0-7942-4d0b-ae0d-709aa7f8d7fd_3p,
2b01d48f-2808-4d2b-b5e0-fa0ef53e7346_3p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
784004e0-9aeb-47e5-9c5f-12b883e353ac_3p,
eab30b13-9ecc-48af-9a1a-cae23815efbc_3p,
a652f390-6821-4125-9fd0-c118d2a1e7a2_3p,
09045837-46dd-45c8-ad94-7b7c171aeef0_3p,
7cb67be0-5127-4489-aeb1-7164e3b5af92_3p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
79152685-f997-47b4-aef1-9e391daef62f_3p,
3863988a-95b2-4b54-a63c-ce0727dc2084_3p,
218debd1-a05b-48c0-b827-e4249b90263c_3p,
6d855fa7-5396-4b31-ac62-f63859f5a3ef_3p,
dfb73e15-8f36-4d94-b21e-ec65fc5d2f75_3p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

result, different session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
92ee92f4-0792-4f9a-ba56-45989cd80572_3p,
606b3a57-c3cf-45fe-a079-3b1566f6abd1_3p,
d40bde18-63e0-419d-b233-1dc09c4c5f94_3p,
df73d3b8-d807-46cf-9f77-65d30bc06fa4_3p,
ba872d52-1a40-47cf-9575-e78f927ff5b3_3p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: null

result, different session: null

unsupported: true

passed: undefined

test failed: false
Unsupported
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

result, different session: 
Error: The operation is insecure.,
Error: The operation is insecure.,
Error: The operation is insecure.

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
3a254b81-9486-4572-a38a-b4bd82252b3e_3p,
c0ee6f27-d50e-4975-9a0a-523696e05cbf_3p,
89d605d6-80b6-488e-8454-c4de9c5dfe6b_3p,
c190dea2-f7a7-4c44-aac0-2cfb817ddac6_3p,
d0752b25-d241-400f-a69f-21c3ff242a56_3p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: (secret) => localStorage.setItem("secret", secret)

read: () => localStorage.getItem("secret")

result, same session: 
c82dabdd-cc1d-4366-ab2f-6e65bea8b078_3p,
fad257e3-bcc3-4aa7-b541-2b4c42aee488_3p,
3da92bf0-3ab6-4d33-9452-fac73e80ebba_3p,
d276eb0f-b849-45c6-97b9-57596abf3678_3p,
b64d49aa-de07-400a-9631-2935db968706_3p

result, different session: , , , , 

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
prefetch cache
A <link rel='prefetch'...> suggests to browsers they should fetch a resource ahead of time and cache it. But if browsers don't partition this cache, it can be used to track users across websites.
              
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Failed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 1, 1, 1, 1, 1

unsupported: false, false, false, false, false

passed: false, false, false, false, false

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received

result, different session: 
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received

result, different session: 
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received,
Error: No requests received

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: Error: No requests received

result, different session: Error: No requests received

unsupported: true

passed: undefined

test failed: false
Unsupported
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 
Error: No requests received,
Error: No requests received,
Error: No requests received

result, different session: 
Error: No requests received,
Error: No requests received,
Error: No requests received

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      return key;
    }

read: async (key) => {
      let link = document.createElement("link");
      link.rel = "prefetch";
      link.href = testURI("resource", "prefetch", key);
      document.getElementsByTagName("head")[0].appendChild(link);
      await sleepMs(500);
      let response = await fetch(
        testURI("ctr", "prefetch", key), {"cache": "reload"});
      let countString = (await response.text()).trim();
      if (parseInt(countString) === 0) {
        throw new Error("No requests received");
      }
      return countString;
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Web SQL Database
The Web SQL Database is a deprecated web API for storing data in an SQL database.
              
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: Error: Web SQL is deprecated

result, different session: Error: Web SQL is deprecated

unsupported: true

passed: undefined

test failed: false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true

passed: undefined

test failed: false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
Unsupported
write: async (key) => {
      if (!window.openDatabase) {
        throw new Error("gported");
      }
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let tx = new Promise((resolve) => database.transaction(tx => {
        tx.executeSql(
          `CREATE TABLE IF NOT EXISTS cache(
             id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
             name TEXT NOT NULL,
             value TEXT NOT NULL,
             UNIQUE (name)
           )`,
          [], (tx, rs) => {}, (tx, err) => {});
        tx.executeSql(
          `INSERT OR REPLACE INTO cache(name, value)
           VALUES(?, ?)`,
          ["secret", key], (tx, rs) => {}, (tx, rs) => {});
      }));
    }

read: async () => {
      let database = window.openDatabase("sqlite_supercookie", "", "supercookie", 1024 * 1024);
      let result = await new Promise((resolve, reject) => database.transaction(tx => {
        tx.executeSql(
          "SELECT value FROM cache WHERE name=?",
          ["secret"],
          (tx, rs) => resolve(rs),
          (tx, err) => reject(err));
      }));
      return result.rows.item(0).value;
    }

result, same session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

result, different session: 
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function,
Error: window.openDatabase is not a function

unsupported: true, true, true, true, true

passed: undefined

test failed: false, false, false, false, false
XMLHttpRequest cache
Similar to the newer Fetch API, any resource received may be cached by the browser. The cache is potentially vulnerable to cross-site tracking attack.
              
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 2

result, different session: 3

unsupported: false

passed: true

test failed: false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1

result, different session: 2, 2, 2

unsupported: false, false, false

passed: true, true, true

test failed: false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false
Passed
write: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      return key;
    }

read: async (key) => {
      const req = new XMLHttpRequest();
      const loadPromise = new Promise(resolve => req.addEventListener("load", resolve));
      req.open("GET", testURI("resource", "xhr", key));
      req.send();
      await loadPromise;
      let countResponse = await fetch(testURI("ctr", "xhr", key),
                                      {cache: "reload"});
      return (await countResponse.text()).trim();
    }

result, same session: 1, 1, 1, 1, 1

result, different session: 2, 2, 2, 2, 2

unsupported: false, false, false, false, false

passed: true, true, true, true, true

test failed: false, false, false, false, false