mirror of
https://github.com/kkapsner/CanvasBlocker
synced 2025-01-05 11:32:20 +01:00
312 lines
9.4 KiB
JavaScript
312 lines
9.4 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("./iframeProtection", {});
|
|
}
|
|
|
|
const settings = require("./settings");
|
|
const lists = require("./lists");
|
|
|
|
function isWhitelisted(url){
|
|
return lists.get("white").match(url) || settings.get("blockMode", url).startsWith("allow");
|
|
}
|
|
|
|
function createChangeProperty(window){
|
|
function changeProperty(object, name, type, changed){
|
|
const descriptor = Object.getOwnPropertyDescriptor(object, name);
|
|
const original = descriptor[type];
|
|
descriptor[type] = changed;
|
|
Object.defineProperty(object, name, descriptor);
|
|
registerChangedProperty(object, name, descriptor, type, original);
|
|
}
|
|
const changedProperties = [];
|
|
// eslint-disable-next-line max-params
|
|
function registerChangedProperty(object, name, descriptor, type, original){
|
|
changedProperties.push({object, name, descriptor, type, original});
|
|
}
|
|
if (settings.isStillDefault){
|
|
settings.onloaded(function(){
|
|
if (isWhitelisted(window.location)){
|
|
changedProperties.forEach(function({object, name, descriptor, type, original}){
|
|
descriptor[type] = original;
|
|
Object.defineProperty(object, name, descriptor);
|
|
});
|
|
changedProperties.length = 0;
|
|
}
|
|
});
|
|
}
|
|
else {
|
|
if (isWhitelisted(window.location)){
|
|
return false;
|
|
}
|
|
}
|
|
return changeProperty;
|
|
}
|
|
|
|
function protectFrameProperties({window, wrappedWindow, changeProperty, singleCallback}){
|
|
["HTMLIFrameElement", "HTMLFrameElement"].forEach(function(constructorName){
|
|
const constructor = window[constructorName];
|
|
const wrappedConstructor = wrappedWindow[constructorName];
|
|
|
|
const contentWindowDescriptor = Object.getOwnPropertyDescriptor(
|
|
constructor.prototype,
|
|
"contentWindow"
|
|
);
|
|
const originalContentWindowGetter = contentWindowDescriptor.get;
|
|
const contentWindowTemp = {
|
|
get contentWindow(){
|
|
const window = originalContentWindowGetter.call(this);
|
|
if (window){
|
|
singleCallback(window);
|
|
}
|
|
return window;
|
|
}
|
|
};
|
|
changeProperty(wrappedConstructor.prototype, "contentWindow", "get", exportFunction(
|
|
Object.getOwnPropertyDescriptor(contentWindowTemp, "contentWindow").get,
|
|
window
|
|
));
|
|
|
|
const contentDocumentDescriptor = Object.getOwnPropertyDescriptor(
|
|
constructor.prototype,
|
|
"contentDocument"
|
|
);
|
|
const originalContentDocumentGetter = contentDocumentDescriptor.get;
|
|
const contentDocumentTemp = {
|
|
get contentDocument(){
|
|
const document = originalContentDocumentGetter.call(this);
|
|
if (document){
|
|
singleCallback(document.defaultView);
|
|
}
|
|
return document;
|
|
}
|
|
};
|
|
changeProperty(wrappedConstructor.prototype, "contentDocument", "get", exportFunction(
|
|
Object.getOwnPropertyDescriptor(contentDocumentTemp, "contentDocument").get,
|
|
window
|
|
));
|
|
});
|
|
}
|
|
|
|
function protectDOMModifications({window, wrappedWindow, changeProperty, allCallback}){
|
|
[
|
|
// useless as length could be obtained before the iframe is created and window.frames === window
|
|
// {
|
|
// object: wrappedWindow,
|
|
// methods: [],
|
|
// getters: ["length", "frames"],
|
|
// setters: []
|
|
// },
|
|
{
|
|
object: wrappedWindow.Node.prototype,
|
|
methods: ["appendChild", "insertBefore", "replaceChild"],
|
|
getters: [],
|
|
setters: []
|
|
},
|
|
{
|
|
object: wrappedWindow.Element.prototype,
|
|
methods: [
|
|
"append", "prepend",
|
|
"insertAdjacentElement", "insertAdjacentHTML", "insertAdjacentText",
|
|
"replaceWith"
|
|
],
|
|
getters: [],
|
|
setters: [
|
|
"innerHTML",
|
|
"outerHTML"
|
|
]
|
|
}
|
|
].forEach(function(protectionDefinition){
|
|
const object = protectionDefinition.object;
|
|
protectionDefinition.methods.forEach(function(method){
|
|
const descriptor = Object.getOwnPropertyDescriptor(object, method);
|
|
const original = descriptor.value;
|
|
changeProperty(object, method, "value", exportFunction(eval(`(function ${method}(){
|
|
const value = arguments.length?
|
|
original.apply(this, window.Array.from(arguments)):
|
|
original.call(this);
|
|
allCallback();
|
|
return value;
|
|
})`), window));
|
|
});
|
|
protectionDefinition.getters.forEach(function(property){
|
|
const temp = eval(`({
|
|
get ${property}(){
|
|
const ret = this.${property};
|
|
allCallback();
|
|
return ret;
|
|
}
|
|
})`);
|
|
changeProperty(object, property, "get", exportFunction(
|
|
Object.getOwnPropertyDescriptor(temp, property).get,
|
|
window
|
|
));
|
|
});
|
|
protectionDefinition.setters.forEach(function(property){
|
|
const descriptor = Object.getOwnPropertyDescriptor(object, property);
|
|
const setter = descriptor.set;
|
|
const temp = eval(`({
|
|
set ${property}(value){
|
|
const ret = setter.call(this, value);
|
|
// const ret = this.${property} = value;
|
|
allCallback();
|
|
return ret;
|
|
}
|
|
})`);
|
|
changeProperty(object, property, "set", exportFunction(
|
|
Object.getOwnPropertyDescriptor(temp, property).set, window
|
|
));
|
|
});
|
|
});
|
|
}
|
|
|
|
function enableMutationObserver({window, allCallback}){
|
|
const observer = new MutationObserver(allCallback);
|
|
let observing = false;
|
|
function observe(){
|
|
if (
|
|
!observing &&
|
|
window.document
|
|
){
|
|
observer.observe(window.document, {subtree: true, childList: true});
|
|
observing = true;
|
|
}
|
|
}
|
|
observe();
|
|
window.document.addEventListener("DOMContentLoaded", function(){
|
|
if (observing){
|
|
observer.disconnect();
|
|
observing = false;
|
|
}
|
|
});
|
|
return observe;
|
|
}
|
|
|
|
function protectDocumentWrite({window, wrappedWindow, changeProperty, observe, allCallback}){
|
|
const documentWriteDescriptorOnHTMLDocument = Object.getOwnPropertyDescriptor(
|
|
wrappedWindow.HTMLDocument.prototype,
|
|
"write"
|
|
);
|
|
const documentWriteDescriptor = documentWriteDescriptorOnHTMLDocument || Object.getOwnPropertyDescriptor(
|
|
wrappedWindow.Document.prototype,
|
|
"write"
|
|
);
|
|
const documentWrite = documentWriteDescriptor.value;
|
|
changeProperty(
|
|
documentWriteDescriptorOnHTMLDocument?
|
|
wrappedWindow.HTMLDocument.prototype:
|
|
wrappedWindow.Document.prototype,
|
|
// eslint-disable-next-line no-unused-vars
|
|
"write", "value", exportFunction(function write(markup){
|
|
for (let i = 0, l = arguments.length; i < l; i += 1){
|
|
const str = "" + arguments[i];
|
|
// weird problem with waterfox and google docs
|
|
const parts = (
|
|
str.match(/^\s*<!doctype/i) &&
|
|
!str.match(/frame/i)
|
|
)? [str]: str.split(/(?=<)/);
|
|
const length = parts.length;
|
|
const scripts = window.document.getElementsByTagName("script");
|
|
for (let i = 0; i < length; i += 1){
|
|
documentWrite.call(this, parts[i]);
|
|
allCallback();
|
|
if (scripts.length && scripts[scripts.length - 1].src){
|
|
observe();
|
|
}
|
|
}
|
|
}
|
|
}, window)
|
|
);
|
|
|
|
const documentWritelnDescriptorOnHTMLDocument = Object.getOwnPropertyDescriptor(
|
|
wrappedWindow.HTMLDocument.prototype,
|
|
"writeln"
|
|
);
|
|
const documentWritelnDescriptor = documentWritelnDescriptorOnHTMLDocument || Object.getOwnPropertyDescriptor(
|
|
wrappedWindow.Document.prototype,
|
|
"writeln"
|
|
);
|
|
const documentWriteln = documentWritelnDescriptor.value;
|
|
changeProperty(
|
|
documentWritelnDescriptorOnHTMLDocument?
|
|
wrappedWindow.HTMLDocument.prototype:
|
|
wrappedWindow.Document.prototype,
|
|
"writeln", "value", exportFunction(
|
|
// eslint-disable-next-line no-unused-vars
|
|
function writeln(markup){
|
|
for (let i = 0, l = arguments.length; i < l; i += 1){
|
|
const str = "" + arguments[i];
|
|
const parts = str.split(/(?=<)/);
|
|
const length = parts.length;
|
|
const scripts = window.document.getElementsByTagName("script");
|
|
for (let i = 0; i < length; i += 1){
|
|
documentWrite.call(this, parts[i]);
|
|
allCallback();
|
|
if (scripts.length && scripts[scripts.length - 1].src){
|
|
observe();
|
|
}
|
|
}
|
|
}
|
|
documentWriteln.call(this, "");
|
|
},
|
|
window
|
|
)
|
|
);
|
|
}
|
|
|
|
function protectWindowOpen({window, wrappedWindow, changeProperty, singleCallback}){
|
|
const windowOpenDescriptor = Object.getOwnPropertyDescriptor(
|
|
wrappedWindow,
|
|
"open"
|
|
);
|
|
const windowOpen = windowOpenDescriptor.value;
|
|
const getDocument = Object.getOwnPropertyDescriptor(
|
|
window,
|
|
"document"
|
|
).get;
|
|
changeProperty(
|
|
wrappedWindow,
|
|
"open", "value", exportFunction(function open(){
|
|
const newWindow = arguments.length?
|
|
windowOpen.apply(this, window.Array.from(arguments)):
|
|
windowOpen.call(this);
|
|
if (newWindow){
|
|
// if we use windowOpen from the normal window we see some SOP errors
|
|
// BUT we need the unwrapped window...
|
|
singleCallback(getDocument.call(newWindow).defaultView);
|
|
}
|
|
return newWindow;
|
|
}, window)
|
|
);
|
|
}
|
|
|
|
scope.protect = function protect(window, wrappedWindow, singleCallback, allCallback){
|
|
const changeProperty = createChangeProperty(window);
|
|
|
|
if (!changeProperty){
|
|
return;
|
|
}
|
|
|
|
const api = {window, wrappedWindow, changeProperty, singleCallback, allCallback};
|
|
|
|
protectFrameProperties(api);
|
|
|
|
protectDOMModifications(api);
|
|
|
|
// MutationObserver to intercept iFrames while generating the DOM.
|
|
api.observe = enableMutationObserver(api);
|
|
|
|
// MutationObserver does not trigger fast enough when document.write is used
|
|
protectDocumentWrite(api);
|
|
|
|
protectWindowOpen(api);
|
|
};
|
|
}()); |