mirror of
https://github.com/kkapsner/CanvasBlocker
synced 2025-01-05 03:21:52 +01:00
227 lines
6.7 KiB
JavaScript
227 lines
6.7 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(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 modifiedNavigatorAPI = require("./modifiedNavigatorAPI");
|
|
modifiedNavigatorAPI.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 {
|
|
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;
|
|
}
|
|
const canvasBlockerData = wrappedWindow.matchMedia(extensionSecret[0]);
|
|
if (canvasBlockerData.secret === extensionSecret[1]){
|
|
if (wrappedWindow.top === wrappedWindow){
|
|
canvasBlockerData.undoIntercept(extension.extensionID);
|
|
}
|
|
else {
|
|
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;
|
|
extension.changeProperty(window, "matchMedia", {
|
|
object: wrappedWindow,
|
|
name: "matchMedia",
|
|
type: "value",
|
|
changed: extension.exportFunctionWithName(function matchMedia(query){
|
|
if (query === extensionSecret[0]){
|
|
return {
|
|
secret: extensionSecret[1],
|
|
undoIntercept: function(token){
|
|
if (token === extension.extensionID){
|
|
extension.revertProperties(window);
|
|
}
|
|
}
|
|
};
|
|
}
|
|
else {
|
|
return arguments.length > 1?
|
|
originalMatchMedia.call(this, ...arguments):
|
|
originalMatchMedia.call(this, query);
|
|
}
|
|
}, window, originalMatchMedia.name)
|
|
});
|
|
|
|
interceptedWindows.set(wrappedWindow, true);
|
|
return true;
|
|
}
|
|
|
|
logging.message("register listener for messages from background script");
|
|
extension.message.on(function(data){
|
|
if (data["canvasBlocker-unload"]){
|
|
extension.revertProperties();
|
|
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)); |