CanvasBlocker/lib/randomSupplies.js

277 lines
7.2 KiB
JavaScript

/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
(function(){
"use strict";
let scope;
if ((typeof exports) !== "undefined"){
scope = exports;
}
else {
scope = require.register("./randomSupplies", {});
}
const rngTemplate = {
getBitRng: function(length, window){
const rng = this.getRng(Math.ceil(length / 32), window);
let rnd = 0;
let mask = 0xffffffff * 2;
return function(value, i){
if (mask > 0xffffffff){
mask = 1;
rnd = rng(i / 32);
}
let bit = 1 * (!!(rnd & mask));
mask *= 2;
return bit;
};
},
getIndexRng: function(length, maxIndex, window){
const rng = this.getRng(length, window);
return function(i){
return Math.floor(rng(i) / 0xffffffff * maxIndex);
};
},
getValueRng: function(length, window){
const rng = this.getBitRng(length, window);
return function(value, i){
const rnd = rng(value, i);
// XOR the last bit to alter it... or not
return value ^ (rnd & 0x01);
};
},
getPixelRng: function(length, window, ignoredColors){
const rng = this.getValueRng(length, window);
// eslint-disable-next-line max-params
return function(r, g, b, a, i){
const index = String.fromCharCode(r, g, b, a);
if (ignoredColors[index]){
return [r, g, b, a];
}
const baseIndex = i * 4;
return [
rng(r, baseIndex + 0),
rng(g, baseIndex + 1),
rng(b, baseIndex + 2),
rng(a, baseIndex + 3)
];
};
}
};
const settings = require("./settings");
const extension = require("./extension");
function getDomain(window){
if (settings.sharePersistentRndBetweenDomains){
return "shared://domain";
}
if (!window.location.href || window.location.href === "about:blank"){
if (window !== window.parent && window.parent){
return getDomain(window.parent);
}
else if (window.opener){
return getDomain(window.opener);
}
}
return window.location.host;
}
let persistentRnd = Object.create(null);
let cookieStoreId = false;
function getCookieStoreId(){
while (cookieStoreId === false){
extension.waitSync("to wait for cookie store id");
}
return cookieStoreId;
}
settings.onloaded(function(){
try {
let storedData = JSON.parse(
extension.inIncognitoContext?
settings.persistentIncognitoRndStorage:
settings.persistentRndStorage
);
for (let domain in storedData){
const value = storedData[domain];
if (
Array.isArray(value) &&
value.length === 128 &&
value.every(function(value){
return typeof value === "number" && value >= 0 && value < 256;
})
){
persistentRnd[domain] = new Uint8Array(value);
}
}
}
catch (error){
// JSON is not valid -> ignore it
}
});
const getPersistentRnd = (function(){
extension.message.on(function(data){
if (data["canvasBlocker-set-domain-rnd"]){
const {domain, incognito, rnd} = data["canvasBlocker-set-domain-rnd"];
if (incognito === extension.inIncognitoContext){
persistentRnd[domain] = new Uint8Array(rnd);
}
}
if (data["canvasBlocker-clear-domain-rnd"]){
const domain = getCookieStoreId() + getDomain(window);
let ownPersistendRnd = data["canvasBlocker-clear-domain-rnd"] !== "force"? persistentRnd[domain]: false;
persistentRnd = Object.create(null);
if (ownPersistendRnd){
persistentRnd[domain] = ownPersistendRnd;
extension.message.send({
"canvasBlocker-new-domain-rnd": {
domain,
incognito: extension.inIncognitoContext,
rnd: Array.from(persistentRnd[domain])
}
});
}
}
});
return function getPersistentRnd(window){
const domain = getCookieStoreId() + getDomain(window);
if (!persistentRnd[domain]){
// create the (sub-)domains random numbers if not existing
persistentRnd[domain] = new Uint8Array(128);
window.crypto.getRandomValues(persistentRnd[domain]);
extension.message.send({
"canvasBlocker-new-domain-rnd": {
domain,
incognito: extension.inIncognitoContext,
rnd: Array.from(persistentRnd[domain])
}
});
}
return persistentRnd[domain];
};
}());
scope.persistent = Object.create(rngTemplate);
scope.persistent.name = "persistent";
scope.persistent.setDomainRnd = function(domain, rnd){
persistentRnd[domain] = new Uint8Array(rnd);
};
scope.persistent.setCookieStoreId = function(newCookieStoreId){
if (typeof newCookieStoreId === "string"){
cookieStoreId = (
newCookieStoreId !== "" &&
newCookieStoreId !== "firefox-default"
)? newCookieStoreId + "@": "";
}
};
scope.persistent.getRng = function(length, window){
const bitSet = new Uint32Array(getPersistentRnd(window).buffer);
const bitSetLength = bitSet.length;
return function(i){
return bitSet[i % bitSetLength];
};
};
scope.persistent.getBitRng = function(length, window){
const bitSet = getPersistentRnd(window);
return function(value, i){
// use the last 7 bits from the value for the index of the
// random number
const index = value & 0x7F;
// use the last 3 bits from the position and the first bit from
// from the value to get bit to use from the random number
const bitIndex = ((i & 0x03) << 1) | (value >>> 7);
// extract the bit
const bit = (bitSet[index] >>> bitIndex) & 0x01;
return bit;
};
};
scope.constant = Object.create(rngTemplate);
scope.constant.name = "constant";
scope.constant.getRng = function(length, window){
return scope.nonPersistent.getRng(length, window);
};
scope.constant.getPixelRng = (function(){
const colors = Object.create(null);
return function getConstantPixelRng(length, window, ignoredColors){
const rng = scope.nonPersistent.getValueRng(1024, window);
// eslint-disable-next-line max-params
return function(r, g, b, a, i){
const index = String.fromCharCode(r, g, b, a);
if (ignoredColors[index]){
return [r, g, b, a];
}
let color = colors[index];
if (!color){
color = [
rng(r, 0),
rng(g, 0),
rng(b, 0),
rng(a, 0)
];
colors[index] = color;
}
return color;
};
};
}());
scope.nonPersistent = Object.create(rngTemplate);
scope.nonPersistent.name = "nonPersistent";
scope.nonPersistent.getRng = function(length, window){
const maxLength = 0x4000;
let randomI = maxLength;
let randomNumbers = new Uint32Array(Math.min(maxLength, length));
return function(i){
if (randomI >= randomNumbers.length){
// refill the random number bucket if empty
randomI = 0;
if (length - i < maxLength){
randomNumbers = new Uint32Array(Math.max(1, length - i));
}
window.crypto.getRandomValues(randomNumbers);
}
const rnd = randomNumbers[randomI];
randomI += 1;
return rnd;
};
};
scope.white = {
name: "white",
getRng: function(){
return function(){
return 255;
};
},
getBitRng: function(){
return function(){
return 1;
};
},
getIndexRng: function(){
return function(){
return 0;
};
},
getValueRng: function(){
return this.getRng();
},
getPixelRng: function(){
return function(){
return [255, 255, 255, 255];
};
}
};
}());