/** * This file is part of Haketilo. * * Function: Content scripts - main script. * * Copyright (C) 2022 Wojtek Kosior * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * As additional permission under GNU GPL version 3 section 7, you * may distribute forms of that code without the copy of the GNU * GPL normally required by section 4, provided you include this * license notice and, in case of non-source distribution, a URL * through which recipients can access the Corresponding Source. * If you modify file(s) with this exception, you may extend this * exception to your version of the file(s), but you are not * obligated to do so. If you do not wish to do so, delete this * exception statement from your version. * * As a special exception to the GPL, any HTML file which merely * makes function calls to this code, and for that purpose * includes it by reference shall be deemed a separate work for * copyright law purposes. If you modify this code, you may extend * this exception to your version of the code, but you are not * obligated to do so. If you do not wish to do so, delete this * exception statement from your version. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <https://www.gnu.org/licenses/>. * * I, Wojtek Kosior, thereby promise not to sue for violation of this file's * license. Although I request that you do not make use of this code in a * proprietary program, I am not going to enforce this in court. */ #IMPORT content/repo_query_cacher.js #IMPORT content/haketilo_apis.js #FROM common/browser.js IMPORT browser #FROM common/misc.js IMPORT is_privileged_url #FROM common/policy.js IMPORT decide_policy, fallback_policy #FROM content/policy_enforcing.js IMPORT enforce_blocking #IF CHROMIUM && MV2 function synchronously_get_policy(url) { const encoded_url = encodeURIComponent(url); const request_url = `${browser.runtime.getURL("dummy")}?url=${encoded_url}`; try { var xhttp = new XMLHttpRequest(); xhttp.open("GET", request_url, false); xhttp.send(); } catch(e) { console.error("Haketilo: Failure to synchronously fetch policy for url.", e); return fallback_policy(); } try { const policy = /^[^?]*\?settings=(.*)$/.exec(xhttp.responseURL)[1]; return JSON.parse(decodeURIComponent(policy)); } catch(e) { console.error("Haketilo: Failure to process synchronously fetched policy for url.", e); return fallback_policy() } } #ENDIF let already_run = false, resolve_page_info, page_info_prom = new Promise(cb => resolve_page_info = cb); function on_page_info_request([type], sender, respond_cb) { if (type !== "page_info") return; page_info_prom.then(respond_cb); return true; } #IF MOZILLA || MV3 globalThis.haketilo_content_script_main = async function() { #ELSE async function main() { #ENDIF if (already_run) return; already_run = true; if (is_privileged_url(document.URL)) return; if (window.top === window) { browser.runtime.onMessage.addListener(on_page_info_request); repo_query_cacher.start(); } #IF MOZILLA || MV3 try { var policy = decide_policy(globalThis.haketilo_pattern_tree, document.URL, globalThis.haketilo_default_allow, globalThis.haketilo_secret); } catch(e) { console.error("Haketilo:", e); var policy = fallback_policy(); } #ELSE const policy = synchronously_get_policy(document.URL); #ENDIF const page_info = Object.assign({url: document.URL}, policy); ["csp", "nonce"].forEach(prop => delete page_info[prop]); if ("payload" in policy) { const msg = ["indexeddb_files", policy.payload.identifier]; var scripts_prom = browser.runtime.sendMessage(msg); } await enforce_blocking(policy); if ("payload" in policy) { const script_response = await scripts_prom; if ("error" in script_response) { resolve_page_info(Object.assign(page_info, script_response)); return; } else { haketilo_apis.start(); const version = browser.runtime.getManifest().version; const scripts = [ `window.haketilo_version = ${JSON.stringify(version)};`, ...script_response.files ]; for (const script_contents of scripts) { const html_ns = "http://www.w3.org/1999/xhtml"; const script = document.createElementNS(html_ns, "script"); const load_prom = new Promise( (...cbs) => [script.onload, script.onerror] = cbs ); const blobby_opts = {type: "text/javascript;charset=UTF-8"}; const blobby = new Blob([script_contents], blobby_opts); script.src = URL.createObjectURL(blobby); script.setAttribute("nonce", policy.nonce); document.documentElement.append(script); await load_prom; script.remove(); } } } resolve_page_info(page_info); } #IF MOZILLA || MV3 function main() { if (globalThis.haketilo_pattern_tree !== undefined) globalThis.haketilo_content_script_main(); } #ENDIF #IF !UNIT_TEST main(); #ENDIF