/******************************************************************************* uBlock Origin - a browser extension to block requests. Copyright (C) 2014-2018 Raymond Hill 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://github.com/gorhill/uBlock */ 'use strict'; /******************************************************************************/ // Load all: executed once. µBlock.restart = (function() { /******************************************************************************/ var µb = µBlock; /******************************************************************************/ vAPI.app.onShutdown = function() { µb.staticFilteringReverseLookup.shutdown(); µb.assets.updateStop(); µb.staticNetFilteringEngine.reset(); µb.staticExtFilteringEngine.reset(); µb.sessionFirewall.reset(); µb.permanentFirewall.reset(); µb.sessionURLFiltering.reset(); µb.permanentURLFiltering.reset(); µb.hnSwitches.reset(); }; /******************************************************************************/ var processCallbackQueue = function(queue, callback) { var processOne = function() { var fn = queue.pop(); if ( fn ) { fn(processOne); } else if ( typeof callback === 'function' ) { callback(); } }; processOne(); }; /******************************************************************************/ // Final initialization steps after all needed assets are in memory. // - Initialize internal state with maybe already existing tabs. // - Schedule next update operation. var onAllReady = function() { // https://github.com/chrisaljoudi/uBlock/issues/184 // Check for updates not too far in the future. µb.assets.addObserver(µb.assetObserver.bind(µb)); µb.scheduleAssetUpdater(µb.userSettings.autoUpdate ? 7 * 60 * 1000 : 0); // vAPI.cloud is optional. if ( µb.cloudStorageSupported ) { vAPI.cloud.start([ 'tpFiltersPane', 'myFiltersPane', 'myRulesPane', 'whitelistPane' ]); } µb.contextMenu.update(null); µb.firstInstall = false; processCallbackQueue(µb.onStartCompletedQueue); }; /******************************************************************************/ // Filtering engines dependencies: // - PSL var onPSLReady = function() { µb.selfieManager.load(function(valid) { if ( valid === true ) { return onAllReady(); } µb.loadFilterLists(onAllReady); }); }; /******************************************************************************/ // To bring older versions up to date var onVersionReady = function(lastVersion) { if ( lastVersion === vAPI.app.version ) { return; } // Since AMO does not allow updating resources.txt, force a reload when a // new version is detected, as resources.txt may have changed since last // release. This will be done only for release versions of Firefox. if ( vAPI.webextFlavor.soup.has('firefox') && /(b|rc)\d+$/.test(vAPI.app.version) === false ) { µb.redirectEngine.invalidateResourcesSelfie(); } // From 1.15.19b9 and above, the `behind-the-scene` scope is no longer // whitelisted by default, and network requests from that scope will be // subject to filtering by default. // // Following code is to remove the `behind-the-scene` scope when updating // from a version older than 1.15.19b9. // This will apply only to webext versions of uBO, as the following would // certainly cause too much breakage in Firefox legacy given that uBO can // see ALL network requests. // Remove when everybody is beyond 1.15.19b8. (function patch1015019008(s) { if ( vAPI.firefox !== undefined ) { return; } var match = /^(\d+)\.(\d+)\.(\d+)(?:\D+(\d+))?/.exec(s); if ( match === null ) { return; } var v = parseInt(match[1], 10) * 1000 * 1000 * 1000 + parseInt(match[2], 10) * 1000 * 1000 + parseInt(match[3], 10) * 1000 + (match[4] ? parseInt(match[4], 10) : 0); if ( /rc\d+$/.test(s) ) { v += 100; } if ( v > 1015019008 ) { return; } if ( µb.getNetFilteringSwitch('http://behind-the-scene/') ) { return; } var fwRules = [ 'behind-the-scene * * noop', 'behind-the-scene * image noop', 'behind-the-scene * 3p noop', 'behind-the-scene * inline-script noop', 'behind-the-scene * 1p-script noop', 'behind-the-scene * 3p-script noop', 'behind-the-scene * 3p-frame noop' ].join('\n'); µb.sessionFirewall.fromString(fwRules, true); µb.permanentFirewall.fromString(fwRules, true); µb.savePermanentFirewallRules(); µb.hnSwitches.fromString([ 'no-large-media: behind-the-scene false' ].join('\n'), true); µb.saveHostnameSwitches(); µb.toggleNetFilteringSwitch('http://behind-the-scene/', '', true); })(lastVersion); vAPI.storage.set({ version: vAPI.app.version }); }; /******************************************************************************/ // https://github.com/chrisaljoudi/uBlock/issues/226 // Whitelist in memory. // Whitelist parser needs PSL to be ready. // gorhill 2014-12-15: not anymore var onNetWhitelistReady = function(netWhitelistRaw) { µb.netWhitelist = µb.whitelistFromString(netWhitelistRaw); µb.netWhitelistModifyTime = Date.now(); }; /******************************************************************************/ // User settings are in memory var onUserSettingsReady = function(fetched) { var userSettings = µb.userSettings; fromFetch(userSettings, fetched); if ( µb.privacySettingsSupported ) { vAPI.browserSettings.set({ 'hyperlinkAuditing': !userSettings.hyperlinkAuditingDisabled, 'prefetching': !userSettings.prefetchingDisabled, 'webrtcIPAddress': !userSettings.webrtcIPAddressHidden }); } µb.permanentFirewall.fromString(fetched.dynamicFilteringString); µb.sessionFirewall.assign(µb.permanentFirewall); µb.permanentURLFiltering.fromString(fetched.urlFilteringString); µb.sessionURLFiltering.assign(µb.permanentURLFiltering); µb.hnSwitches.fromString(fetched.hostnameSwitchesString); // https://github.com/gorhill/uBlock/issues/1892 // For first installation on a battery-powered device, disable generic // cosmetic filtering. if ( µb.firstInstall && vAPI.webextFlavor.soup.has('mobile') ) { userSettings.ignoreGenericCosmeticFilters = true; } }; /******************************************************************************/ // Housekeeping, as per system setting changes var onSystemSettingsReady = function(fetched) { var mustSaveSystemSettings = false; if ( fetched.compiledMagic !== µb.systemSettings.compiledMagic ) { µb.assets.remove(/^compiled\//); mustSaveSystemSettings = true; } if ( fetched.selfieMagic !== µb.systemSettings.selfieMagic ) { mustSaveSystemSettings = true; } if ( mustSaveSystemSettings ) { fetched.selfie = null; µb.selfieManager.destroy(); vAPI.storage.set(µb.systemSettings); } }; /******************************************************************************/ var onFirstFetchReady = function(fetched) { // https://github.com/gorhill/uBlock/issues/747 µb.firstInstall = fetched.version === '0.0.0.0'; // Order is important -- do not change: onSystemSettingsReady(fetched); fromFetch(µb.localSettings, fetched); onUserSettingsReady(fetched); fromFetch(µb.restoreBackupSettings, fetched); onNetWhitelistReady(fetched.netWhitelist); onVersionReady(fetched.version); µb.loadPublicSuffixList(onPSLReady); µb.loadRedirectResources(); }; /******************************************************************************/ var toFetch = function(from, fetched) { for ( var k in from ) { if ( from.hasOwnProperty(k) === false ) { continue; } fetched[k] = from[k]; } }; var fromFetch = function(to, fetched) { for ( var k in to ) { if ( to.hasOwnProperty(k) === false ) { continue; } if ( fetched.hasOwnProperty(k) === false ) { continue; } to[k] = fetched[k]; } }; /******************************************************************************/ var onSelectedFilterListsLoaded = function() { var fetchableProps = { 'compiledMagic': '', 'dynamicFilteringString': [ 'behind-the-scene * * noop', 'behind-the-scene * image noop', 'behind-the-scene * 3p noop', 'behind-the-scene * inline-script noop', 'behind-the-scene * 1p-script noop', 'behind-the-scene * 3p-script noop', 'behind-the-scene * 3p-frame noop' ].join('\n'), 'urlFilteringString': '', 'hostnameSwitchesString': [ 'no-large-media: behind-the-scene false' ].join('\n'), 'lastRestoreFile': '', 'lastRestoreTime': 0, 'lastBackupFile': '', 'lastBackupTime': 0, 'netWhitelist': µb.netWhitelistDefault, 'selfieMagic': '', 'version': '0.0.0.0' }; toFetch(µb.localSettings, fetchableProps); toFetch(µb.userSettings, fetchableProps); toFetch(µb.restoreBackupSettings, fetchableProps); vAPI.storage.get(fetchableProps, onFirstFetchReady); }; /******************************************************************************/ // TODO(seamless migration): // Eventually selected filter list keys will be loaded as a fetchable // property. Until then we need to handle backward and forward // compatibility, this means a special asynchronous call to load selected // filter lists. var onAdminSettingsRestored = function() { µb.loadSelectedFilterLists(onSelectedFilterListsLoaded); }; /******************************************************************************/ return function() { processCallbackQueue(µb.onBeforeStartQueue, function() { // https://github.com/gorhill/uBlock/issues/531 µb.restoreAdminSettings(onAdminSettingsRestored); }); }; /******************************************************************************/ })(); /******************************************************************************/ µBlock.restart();