Project

General

Profile

Download (4.83 KB) Statistics
| Branch: | Tag: | Revision:

haketilo / content / content.js @ cf838016

1
/**
2
 * This file is part of Haketilo.
3
 *
4
 * Function: Content scripts - main script.
5
 *
6
 * Copyright (C) 2022 Wojtek Kosior
7
 *
8
 * This program is free software: you can redistribute it and/or modify
9
 * it under the terms of the GNU General Public License as published by
10
 * the Free Software Foundation, either version 3 of the License, or
11
 * (at your option) any later version.
12
 *
13
 * This program is distributed in the hope that it will be useful,
14
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16
 * GNU General Public License for more details.
17
 *
18
 * As additional permission under GNU GPL version 3 section 7, you
19
 * may distribute forms of that code without the copy of the GNU
20
 * GPL normally required by section 4, provided you include this
21
 * license notice and, in case of non-source distribution, a URL
22
 * through which recipients can access the Corresponding Source.
23
 * If you modify file(s) with this exception, you may extend this
24
 * exception to your version of the file(s), but you are not
25
 * obligated to do so. If you do not wish to do so, delete this
26
 * exception statement from your version.
27
 *
28
 * As a special exception to the GPL, any HTML file which merely
29
 * makes function calls to this code, and for that purpose
30
 * includes it by reference shall be deemed a separate work for
31
 * copyright law purposes. If you modify this code, you may extend
32
 * this exception to your version of the code, but you are not
33
 * obligated to do so. If you do not wish to do so, delete this
34
 * exception statement from your version.
35
 *
36
 * You should have received a copy of the GNU General Public License
37
 * along with this program.  If not, see <https://www.gnu.org/licenses/>.
38
 *
39
 * I, Wojtek Kosior, thereby promise not to sue for violation of this file's
40
 * license. Although I request that you do not make use of this code in a
41
 * proprietary program, I am not going to enforce this in court.
42
 */
43

    
44
#IMPORT content/repo_query_cacher.js
45

    
46
#FROM common/browser.js           IMPORT browser
47
#FROM common/misc.js              IMPORT is_privileged_url
48
#FROM common/policy.js            IMPORT decide_policy, fallback_policy
49
#FROM content/policy_enforcing.js IMPORT enforce_blocking
50

    
51
#IF CHROMIUM && MV2
52
function synchronously_get_policy(url)
53
{
54
    const encoded_url = encodeURIComponent(url);
55
    const request_url = `${browser.runtime.getURL("dummy")}?url=${encoded_url}`;
56

    
57
    try {
58
	var xhttp = new XMLHttpRequest();
59
	xhttp.open("GET", request_url, false);
60
	xhttp.send();
61
    } catch(e) {
62
	console.error("Failure to synchronously fetch policy for url.", e);
63
	return fallback_policy();
64
    }
65

    
66
    try {
67
	const policy = /^[^?]*\?settings=(.*)$/.exec(xhttp.responseURL)[1];
68
	return JSON.parse(decodeURIComponent(policy));
69
    } catch(e) {
70
	console.error("Failure to process synchronously fetched policy for url.", e);
71
	return fallback_policy()
72
    }
73
}
74
#ENDIF
75

    
76
let already_run = false, resolve_page_info,
77
    page_info_prom = new Promise(cb => resolve_page_info = cb);
78

    
79
function on_page_info_request([type], sender, respond_cb) {
80
    if (type !== "page_info")
81
	return;
82

    
83
    page_info_prom.then(respond_cb);
84

    
85
    return true;
86
}
87

    
88
#IF MOZILLA || MV3
89
globalThis.haketilo_content_script_main = async function() {
90
#ELSE
91
async function main() {
92
#ENDIF
93
    if (already_run)
94
        return;
95

    
96
    already_run = true;
97

    
98
    if (is_privileged_url(document.URL))
99
	return;
100

    
101
    browser.runtime.onMessage.addListener(on_page_info_request);
102
    repo_query_cacher.start();
103

    
104
#IF MOZILLA || MV3
105
    try {
106
	var policy = decide_policy(globalThis.haketilo_pattern_tree,
107
				   document.URL,
108
				   globalThis.haketilo_default_allow,
109
				   globalThis.haketilo_secret);
110
    } catch(e) {
111
	console.error(e);
112
	var policy = fallback_policy();
113
    }
114
#ELSE
115
    const policy = synchronously_get_policy(document.URL);
116
#ENDIF
117

    
118
    const page_info = Object.assign({url: document.URL}, policy);
119
    ["csp", "nonce"].forEach(prop => delete page_info[prop]);
120

    
121
    if ("payload" in policy) {
122
	const msg = ["indexeddb_files", policy.payload.identifier];
123
	var scripts_prom = browser.runtime.sendMessage(msg);
124
    }
125

    
126
    await enforce_blocking(policy);
127

    
128
    if ("payload" in policy) {
129
	const script_response = await scripts_prom;
130

    
131
	if ("error" in script_response) {
132
	    resolve_page_info(Object.assign(page_info, script_response));
133
	    return;
134
	} else {
135
	    for (const script_contents of script_response.files) {
136
		const html_ns = "http://www.w3.org/1999/xhtml";
137
		const script = document.createElementNS(html_ns, "script");
138

    
139
		script.innerText = script_contents;
140
		script.setAttribute("nonce", policy.nonce);
141
		document.documentElement.append(script);
142
		script.remove();
143
	    }
144
	}
145
    }
146

    
147
    resolve_page_info(page_info);
148
}
149

    
150
#IF MOZILLA || MV3
151
function main() {
152
    if (globalThis.haketilo_pattern_tree !== undefined)
153
	globalThis.haketilo_content_script_main();
154
}
155
#ENDIF
156

    
157
#IF !UNIT_TEST
158
main();
159
#ENDIF
(1-1/3)