123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253 |
- /*******************************************************************************
- ηMatrix - a browser extension to black/white list requests.
- Copyright (C) 2014-2019 The uMatrix/uBlock Origin authors
- Copyright (C) 2019-2022 Alessio Vanni
- 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.
- You should have received a copy of the GNU General Public License
- along with this program. If not, see {http://www.gnu.org/licenses/}.
- Home: https://gitlab.com/vannilla/ematrix
- uMatrix Home: https://github.com/gorhill/uMatrix
- */
- 'use strict';
- const {classes: Cc, interfaces: Ci, utils: Cu} = Components;
- // Accessing the context of the background page:
- // var win = Services.appShell.hiddenDOMWindow.document.querySelector('iframe[src*=ematrix]').contentWindow;
- let windowlessBrowser = null;
- let windowlessBrowserPL = null;
- let bgProcess = null;
- let version;
- const restartListener = {
- get messageManager() {
- return Cc['@mozilla.org/parentprocessmessagemanager;1']
- .getService(Ci.nsIMessageListenerManager);
- },
- receiveMessage: function() {
- shutdown();
- startup();
- }
- };
- // https://github.com/gorhill/uBlock/issues/2493
- // Fix by https://github.com/gijsk
- // imported from https://github.com/gorhill/uBlock/pull/2497
- function startup(data, reason) {
- if (data !== undefined) {
- version = data.version;
- }
- // Already started?
- if (bgProcess !== null) {
- return;
- }
- waitForHiddenWindow();
- }
- function createBgProcess(parentDocument) {
- bgProcess = parentDocument
- .documentElement
- .appendChild(parentDocument
- .createElementNS('http://www.w3.org/1999/xhtml',
- 'iframe'));
- bgProcess.setAttribute('src',
- 'chrome://ematrix/content/background.html#'
- + version);
- // https://developer.mozilla.org/en-US/docs/Mozilla/Tech/XPCOM/Reference/Interface/nsIMessageListenerManager#addMessageListener%28%29
- // "If the same listener registers twice for the same message, the
- // "second registration is ignored."
- restartListener
- .messageManager
- .addMessageListener('ematrix-restart', restartListener);
- }
- function getWindowlessBrowserFrame(appShell) {
- windowlessBrowser = appShell.createWindowlessBrowser(true);
- windowlessBrowser.QueryInterface(Ci.nsIInterfaceRequestor);
- let webProgress = windowlessBrowser.getInterface(Ci.nsIWebProgress);
- Cu.import('resource://gre/modules/XPCOMUtils.jsm');
- windowlessBrowserPL = {
- QueryInterface: XPCOMUtils.generateQI([
- Ci.nsIWebProgressListener,
- Ci.nsIWebProgressListener2,
- Ci.nsISupportsWeakReference
- ]),
- onStateChange: function(wbp, request, stateFlags, status) {
- if (!request) {
- return;
- }
- if (stateFlags & Ci.nsIWebProgressListener.STATE_STOP) {
- webProgress.removeProgressListener(windowlessBrowserPL);
- windowlessBrowserPL = null;
- createBgProcess(windowlessBrowser.document);
- }
- }
- };
- webProgress.addProgressListener(windowlessBrowserPL,
- Ci.nsIWebProgress.NOTIFY_STATE_DOCUMENT);
- windowlessBrowser.document.location =
- "data:application/vnd.mozilla.xul+xml;charset=utf-8,<window%20id='ematrix-win'/>";
- }
- function waitForHiddenWindow() {
- let appShell = Cc['@mozilla.org/appshell/appShellService;1']
- .getService(Ci.nsIAppShellService);
- let isReady = function() {
- var hiddenDoc;
- try {
- hiddenDoc = appShell.hiddenDOMWindow &&
- appShell.hiddenDOMWindow.document;
- } catch (ex) {
- }
- // Do not test against `loading`: it does appear `readyState` could be
- // undefined if looked up too early.
- if (!hiddenDoc || hiddenDoc.readyState !== 'complete') {
- return false;
- }
- // In theory, it should be possible to create a windowless browser
- // immediately, without waiting for the hidden window to have loaded
- // completely. However, in practice, on Windows this seems to lead
- // to a broken Firefox appearance. To avoid this, we only create the
- // windowless browser here. We'll use that rather than the hidden
- // window for the actual background page (windowless browsers are
- // also what the webextension implementation in Firefox uses for
- // background pages).
- getWindowlessBrowserFrame(appShell);
- return true;
- };
- if (isReady()) {
- return;
- }
- // https://github.com/gorhill/uBlock/issues/749
- // Poll until the proper environment is set up -- or give up eventually.
- // We poll frequently early on but relax poll delay as time pass.
- let tryDelay = 5;
- let trySum = 0;
- // https://trac.torproject.org/projects/tor/ticket/19438
- // Try for a longer period.
- // ηMatrix: I doubt this applies to us...
- let tryMax = 600011;
- let timer = Cc['@mozilla.org/timer;1']
- .createInstance(Ci.nsITimer);
- let checkLater = function() {
- trySum += tryDelay;
- if (trySum >= tryMax) {
- timer = null;
- return;
- }
- timer.init(timerObserver, tryDelay, timer.TYPE_ONE_SHOT);
- tryDelay *= 2;
- if (tryDelay > 503) {
- tryDelay = 503;
- }
- };
- var timerObserver = {
- observe: function() {
- timer.cancel();
- if (isReady()) {
- timer = null;
- } else {
- checkLater();
- }
- }
- };
- checkLater();
- }
- function shutdown(data, reason) {
- if (reason === APP_SHUTDOWN) {
- return;
- }
- if (bgProcess !== null) {
- bgProcess.parentNode.removeChild(bgProcess);
- bgProcess = null;
- }
- if (windowlessBrowser !== null) {
- // close() does not exist for older versions of Firefox.
- // ηMatrix: how old? But keeping it doesn't really hurt that much.
- if (typeof windowlessBrowser.close === 'function') {
- windowlessBrowser.close();
- }
- windowlessBrowser = null;
- windowlessBrowserPL = null;
- }
- if (data === undefined) {
- return;
- }
- // Remove the restartObserver only when the extension is being disabled
- restartListener
- .messageManager
- .removeMessageListener('ematrix-restart', restartListener);
- }
- function install(data, reason) {
- // https://bugzil.la/719376
- Cc['@mozilla.org/intl/stringbundle;1']
- .getService(Ci.nsIStringBundleService)
- .flushBundles();
- if (reason === ADDON_UPGRADE) {
- let Services =
- Cu.import('resource://gre/modules/Services.jsm', null).Services
- Services.obs.notifyObservers(null, 'chrome-flush-caches', null);
- Services.obs.notifyObservers(null, 'message-manager-flush-caches', null);
- }
- }
- function uninstall(data, aReason) {
- if (aReason !== ADDON_UNINSTALL) {
- return;
- }
- // To cleanup vAPI.localStorage in vapi-common.js, aka
- // "extensions.ematrix.*" in `about:config`.
- Cu.import('resource://gre/modules/Services.jsm', null)
- .Services.prefs.getBranch('extensions.ematrix.')
- .deleteBranch('');
- }
|