Project

General

Profile

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

haketilo / content / content.js @ f650ee99

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
    browser.runtime.onMessage.addListener(on_page_info_request);
103
    repo_query_cacher.start();
104

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

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

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

    
127
    await enforce_blocking(policy);
128

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

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

    
138
	    const version = browser.runtime.getManifest().version;
139
	    const scripts = [
140
		`window.haketilo_version = ${JSON.stringify(version)};`,
141
		...script_response.files
142
	    ];
143
	    for (const script_contents of scripts) {
144
		const html_ns = "http://www.w3.org/1999/xhtml";
145
		const script = document.createElementNS(html_ns, "script");
146

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

    
157
    resolve_page_info(page_info);
158
}
159

    
160
#IF MOZILLA || MV3
161
function main() {
162
    if (globalThis.haketilo_pattern_tree !== undefined)
163
	globalThis.haketilo_content_script_main();
164
}
165
#ENDIF
166

    
167
#IF !UNIT_TEST
168
main();
169
#ENDIF
(1-1/4)