summaryrefslogtreecommitdiff
path: root/src/bg/RequestUtil.js
diff options
context:
space:
mode:
authorhackademix2018-07-02 01:50:32 +0200
committerhackademix2018-07-02 01:50:32 +0200
commit81bd93a72d0b49fd78908f7c467ae9f324791e23 (patch)
treedaacfeb158f1f6512532d7e4ccafc6da68494aa4 /src/bg/RequestUtil.js
parent673c881559e4f9274be5b54ca35552e12cd8cd76 (diff)
downloadnoscript-81bd93a72d0b49fd78908f7c467ae9f324791e23.tar.gz
noscript-81bd93a72d0b49fd78908f7c467ae9f324791e23.tar.xz
noscript-81bd93a72d0b49fd78908f7c467ae9f324791e23.zip
Simplified and apparently more reliable+flexible+efficient dynamic script injection method.
Diffstat (limited to 'src/bg/RequestUtil.js')
-rw-r--r--src/bg/RequestUtil.js161
1 files changed, 46 insertions, 115 deletions
diff --git a/src/bg/RequestUtil.js b/src/bg/RequestUtil.js
index 1ebbbaa..2020286 100644
--- a/src/bg/RequestUtil.js
+++ b/src/bg/RequestUtil.js
@@ -1,130 +1,61 @@
'use strict';
{
- let runningScripts = new Map();
-
+ let pendingRequests = new Map();
+ let cleanup = r => {
+ pendingRequests.delete(r.requestId);
+ };
+ let filter = {
+ urls: ["<all_urls>"],
+ types: ["main_frame", "sub_frame", "object"]
+ };
+ browser.webRequest.onCompleted.addListener(cleanup, filter);
+ browser.webRequest.onErrorOccurred.addListener(cleanup, filter);
var RequestUtil = {
async executeOnStart(request, details) {
let {requestId, tabId, frameId} = request;
- details = Object.assign({
- runAt: "document_start",
- frameId,
- }, details);
- browser.tabs.executeScript(tabId, details);
- return;
- let filter = browser.webRequest.filterResponseData(requestId);
- filter.onstart = event => {
- browser.tabs.executeScript(tabId, details);
- debug("Execute on start", details);
- filter.write(new Uint8Array());
- };
- filter.ondata = event => {
- filter.write(event.data);
- filter.disconnect();
-
- }
- },
- async executeOnStartCS(request, details) {
- let {url, requestId, tabId, frameId} = request;
-
- let urlObj = new URL(url);
- if (urlObj.hash || urlObj.port || urlObj.username) {
- urlObj.hash = urlObj.port = urlObj.username = "";
- url = urlObj.toString();
+ let scripts = pendingRequests.get(requestId);
+ let scriptKey = JSON.stringify(details);
+ if (!scripts) {
+ pendingRequests.set(requestId, scripts = new Map());
+ scripts.set(scriptKey, details);
+ } else {
+ scripts.set(scriptKey, details);
+ return;
}
- let wr = browser.webRequest;
- let filter = {
- urls: [`${urlObj.origin}/*`],
- types: ["main_frame", "sub_frame", "object"]
- };
- let finalize;
- let cleanup = r => {
- if (cleanup && r.requestId === requestId) {
- wr.onCompleted.removeListener(cleanup);
- wr.onErrorOccurred.removeListener(cleanup);
- cleanup = null;
- if (finalize) {
- finalize();
- }
- }
- };
- wr.onCompleted.addListener(cleanup, filter);
- wr.onErrorOccurred.addListener(cleanup, filter);
-
- details = Object.assign({
- runAt: "document_start",
- frameId,
- }, details);
-
- if (browser.contentScripts) {
- let js = [{}];
- if (details.file) js[0].file = details.file;
- else if (details.code) js[0].code = details.code;
- let settings = {
- "runAt": details.runAt,
- js,
- matches: [url],
- allFrames: frameId !== 0,
- }
- // let's try to avoid duplicates
- let key = JSON.stringify(settings);
- if (runningScripts.has(key)) {
- let scriptRef = runningScripts.get(key);
- scriptRef.count++;
- return;
- }
- if (settings.allFrames) {
- // let's check whether the same script is registered for top frames:
- // if it is, let's unregister it first to avoid duplicates
- settings.allFrames = false;
- let topKey = JSON.stringify(settings);
- settings.allFrames = true;
- if (runningScripts.has(topKey)) {
- let topScript = runningScripts.get(topKey);
- try {
- topScript.unregister();
- } catch (e) {
- error(e);
- } finally {
- runningScripts.delete(topKey);
- }
- }
- }
- let script = await browser.contentScripts.register(settings);
- debug("Content script %o registered.", settings);
- finalize = () => {
- debug("Finalizing content script %o...", settings);
+ let filter = browser.webRequest.filterResponseData(requestId);
+ let buffer = [];
+ filter.onstart = async event => {
+ filter.write(new Uint8Array());
+ for (let details of scripts.values()) {
+ details = Object.assign({
+ runAt: "document_start",
+ frameId,
+ }, details);
try {
- script.unregister();
- runningScripts.delete(key);
- debug("Content script %o unregistered!", settings);
- } finally {
- finalize = null;
+ await browser.tabs.executeScript(tabId, details);
+ debug("Execute on start OK", request.url, details);
+ } catch (e) {
+ error(e, "Execute on start failed", request.url, details);
}
}
- runningScripts.set(key, script);
- if (!cleanup) { // the request has already been interrupted
- finalize();
+ if (buffer.length) {
+ debug("Flushing %s buffer chunks", buffer.length);
+ for (let chunk of buffer) {
+ filter.write(chunk);
+ }
+ filter.disconnect();
+ buffer = null;
}
-
- return;
- }
-
- function listener(r) {
- if (r.requestId === requestId) {
- browser.tabs.executeScript(tabId, details);
- finalize();
- finalize = null;
+ };
+ filter.ondata = event => {
+ if (buffer) {
+ buffer.push(event.data);
+ return;
}
+ filter.write(event.data);
+ filter.disconnect();
}
- finalize = () => {
- wr.onResponseStarted.removeListener(listener);
- }
- wr.onResponseStarted.addListener(listener, filter);
- debug("Executing %o", details);
-
- },
-
-
+ }
}
}