/* 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(require){ "use strict"; const settings = require("./settings"); const {preIntercept: intercept} = require("./intercept"); const {ask} = require("./askForPermission"); const {sha256String: hashing} = require("./hash"); const {check: originalCheck, checkStack: originalCheckStack} = require("./check"); const extension = require("./extension"); const iframeProtection = require("./iframeProtection"); const logging = require("./logging"); logging.setPrefix("frame script"); // Variable to "unload" the script let enabled = true; logging.message("starting", location.href); function check(message){ if (enabled){ return originalCheck(message); } else { return {type: [], mode: "allow"}; } } function checkStack(stack){ if (enabled){ return originalCheckStack(stack); } else { return true; } } function askWrapper(data){ return ask(data, { _: extension.getTranslation, prefs }); } let extensionSecret; function computeExtensionSecret(){ function hashString(string){ return hashing(new Uint16Array( string.split("").map(function(c){ return c.charCodeAt(0); }) )); } const now = new Date(); const lastTenMinutes = Math.floor(now.getMinutes() / 10) * 10; const nextRun = new Date( now.getFullYear(), now.getMonth(), now.getDate(), now.getHours(), lastTenMinutes + 10, 0, 0 ); window.setTimeout( computeExtensionSecret, nextRun .getTime() - now.getTime() ); let string = extension.extensionID + `${now.getFullYear()}-${now.getMonth() + 1}-${now.getDate()} ${now.getHours()}:${lastTenMinutes}`; extensionSecret = [hashString("input" + string), hashString(string + "output")]; } computeExtensionSecret(); logging.message("open port to background script"); const port = browser.runtime.connect(); if (window === window.top){ logging.message("Is top level window -> tab had navigation -> clear page action"); port.postMessage({"canvasBlocker-clear-page-action": true}); } let tabId; port.onMessage.addListener(function(data){ logging.message("Got data from port", data); if (data.hasOwnProperty("tabId")){ logging.notice("my tab id is", data.tabId); tabId = data.tabId; } if (data.hasOwnProperty("cookieStoreId")){ logging.notice("my tab cookie store id is", data.cookieStoreId); const {persistent: persistentRnd} = require("./randomSupplies"); persistentRnd.setCookieStoreId(data.cookieStoreId); } const persistentRndName = "persistent" + (extension.inIncognitoContext? "Incognito": "") + "Rnd"; if (data.hasOwnProperty(persistentRndName)){ const persistentRndValue = data[persistentRndName]; logging.notice("got persistent random data", persistentRndValue); const {persistent: persistentRnd} = require("./randomSupplies"); Object.keys(persistentRndValue).forEach(function(domain){ logging.verbose("random data for", domain, persistentRndValue[domain]); persistentRnd.setDomainRnd(domain, persistentRndValue[domain]); }); } }); const notifications = []; const notificationCounter = {}; const sentAPIs = {}; function notify(data){ if (!settings.ignoredAPIs[data.api]){ if (settings.storeNotificationData){ notifications.push(data); } if (!notificationCounter[data.messageId]){ notificationCounter[data.messageId] = { count: 0, api: data.api }; } notificationCounter[data.messageId].count += 1; if (!sentAPIs[data.api]){ sentAPIs[data.api] = true; port.postMessage({"canvasBlocker-notify": data}); } } } function prefs(...args){ return settings.get(...args); } const interceptedWindows = new WeakMap(); function interceptWindow(window){ let wrappedTry; try { // eslint-disable-next-line no-unused-vars const href = window.location.href; wrappedTry = extension.getWrapped(window); } catch (error){ // we are unable to read the location due to SOP // therefore we also can not intercept anything. logging.notice("NOT intercepting window due to SOP", window); return false; } const wrappedWindow = wrappedTry; if (!enabled || interceptedWindows.get(wrappedWindow)){ return false; } if (wrappedWindow.matchMedia(extensionSecret[0]) === extensionSecret[1]){ interceptedWindows.set(wrappedWindow, true); return false; } logging.message("intercepting window", window); intercept( {subject: window}, {check, checkStack, ask: askWrapper, notify, prefs} ); logging.message("prepare to intercept (i)frames."); function interceptAllFrames(){ const currentLength = window.length; for (let i = currentLength; i--;){ if (!interceptedWindows.get(wrappedWindow[i])){ interceptWindow(window[i]); } } } iframeProtection.protect(window, wrappedWindow, interceptWindow, interceptAllFrames); const matchMediaDescriptor = Object.getOwnPropertyDescriptor(wrappedWindow, "matchMedia"); const originalMatchMedia = matchMediaDescriptor.value; matchMediaDescriptor.value = extension.exportFunctionWithName(function matchMedia(query){ if (query === extensionSecret[0]){ return extensionSecret[1]; } else { return arguments.length > 1? originalMatchMedia.apply(this, wrappedWindow.Array.from(arguments)): originalMatchMedia.call(this, query); } }, window, originalMatchMedia.name); Object.defineProperty(wrappedWindow, "matchMedia", matchMediaDescriptor); interceptedWindows.set(wrappedWindow, true); return true; } logging.message("register listener for messages from background script"); extension.message.on(function(data){ if (data["canvasBlocker-unload"]){ enabled = false; } if ( data.hasOwnProperty("canvasBlocker-sendNotifications") && data["canvasBlocker-sendNotifications"] === tabId ){ logging.notice("sending notifications:", notifications); extension.message.send({ sender: tabId, url: window.location.href, "canvasBlocker-notificationCounter": notificationCounter, "canvasBlocker-notifications": notifications }); logging.notice("notifications sent"); } }); interceptWindow(window); }(require));