Project

General

Profile

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

haketilo / content / content.js @ cc5f5828

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
#IMPORT content/haketilo_apis.js
46

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

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

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

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

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

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

    
84
    page_info_prom.then(respond_cb);
85

    
86
    return true;
87
}
88

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

    
97
    already_run = true;
98

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

    
102
    if (window.top === window) {
103
        browser.runtime.onMessage.addListener(on_page_info_request);
104
        repo_query_cacher.start();
105
    }
106

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

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

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

    
129
    await enforce_blocking(policy);
130

    
131
    if ("payload" in policy) {
132
	const script_response = await scripts_prom;
133

    
134
	if ("error" in script_response) {
135
	    resolve_page_info(Object.assign(page_info, script_response));
136
	    return;
137
	} else {
138
	    haketilo_apis.start();
139

    
140
	    const version = browser.runtime.getManifest().version;
141
	    const scripts = [
142
		`window.haketilo_version = ${JSON.stringify(version)};`,
143
		...script_response.files
144
	    ];
145
	    for (const script_contents of scripts) {
146
		const html_ns = "http://www.w3.org/1999/xhtml";
147
		const script = document.createElementNS(html_ns, "script");
148
		const load_prom = new Promise(
149
		    (...cbs) => [script.onload, script.onerror] = cbs
150
		);
151

    
152
		const blobby_opts = {type: "text/javascript;charset=UTF-8"};
153
		const blobby = new Blob([script_contents], blobby_opts);
154
		script.src = URL.createObjectURL(blobby);
155
		script.setAttribute("nonce", policy.nonce);
156

    
157
		document.documentElement.append(script);
158
		await load_prom;
159
		script.remove();
160
	    }
161
	}
162
    }
163

    
164
    resolve_page_info(page_info);
165
}
166

    
167
#IF MOZILLA || MV3
168
function main() {
169
    if (globalThis.haketilo_pattern_tree !== undefined)
170
	globalThis.haketilo_content_script_main();
171
}
172
#ENDIF
173

    
174
#IF !UNIT_TEST
175
main();
176
#ENDIF
(1-1/4)