Source code

Revision control

Copy as Markdown

Other Tools

/* -*- indent-tabs-mode: nil; js-indent-level: 2 -*- */
/* 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 https://mozilla.org/MPL/2.0/. */
const lazy = {};
ChromeUtils.defineLazyGetter(lazy, "console", () => {
return console.createInstance({
prefix: "UserCharacteristicsPage",
maxLogLevelPref: "toolkit.telemetry.user_characteristics_ping.logLevel",
});
});
/* This actor is responsible for rendering the canvas elements defined in
* recipes. It renders with both hardware and software rendering.
* It also provides debug information about the canvas rendering
* capabilities of its window (not all windows get HW rendering).
*
* See the recipes object for the list of canvases to render.
* WebGL is still being rendered in toolkit/components/resistfingerprinting/content/usercharacteristics.js
*
*/
export class UserCharacteristicsCanvasRenderingChild extends JSWindowActorChild {
constructor() {
super();
this.destroyed = false;
}
async render(hwRenderingExpected) {
// I couldn't think of a good name. Recipes as in instructions to render.
const runRecipe = async (isAccelerated, recipe) => {
const canvas = this.document.createElement("canvas");
canvas.width = recipe.size[0];
canvas.height = recipe.size[1];
const ctx = canvas.getContext("2d", {
forceSoftwareRendering: !isAccelerated,
});
if (!ctx) {
lazy.console.error("Could not get 2d context");
return { error: "COULD_NOT_GET_CONTEXT" };
}
let debugInfo = null;
try {
debugInfo = ctx.getDebugInfo(true /* ensureTarget */);
} catch (e) {
lazy.console.error(
"Error getting canvas debug info during render: ",
await stringifyError(e)
);
return {
error: "COULD_NOT_GET_DEBUG_INFO",
originalError: await stringifyError(e),
};
}
if (debugInfo.isAccelerated !== isAccelerated) {
lazy.console.error(
`Canvas is not rendered with expected mode. Expected: ${isAccelerated}, got: ${debugInfo.isAccelerated}`
);
return { error: "WRONG_RENDERING_MODE" };
}
try {
await recipe.func(this.contentWindow, canvas, ctx);
} catch (e) {
lazy.console.error("Error rendering canvas: ", await stringifyError(e));
return {
error: "RENDERING_ERROR",
originalError: await stringifyError(e),
};
}
return sha1(canvas.toDataURL("image/png", 1)).catch(stringifyError);
};
const errors = [];
const renderings = new Map();
// Run HW renderings
// Attempt HW rendering regardless of the expected rendering mode.
for (const [name, recipe] of Object.entries(recipes)) {
lazy.console.debug("[HW] Rendering ", name);
const result = await runRecipe(true, recipe);
if (result.error) {
if (!hwRenderingExpected && result.error === "WRONG_RENDERING_MODE") {
// If the rendering mode is wrong, we can ignore the error.
lazy.console.debug(
"Ignoring error because HW rendering is not expected: ",
result.error
);
continue;
}
errors.push({
name,
error: result.error,
originalError: result.originalError,
});
continue;
}
renderings.set(name, result);
}
// Run SW renderings
for (const [name, recipe] of Object.entries(recipes)) {
lazy.console.debug("[SW] Rendering ", name);
const result = await runRecipe(false, recipe);
if (result.error) {
errors.push({
name: name + "software",
error: result.error,
originalError: result.originalError,
});
continue;
}
renderings.set(name + "software", result);
}
const data = new Map();
data.set("renderings", renderings);
data.set("errors", errors);
return data;
}
async getDebugInfo() {
const canvas = this.document.createElement("canvas");
const ctx = canvas.getContext("2d");
if (!ctx) {
return null;
}
try {
return ctx.getDebugInfo(true /* ensureTarget */);
} catch (e) {
lazy.console.error(
"Error getting canvas debug info: ",
await stringifyError(e)
);
return null;
}
}
sendMessage(name, obj, transferables) {
if (this.destroyed) {
return;
}
this.sendAsyncMessage(name, obj, transferables);
}
didDestroy() {
this.destroyed = true;
}
async receiveMessage(msg) {
lazy.console.debug("Actor Child: Got ", msg.name);
switch (msg.name) {
case "CanvasRendering:GetDebugInfo":
return this.getDebugInfo();
case "CanvasRendering:Render":
return this.render(msg.data.hwRenderingExpected);
}
return null;
}
}
const recipes = {
// Metric name => (optionally async) function to render
canvasdata1: {
func: (window, canvas, ctx) => {
ctx.fillStyle = "orange";
ctx.fillRect(100, 100, 50, 50);
},
size: [250, 250],
},
canvasdata2: {
func: (window, canvas, ctx) => {
ctx.fillStyle = "blue";
ctx.beginPath();
ctx.moveTo(50, 50);
ctx.lineTo(200, 200);
ctx.lineTo(175, 100);
ctx.closePath();
ctx.fill();
ctx.strokeStyle = "red";
ctx.lineWidth = 5;
ctx.stroke();
},
size: [250, 250],
},
canvasdata3: {
func: (window, canvas, ctx) => {
const kImageBlob =
"";
return new Promise((resolve, reject) => {
// Calling new Image() fails for some reason, that's why we use new window.Image()
// It is also the only reason we pass window as an argument to all the functions :)
const image = new window.Image();
image.src = kImageBlob;
image.onload = () => {
ctx.drawImage(image, 0, 0, canvas.width, canvas.height);
resolve();
};
image.onerror = e => {
reject(e);
};
});
},
size: [250, 250],
},
canvasdata4: {
func: (window, canvas, ctx) => {
ctx.fillStyle = "orange";
ctx.globalAlpha = 0.5;
ctx.translate(100, 100);
ctx.rotate((45.0 * Math.PI) / 180.0);
ctx.fillRect(0, 0, 50, 50);
ctx.rotate((-15.0 * Math.PI) / 180.0);
ctx.fillRect(0, 0, 50, 50);
},
size: [250, 250],
},
canvasdata5: {
func: (window, canvas, ctx) => {
ctx.fillStyle = "green";
ctx.font = "italic 30px Georgia";
ctx.fillText("The quick brown", 15, 100);
ctx.fillText("fox jumps over", 15, 150);
ctx.fillText("the lazy dog", 15, 200);
},
size: [250, 250],
},
canvasdata6: {
func: (window, canvas, ctx) => {
ctx.fillStyle = "green";
ctx.translate(10, 100);
ctx.rotate((45.0 * Math.PI) / 180.0);
ctx.shadowColor = "blue";
ctx.shadowBlur = 50;
ctx.font = "italic 40px Georgia";
ctx.fillText("The quick", 0, 0);
},
size: [250, 250],
},
canvasdata7: {
func: (window, canvas, ctx) => {
ctx.fillStyle = "green";
ctx.font = "italic 30px system-ui";
ctx.fillText("The quick brown", 15, 100);
ctx.fillText("fox jumps over", 15, 150);
ctx.fillText("the lazy dog", 15, 200);
},
size: [250, 250],
},
canvasdata8: {
func: (window, canvas, ctx) => {
ctx.fillStyle = "green";
ctx.translate(10, 100);
ctx.rotate((45.0 * Math.PI) / 180.0);
ctx.shadowColor = "blue";
ctx.shadowBlur = 50;
ctx.font = "italic 40px system-ui";
ctx.fillText("The quick", 0, 0);
},
size: [250, 250],
},
canvasdata9: {
func: (window, canvas, ctx) => {
ctx.fillStyle = "green";
ctx.font = "italic 30px LocalFiraSans";
ctx.fillText("The quick brown", 15, 100);
ctx.fillText("fox jumps over", 15, 150);
ctx.fillText("the lazy dog", 15, 200);
},
size: [250, 250],
},
canvasdata10: {
func: (window, canvas, ctx) => {
ctx.fillStyle = "green";
ctx.translate(10, 100);
ctx.rotate((45.0 * Math.PI) / 180.0);
ctx.shadowColor = "blue";
ctx.shadowBlur = 50;
ctx.font = "italic 40px LocalFiraSans";
ctx.fillText("The quick", 0, 0);
},
size: [250, 250],
},
// fingerprintjs
// Their fingerprinting code went to the BSL license from MIT in
// So use the version of the code in the parent commit which is still MIT
canvasdata12Fingerprintjs1: {
func: (window, canvas, ctx) => {
ctx.textBaseline = "alphabetic";
ctx.fillStyle = "#f60";
ctx.fillRect(100, 1, 62, 20);
ctx.fillStyle = "#069";
// It's important to use explicit built-in fonts in order to exclude the affect of font preferences
// (there is a separate entropy source for them).
ctx.font = '11pt "Times New Roman"';
// The choice of emojis has a gigantic impact on rendering performance (especially in FF).
// Some newer emojis cause it to slow down 50-200 times.
// There must be no text to the right of the emoji, see https://github.com/fingerprintjs/fingerprintjs/issues/574
// A bare emoji shouldn't be used because the canvas will change depending on the script encoding:
// Escape sequence shouldn't be used too because Terser will turn it into a bare unicode.
const printedText = `Cwm fjordbank gly ${
String.fromCharCode(55357, 56835) /* 😃 */
}`;
ctx.fillText(printedText, 2, 15);
ctx.fillStyle = "rgba(102, 204, 0, 0.2)";
ctx.font = "18pt Arial";
ctx.fillText(printedText, 4, 45);
},
// usercharacteristics.html uses 240x60, but we can't get HW acceleration
// if an axis is less than 128px
size: [240, 128],
},
canvasdata13Fingerprintjs2: {
func: (window, canvas, ctx) => {
// Canvas blending
ctx.globalCompositeOperation = "multiply";
for (const [color, x, y] of [
["#f2f", 40, 40],
["#2ff", 80, 40],
["#ff2", 60, 80],
]) {
ctx.fillStyle = color;
ctx.beginPath();
ctx.arc(x, y, 40, 0, Math.PI * 2, true);
ctx.closePath();
ctx.fill();
}
// Canvas winding
ctx.fillStyle = "#f9c";
ctx.arc(60, 60, 60, 0, Math.PI * 2, true);
ctx.arc(60, 60, 20, 0, Math.PI * 2, true);
ctx.fill("evenodd");
},
// usercharacteristics.html uses 122x110, but we can't get HW acceleration
// if an axis is less than 128px
size: [128, 128],
},
};
async function sha1(message) {
const msgUint8 = new TextEncoder().encode(message);
const hashBuffer = await crypto.subtle.digest("SHA-1", msgUint8);
const hashArray = Array.from(new Uint8Array(hashBuffer));
const hashHex = hashArray.map(b => b.toString(16).padStart(2, "0")).join("");
return hashHex;
}
async function stringifyError(error) {
if (error instanceof Error) {
const stack = error.stack ?? "";
return `${error.toString()} ${stack}`;
}
// A hacky attempt to extract as much as info from error
const errStr = await (async () => {
const asStr = await (async () => error.toString())().catch(() => "");
const asJson = await (async () => JSON.stringify(error))().catch(() => "");
return asStr.length > asJson.len ? asStr : asJson;
})();
return errStr;
}