2014-06-24 00:42:43 +02:00
|
|
|
/*******************************************************************************
|
|
|
|
|
2016-10-30 18:06:23 +01:00
|
|
|
uBlock Origin - a browser extension to block requests.
|
2018-07-24 15:17:18 +02:00
|
|
|
Copyright (C) 2014-present Raymond Hill
|
2014-06-24 00:42:43 +02:00
|
|
|
|
|
|
|
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
|
|
|
|
*/
|
|
|
|
|
2018-04-27 14:36:38 +02:00
|
|
|
/* global punycode, publicSuffixList */
|
2014-11-17 00:27:48 +01:00
|
|
|
|
2014-10-19 13:11:27 +02:00
|
|
|
'use strict';
|
2014-06-24 00:42:43 +02:00
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2015-03-07 05:36:09 +01:00
|
|
|
µBlock.getBytesInUse = function(callback) {
|
|
|
|
if ( typeof callback !== 'function' ) {
|
|
|
|
callback = this.noopFunc;
|
|
|
|
}
|
2018-08-11 16:39:43 +02:00
|
|
|
let bytesInUse;
|
|
|
|
let countdown = 0;
|
|
|
|
|
2019-02-14 19:33:55 +01:00
|
|
|
const process = count => {
|
2018-08-11 16:39:43 +02:00
|
|
|
if ( typeof count === 'number' ) {
|
|
|
|
if ( bytesInUse === undefined ) {
|
|
|
|
bytesInUse = 0;
|
|
|
|
}
|
|
|
|
bytesInUse += count;
|
|
|
|
}
|
|
|
|
countdown -= 1;
|
|
|
|
if ( countdown > 0 ) { return; }
|
2014-06-24 00:42:43 +02:00
|
|
|
µBlock.storageUsed = bytesInUse;
|
2015-03-07 05:36:09 +01:00
|
|
|
callback(bytesInUse);
|
2014-06-24 00:42:43 +02:00
|
|
|
};
|
2018-08-11 16:39:43 +02:00
|
|
|
|
2016-10-30 18:06:23 +01:00
|
|
|
// Not all platforms implement this method.
|
|
|
|
if ( vAPI.storage.getBytesInUse instanceof Function ) {
|
2018-08-11 16:39:43 +02:00
|
|
|
countdown += 1;
|
|
|
|
vAPI.storage.getBytesInUse(null, process);
|
|
|
|
}
|
2019-03-23 02:09:27 +01:00
|
|
|
if (
|
|
|
|
navigator.storage instanceof Object &&
|
|
|
|
navigator.storage.estimate instanceof Function
|
|
|
|
) {
|
2018-08-11 16:39:43 +02:00
|
|
|
countdown += 1;
|
2019-03-23 02:09:27 +01:00
|
|
|
navigator.storage.estimate().then(estimate => {
|
|
|
|
process(estimate.usage);
|
2019-02-14 19:33:55 +01:00
|
|
|
});
|
2018-08-11 16:39:43 +02:00
|
|
|
}
|
|
|
|
if ( countdown === 0 ) {
|
2016-10-30 18:06:23 +01:00
|
|
|
callback();
|
|
|
|
}
|
2014-06-24 00:42:43 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2015-11-29 23:06:58 +01:00
|
|
|
µBlock.saveLocalSettings = (function() {
|
2019-05-15 20:49:12 +02:00
|
|
|
const saveAfter = 4 * 60 * 1000;
|
2014-06-24 00:42:43 +02:00
|
|
|
|
2019-05-15 20:49:12 +02:00
|
|
|
const onTimeout = ( ) => {
|
|
|
|
const µb = µBlock;
|
2016-10-08 16:15:31 +02:00
|
|
|
if ( µb.localSettingsLastModified > µb.localSettingsLastSaved ) {
|
2018-07-24 15:17:18 +02:00
|
|
|
µb.saveLocalSettings();
|
2015-11-29 23:06:58 +01:00
|
|
|
}
|
|
|
|
vAPI.setTimeout(onTimeout, saveAfter);
|
|
|
|
};
|
2014-06-24 00:42:43 +02:00
|
|
|
|
2015-11-29 23:06:58 +01:00
|
|
|
vAPI.setTimeout(onTimeout, saveAfter);
|
|
|
|
|
2018-07-26 00:24:14 +02:00
|
|
|
return function(callback) {
|
|
|
|
this.localSettingsLastSaved = Date.now();
|
|
|
|
vAPI.storage.set(this.localSettings, callback);
|
|
|
|
};
|
2015-11-29 23:06:58 +01:00
|
|
|
})();
|
2014-06-24 00:42:43 +02:00
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
|
|
|
µBlock.saveUserSettings = function() {
|
2015-01-22 03:46:11 +01:00
|
|
|
vAPI.storage.set(this.userSettings);
|
2014-06-24 00:42:43 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2018-02-21 19:29:36 +01:00
|
|
|
µBlock.loadHiddenSettings = function() {
|
2019-02-17 21:40:09 +01:00
|
|
|
return new Promise(resolve => {
|
|
|
|
// >>>> start of executor
|
|
|
|
|
2018-08-30 14:32:56 +02:00
|
|
|
vAPI.storage.get('hiddenSettings', bin => {
|
2019-02-17 21:40:09 +01:00
|
|
|
if ( bin instanceof Object === false ) {
|
|
|
|
return resolve();
|
|
|
|
}
|
2019-02-14 19:33:55 +01:00
|
|
|
const hs = bin.hiddenSettings;
|
2018-02-21 19:29:36 +01:00
|
|
|
if ( hs instanceof Object ) {
|
2019-02-14 19:33:55 +01:00
|
|
|
const hsDefault = this.hiddenSettingsDefault;
|
|
|
|
for ( const key in hsDefault ) {
|
2018-02-21 19:29:36 +01:00
|
|
|
if (
|
|
|
|
hsDefault.hasOwnProperty(key) &&
|
|
|
|
hs.hasOwnProperty(key) &&
|
|
|
|
typeof hs[key] === typeof hsDefault[key]
|
|
|
|
) {
|
2018-08-30 14:32:56 +02:00
|
|
|
this.hiddenSettings[key] = hs[key];
|
2018-02-21 19:29:36 +01:00
|
|
|
}
|
|
|
|
}
|
2019-02-19 18:30:37 +01:00
|
|
|
if ( typeof this.hiddenSettings.suspendTabsUntilReady === 'boolean' ) {
|
|
|
|
this.hiddenSettings.suspendTabsUntilReady =
|
|
|
|
this.hiddenSettings.suspendTabsUntilReady
|
|
|
|
? 'yes'
|
|
|
|
: 'unset';
|
|
|
|
}
|
2018-02-21 19:29:36 +01:00
|
|
|
}
|
|
|
|
if ( vAPI.localStorage.getItem('immediateHiddenSettings') === null ) {
|
2018-08-30 14:32:56 +02:00
|
|
|
this.saveImmediateHiddenSettings();
|
2018-02-21 19:29:36 +01:00
|
|
|
}
|
2019-02-14 19:33:55 +01:00
|
|
|
self.log.verbosity = this.hiddenSettings.consoleLogLevel;
|
2019-02-17 21:40:09 +01:00
|
|
|
resolve();
|
|
|
|
});
|
|
|
|
|
|
|
|
// <<<< end of executor
|
2018-08-30 14:32:56 +02:00
|
|
|
});
|
2018-02-21 19:29:36 +01:00
|
|
|
};
|
|
|
|
|
2018-04-09 23:46:29 +02:00
|
|
|
// Note: Save only the settings which values differ from the default ones.
|
|
|
|
// This way the new default values in the future will properly apply for those
|
|
|
|
// which were not modified by the user.
|
|
|
|
|
2018-02-21 19:29:36 +01:00
|
|
|
µBlock.saveHiddenSettings = function(callback) {
|
2019-02-14 19:33:55 +01:00
|
|
|
const bin = { hiddenSettings: {} };
|
|
|
|
for ( const prop in this.hiddenSettings ) {
|
2018-04-09 23:46:29 +02:00
|
|
|
if (
|
|
|
|
this.hiddenSettings.hasOwnProperty(prop) &&
|
|
|
|
this.hiddenSettings[prop] !== this.hiddenSettingsDefault[prop]
|
|
|
|
) {
|
|
|
|
bin.hiddenSettings[prop] = this.hiddenSettings[prop];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
vAPI.storage.set(bin, callback);
|
2018-02-21 19:29:36 +01:00
|
|
|
this.saveImmediateHiddenSettings();
|
2019-02-14 19:33:55 +01:00
|
|
|
self.log.verbosity = this.hiddenSettings.consoleLogLevel;
|
2018-02-21 19:29:36 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
2016-11-03 16:20:47 +01:00
|
|
|
|
|
|
|
µBlock.hiddenSettingsFromString = function(raw) {
|
2019-05-15 20:49:12 +02:00
|
|
|
const out = Object.assign({}, this.hiddenSettingsDefault);
|
|
|
|
const lineIter = new this.LineIterator(raw);
|
2016-11-03 16:20:47 +01:00
|
|
|
while ( lineIter.eot() === false ) {
|
2019-05-15 20:49:12 +02:00
|
|
|
const line = lineIter.next();
|
|
|
|
const matches = /^\s*(\S+)\s+(.+)$/.exec(line);
|
2016-11-03 16:20:47 +01:00
|
|
|
if ( matches === null || matches.length !== 3 ) { continue; }
|
2019-05-15 20:49:12 +02:00
|
|
|
const name = matches[1];
|
2016-11-03 16:20:47 +01:00
|
|
|
if ( out.hasOwnProperty(name) === false ) { continue; }
|
2019-05-15 20:49:12 +02:00
|
|
|
const value = matches[2];
|
2016-11-03 16:20:47 +01:00
|
|
|
switch ( typeof out[name] ) {
|
|
|
|
case 'boolean':
|
|
|
|
if ( value === 'true' ) {
|
|
|
|
out[name] = true;
|
|
|
|
} else if ( value === 'false' ) {
|
|
|
|
out[name] = false;
|
|
|
|
}
|
|
|
|
break;
|
2016-11-06 22:27:21 +01:00
|
|
|
case 'string':
|
|
|
|
out[name] = value;
|
|
|
|
break;
|
2017-01-18 19:17:47 +01:00
|
|
|
case 'number':
|
|
|
|
out[name] = parseInt(value, 10);
|
|
|
|
if ( isNaN(out[name]) ) {
|
|
|
|
out[name] = this.hiddenSettingsDefault[name];
|
|
|
|
}
|
|
|
|
break;
|
2016-11-03 16:20:47 +01:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2018-02-21 19:29:36 +01:00
|
|
|
return out;
|
2016-11-03 16:20:47 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
µBlock.stringFromHiddenSettings = function() {
|
2019-05-15 20:49:12 +02:00
|
|
|
const out = [];
|
|
|
|
for ( const key of Object.keys(this.hiddenSettings).sort() ) {
|
2016-11-03 16:20:47 +01:00
|
|
|
out.push(key + ' ' + this.hiddenSettings[key]);
|
|
|
|
}
|
|
|
|
return out.join('\n');
|
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2018-02-21 19:29:36 +01:00
|
|
|
// These settings must be available immediately on startup, without delay
|
|
|
|
// through the vAPI.localStorage. Add/remove settings as needed.
|
|
|
|
|
|
|
|
µBlock.saveImmediateHiddenSettings = function() {
|
|
|
|
vAPI.localStorage.setItem(
|
|
|
|
'immediateHiddenSettings',
|
|
|
|
JSON.stringify({
|
2019-02-18 20:41:04 +01:00
|
|
|
consoleLogLevel: this.hiddenSettings.consoleLogLevel,
|
2018-11-16 16:19:06 +01:00
|
|
|
disableWebAssembly: this.hiddenSettings.disableWebAssembly,
|
2018-02-21 19:29:36 +01:00
|
|
|
suspendTabsUntilReady: this.hiddenSettings.suspendTabsUntilReady,
|
|
|
|
})
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2015-02-11 06:26:45 +01:00
|
|
|
µBlock.savePermanentFirewallRules = function() {
|
2018-09-03 20:06:49 +02:00
|
|
|
vAPI.storage.set({
|
|
|
|
dynamicFilteringString: this.permanentFirewall.toString()
|
|
|
|
});
|
2015-03-27 18:00:55 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2015-05-21 20:15:17 +02:00
|
|
|
µBlock.savePermanentURLFilteringRules = function() {
|
2018-09-03 20:06:49 +02:00
|
|
|
vAPI.storage.set({
|
|
|
|
urlFilteringString: this.permanentURLFiltering.toString()
|
|
|
|
});
|
2015-05-21 20:15:17 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2015-03-27 18:00:55 +01:00
|
|
|
µBlock.saveHostnameSwitches = function() {
|
2018-09-03 20:06:49 +02:00
|
|
|
vAPI.storage.set({
|
|
|
|
hostnameSwitchesString: this.permanentSwitches.toString()
|
|
|
|
});
|
2014-12-31 23:26:17 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2014-08-25 02:52:34 +02:00
|
|
|
µBlock.saveWhitelist = function() {
|
2018-09-03 20:06:49 +02:00
|
|
|
vAPI.storage.set({
|
2019-06-25 17:57:14 +02:00
|
|
|
netWhitelist: this.arrayFromWhitelist(this.netWhitelist)
|
2018-09-03 20:06:49 +02:00
|
|
|
});
|
2014-08-25 02:52:34 +02:00
|
|
|
this.netWhitelistModifyTime = Date.now();
|
|
|
|
};
|
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
/*******************************************************************************
|
2014-08-25 02:52:34 +02:00
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
TODO(seamless migration):
|
|
|
|
The code related to 'remoteBlacklist' can be removed when I am confident
|
|
|
|
all users have moved to a version of uBO which no longer depends on
|
|
|
|
the property 'remoteBlacklists, i.e. v1.11 and beyond.
|
2015-03-30 14:12:41 +02:00
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
**/
|
2015-03-30 14:12:41 +02:00
|
|
|
|
2019-02-17 21:40:09 +01:00
|
|
|
µBlock.loadSelectedFilterLists = function() {
|
|
|
|
return new Promise(resolve => {
|
|
|
|
// >>>> start of executor
|
|
|
|
|
|
|
|
vAPI.storage.get('selectedFilterLists', bin => {
|
2017-12-14 22:42:54 +01:00
|
|
|
// Select default filter lists if first-time launch.
|
2019-02-17 21:40:09 +01:00
|
|
|
if (
|
|
|
|
bin instanceof Object === false ||
|
|
|
|
Array.isArray(bin.selectedFilterLists) === false
|
|
|
|
) {
|
2019-02-18 12:18:30 +01:00
|
|
|
this.assets.metadata(availableLists => {
|
2019-02-17 21:40:09 +01:00
|
|
|
this.saveSelectedFilterLists(
|
|
|
|
this.autoSelectRegionalFilterLists(availableLists)
|
2018-01-01 13:52:03 +01:00
|
|
|
);
|
2019-02-17 21:40:09 +01:00
|
|
|
resolve();
|
2017-01-26 16:17:38 +01:00
|
|
|
});
|
|
|
|
return;
|
2015-03-30 14:12:41 +02:00
|
|
|
}
|
2019-02-17 21:40:09 +01:00
|
|
|
this.selectedFilterLists = bin.selectedFilterLists;
|
|
|
|
resolve();
|
|
|
|
});
|
|
|
|
|
|
|
|
// <<<< end of executor
|
2017-01-18 19:17:47 +01:00
|
|
|
});
|
|
|
|
};
|
2015-03-30 14:12:41 +02:00
|
|
|
|
2017-02-03 14:12:26 +01:00
|
|
|
µBlock.saveSelectedFilterLists = function(newKeys, append, callback) {
|
|
|
|
if ( typeof append === 'function' ) {
|
|
|
|
callback = append;
|
|
|
|
append = false;
|
|
|
|
}
|
2019-05-15 20:49:12 +02:00
|
|
|
const oldKeys = this.selectedFilterLists.slice();
|
2017-01-26 16:17:38 +01:00
|
|
|
if ( append ) {
|
|
|
|
newKeys = newKeys.concat(oldKeys);
|
|
|
|
}
|
2019-05-15 20:49:12 +02:00
|
|
|
const newSet = new Set(newKeys);
|
2017-01-26 16:17:38 +01:00
|
|
|
// Purge unused filter lists from cache.
|
2019-05-15 20:49:12 +02:00
|
|
|
for ( let i = 0, n = oldKeys.length; i < n; i++ ) {
|
2017-01-26 16:17:38 +01:00
|
|
|
if ( newSet.has(oldKeys[i]) === false ) {
|
|
|
|
this.removeFilterList(oldKeys[i]);
|
2017-01-22 22:05:16 +01:00
|
|
|
}
|
2017-01-26 16:17:38 +01:00
|
|
|
}
|
2018-06-01 13:54:31 +02:00
|
|
|
newKeys = Array.from(newSet);
|
2017-01-26 16:17:38 +01:00
|
|
|
this.selectedFilterLists = newKeys;
|
2019-05-15 20:49:12 +02:00
|
|
|
vAPI.storage.set({ selectedFilterLists: newKeys }, callback);
|
2017-01-18 19:17:47 +01:00
|
|
|
};
|
|
|
|
|
2015-03-30 14:12:41 +02:00
|
|
|
/******************************************************************************/
|
|
|
|
|
2017-01-22 22:05:16 +01:00
|
|
|
µBlock.applyFilterListSelection = function(details, callback) {
|
2019-05-15 20:49:12 +02:00
|
|
|
let selectedListKeySet = new Set(this.selectedFilterLists);
|
|
|
|
let externalLists = this.userSettings.externalLists;
|
2017-01-22 22:05:16 +01:00
|
|
|
|
|
|
|
// Filter lists to select
|
|
|
|
if ( Array.isArray(details.toSelect) ) {
|
|
|
|
if ( details.merge ) {
|
2019-05-15 20:49:12 +02:00
|
|
|
for ( let i = 0, n = details.toSelect.length; i < n; i++ ) {
|
2017-01-22 22:05:16 +01:00
|
|
|
selectedListKeySet.add(details.toSelect[i]);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
selectedListKeySet = new Set(details.toSelect);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Imported filter lists to remove
|
|
|
|
if ( Array.isArray(details.toRemove) ) {
|
2019-05-15 20:49:12 +02:00
|
|
|
const removeURLFromHaystack = (haystack, needle) => {
|
2017-01-22 22:05:16 +01:00
|
|
|
return haystack.replace(
|
|
|
|
new RegExp(
|
|
|
|
'(^|\\n)' +
|
2019-05-15 20:49:12 +02:00
|
|
|
this.escapeRegex(needle) +
|
2017-01-22 22:05:16 +01:00
|
|
|
'(\\n|$)', 'g'),
|
|
|
|
'\n'
|
|
|
|
).trim();
|
|
|
|
};
|
2019-05-15 20:49:12 +02:00
|
|
|
for ( let i = 0, n = details.toRemove.length; i < n; i++ ) {
|
|
|
|
const assetKey = details.toRemove[i];
|
2017-01-22 22:05:16 +01:00
|
|
|
selectedListKeySet.delete(assetKey);
|
|
|
|
externalLists = removeURLFromHaystack(externalLists, assetKey);
|
|
|
|
this.removeFilterList(assetKey);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Filter lists to import
|
|
|
|
if ( typeof details.toImport === 'string' ) {
|
|
|
|
// https://github.com/gorhill/uBlock/issues/1181
|
2019-05-15 20:49:12 +02:00
|
|
|
// Try mapping the URL of an imported filter list to the assetKey
|
|
|
|
// of an existing stock list.
|
|
|
|
const assetKeyFromURL = url => {
|
|
|
|
const needle = url.replace(/^https?:/, '');
|
|
|
|
const assets = this.availableFilterLists;
|
|
|
|
for ( const assetKey in assets ) {
|
|
|
|
const asset = assets[assetKey];
|
2017-01-22 22:05:16 +01:00
|
|
|
if ( asset.content !== 'filters' ) { continue; }
|
|
|
|
if ( typeof asset.contentURL === 'string' ) {
|
|
|
|
if ( asset.contentURL.endsWith(needle) ) { return assetKey; }
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if ( Array.isArray(asset.contentURL) === false ) { continue; }
|
2019-05-15 20:49:12 +02:00
|
|
|
for ( let i = 0, n = asset.contentURL.length; i < n; i++ ) {
|
2017-01-22 22:05:16 +01:00
|
|
|
if ( asset.contentURL[i].endsWith(needle) ) {
|
|
|
|
return assetKey;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return url;
|
|
|
|
};
|
2019-05-15 20:49:12 +02:00
|
|
|
const importedSet = new Set(this.listKeysFromCustomFilterLists(externalLists));
|
|
|
|
const toImportSet = new Set(this.listKeysFromCustomFilterLists(details.toImport));
|
|
|
|
for ( const urlKey of toImportSet ) {
|
2017-05-19 16:12:55 +02:00
|
|
|
if ( importedSet.has(urlKey) ) { continue; }
|
2019-05-15 20:49:12 +02:00
|
|
|
const assetKey = assetKeyFromURL(urlKey);
|
2017-05-19 16:12:55 +02:00
|
|
|
if ( assetKey === urlKey ) {
|
|
|
|
importedSet.add(urlKey);
|
2017-01-22 22:05:16 +01:00
|
|
|
}
|
|
|
|
selectedListKeySet.add(assetKey);
|
|
|
|
}
|
2018-06-01 13:54:31 +02:00
|
|
|
externalLists = Array.from(importedSet).sort().join('\n');
|
2017-01-22 22:05:16 +01:00
|
|
|
}
|
|
|
|
|
2019-05-15 20:49:12 +02:00
|
|
|
const result = Array.from(selectedListKeySet);
|
2017-01-22 22:05:16 +01:00
|
|
|
if ( externalLists !== this.userSettings.externalLists ) {
|
|
|
|
this.userSettings.externalLists = externalLists;
|
|
|
|
vAPI.storage.set({ externalLists: externalLists });
|
|
|
|
}
|
|
|
|
this.saveSelectedFilterLists(result);
|
|
|
|
if ( typeof callback === 'function' ) {
|
|
|
|
callback(result);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
|
|
|
µBlock.listKeysFromCustomFilterLists = function(raw) {
|
2019-05-15 20:49:12 +02:00
|
|
|
const out = new Set();
|
|
|
|
const reIgnore = /^[!#]/;
|
|
|
|
const reValid = /^[a-z-]+:\/\/\S+/;
|
|
|
|
const lineIter = new this.LineIterator(raw);
|
2017-01-22 22:05:16 +01:00
|
|
|
while ( lineIter.eot() === false ) {
|
2019-05-15 20:49:12 +02:00
|
|
|
const location = lineIter.next().trim();
|
|
|
|
if ( reIgnore.test(location) || !reValid.test(location) ) { continue; }
|
2017-01-22 22:05:16 +01:00
|
|
|
out.add(location);
|
|
|
|
}
|
2018-06-01 13:54:31 +02:00
|
|
|
return Array.from(out);
|
2017-01-22 22:05:16 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2014-07-13 02:32:44 +02:00
|
|
|
µBlock.saveUserFilters = function(content, callback) {
|
2015-12-07 14:59:22 +01:00
|
|
|
// https://github.com/gorhill/uBlock/issues/1022
|
2019-05-15 20:49:12 +02:00
|
|
|
// Be sure to end with an empty line.
|
2015-12-07 14:59:22 +01:00
|
|
|
content = content.trim();
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( content !== '' ) { content += '\n'; }
|
2015-08-11 21:29:14 +02:00
|
|
|
this.assets.put(this.userFiltersPath, content, callback);
|
2017-01-18 19:17:47 +01:00
|
|
|
this.removeCompiledFilterList(this.userFiltersPath);
|
2014-07-13 02:32:44 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
µBlock.loadUserFilters = function(callback) {
|
|
|
|
return this.assets.get(this.userFiltersPath, callback);
|
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2019-01-08 13:37:50 +01:00
|
|
|
µBlock.appendUserFilters = function(filters, options) {
|
|
|
|
filters = filters.trim();
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( filters.length === 0 ) { return; }
|
2015-02-24 00:31:29 +01:00
|
|
|
|
2019-01-08 13:37:50 +01:00
|
|
|
// https://github.com/uBlockOrigin/uBlock-issues/issues/372
|
|
|
|
// Auto comment using user-defined template.
|
|
|
|
let comment = '';
|
|
|
|
if (
|
|
|
|
options instanceof Object &&
|
|
|
|
options.autoComment === true &&
|
|
|
|
this.hiddenSettings.autoCommentFilterTemplate.indexOf('{{') !== -1
|
|
|
|
) {
|
|
|
|
const d = new Date();
|
|
|
|
comment =
|
|
|
|
'! ' +
|
|
|
|
this.hiddenSettings.autoCommentFilterTemplate
|
|
|
|
.replace('{{date}}', d.toLocaleDateString())
|
|
|
|
.replace('{{time}}', d.toLocaleTimeString())
|
|
|
|
.replace('{{origin}}', options.origin);
|
|
|
|
}
|
|
|
|
|
2018-12-15 16:46:17 +01:00
|
|
|
const onSaved = ( ) => {
|
|
|
|
const compiledFilters = this.compileFilters(
|
|
|
|
filters,
|
|
|
|
{ assetKey: this.userFiltersPath }
|
|
|
|
);
|
|
|
|
const snfe = this.staticNetFilteringEngine;
|
|
|
|
const cfe = this.cosmeticFilteringEngine;
|
|
|
|
const acceptedCount = snfe.acceptedCount + cfe.acceptedCount;
|
|
|
|
const discardedCount = snfe.discardedCount + cfe.discardedCount;
|
|
|
|
this.applyCompiledFilters(compiledFilters, true);
|
|
|
|
const entry = this.availableFilterLists[this.userFiltersPath];
|
|
|
|
const deltaEntryCount =
|
|
|
|
snfe.acceptedCount +
|
|
|
|
cfe.acceptedCount - acceptedCount;
|
|
|
|
const deltaEntryUsedCount =
|
|
|
|
deltaEntryCount -
|
|
|
|
(snfe.discardedCount + cfe.discardedCount - discardedCount);
|
2015-04-27 00:31:51 +02:00
|
|
|
entry.entryCount += deltaEntryCount;
|
|
|
|
entry.entryUsedCount += deltaEntryUsedCount;
|
2018-12-15 16:46:17 +01:00
|
|
|
vAPI.storage.set({ 'availableFilterLists': this.availableFilterLists });
|
|
|
|
this.staticNetFilteringEngine.freeze();
|
|
|
|
this.redirectEngine.freeze();
|
|
|
|
this.staticExtFilteringEngine.freeze();
|
|
|
|
this.selfieManager.destroy();
|
2015-02-24 00:31:29 +01:00
|
|
|
};
|
|
|
|
|
2018-12-15 16:46:17 +01:00
|
|
|
const onLoaded = details => {
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( details.error ) { return; }
|
2019-01-08 13:37:50 +01:00
|
|
|
// The comment, if any, will be applied if and only if it is different
|
|
|
|
// from the last comment found in the user filter list.
|
|
|
|
if ( comment !== '' ) {
|
|
|
|
const pos = details.content.lastIndexOf(comment);
|
|
|
|
if (
|
|
|
|
pos === -1 ||
|
|
|
|
details.content.indexOf('\n!', pos + 1) !== -1
|
|
|
|
) {
|
|
|
|
filters = '\n' + comment + '\n' + filters;
|
|
|
|
}
|
|
|
|
}
|
2015-04-07 03:26:05 +02:00
|
|
|
// https://github.com/chrisaljoudi/uBlock/issues/976
|
2019-01-08 13:37:50 +01:00
|
|
|
// If we reached this point, the filter quite probably needs to be
|
|
|
|
// added for sure: do not try to be too smart, trying to avoid
|
|
|
|
// duplicates at this point may lead to more issues.
|
|
|
|
this.saveUserFilters(details.content.trim() + '\n' + filters, onSaved);
|
2014-07-13 02:32:44 +02:00
|
|
|
};
|
2014-09-08 23:46:58 +02:00
|
|
|
|
2015-02-24 00:31:29 +01:00
|
|
|
this.loadUserFilters(onLoaded);
|
2014-07-13 02:32:44 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
µBlock.autoSelectRegionalFilterLists = function(lists) {
|
2019-05-15 20:49:12 +02:00
|
|
|
const selectedListKeys = [ this.userFiltersPath ];
|
|
|
|
for ( const key in lists ) {
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( lists.hasOwnProperty(key) === false ) { continue; }
|
2019-05-15 20:49:12 +02:00
|
|
|
const list = lists[key];
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( list.off !== true ) {
|
|
|
|
selectedListKeys.push(key);
|
|
|
|
continue;
|
2015-02-25 22:51:04 +01:00
|
|
|
}
|
2017-11-09 18:53:05 +01:00
|
|
|
if ( this.listMatchesEnvironment(list) ) {
|
2017-01-18 19:17:47 +01:00
|
|
|
selectedListKeys.push(key);
|
|
|
|
list.off = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return selectedListKeys;
|
|
|
|
};
|
2015-02-25 22:51:04 +01:00
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
/******************************************************************************/
|
|
|
|
|
|
|
|
µBlock.getAvailableLists = function(callback) {
|
2019-05-15 20:49:12 +02:00
|
|
|
let oldAvailableLists = {},
|
2017-01-18 19:17:47 +01:00
|
|
|
newAvailableLists = {};
|
|
|
|
|
|
|
|
// User filter list.
|
|
|
|
newAvailableLists[this.userFiltersPath] = {
|
2018-04-09 21:45:25 +02:00
|
|
|
group: 'user',
|
2017-01-18 19:17:47 +01:00
|
|
|
title: vAPI.i18n('1pPageName')
|
|
|
|
};
|
|
|
|
|
|
|
|
// Custom filter lists.
|
2019-05-15 20:49:12 +02:00
|
|
|
const importedListKeys = this.listKeysFromCustomFilterLists(
|
|
|
|
this.userSettings.externalLists
|
|
|
|
);
|
|
|
|
for ( const listKey of importedListKeys ) {
|
|
|
|
const entry = {
|
2017-01-18 19:17:47 +01:00
|
|
|
content: 'filters',
|
2017-07-03 15:08:46 +02:00
|
|
|
contentURL: listKey,
|
2017-01-18 19:17:47 +01:00
|
|
|
external: true,
|
|
|
|
group: 'custom',
|
|
|
|
submitter: 'user',
|
|
|
|
title: ''
|
|
|
|
};
|
|
|
|
newAvailableLists[listKey] = entry;
|
|
|
|
this.assets.registerAssetSource(listKey, entry);
|
|
|
|
}
|
|
|
|
|
2017-07-03 15:08:46 +02:00
|
|
|
// Convert a no longer existing stock list into an imported list.
|
2019-05-15 20:49:12 +02:00
|
|
|
const customListFromStockList = assetKey => {
|
|
|
|
const oldEntry = oldAvailableLists[assetKey];
|
2017-07-03 15:08:46 +02:00
|
|
|
if ( oldEntry === undefined || oldEntry.off === true ) { return; }
|
2019-05-15 20:49:12 +02:00
|
|
|
let listURL = oldEntry.contentURL;
|
2017-07-03 15:08:46 +02:00
|
|
|
if ( Array.isArray(listURL) ) {
|
|
|
|
listURL = listURL[0];
|
|
|
|
}
|
2019-05-15 20:49:12 +02:00
|
|
|
const newEntry = {
|
2017-07-03 15:08:46 +02:00
|
|
|
content: 'filters',
|
|
|
|
contentURL: listURL,
|
|
|
|
external: true,
|
|
|
|
group: 'custom',
|
|
|
|
submitter: 'user',
|
|
|
|
title: oldEntry.title || ''
|
|
|
|
};
|
|
|
|
newAvailableLists[listURL] = newEntry;
|
2019-05-15 20:49:12 +02:00
|
|
|
this.assets.registerAssetSource(listURL, newEntry);
|
2017-07-03 15:08:46 +02:00
|
|
|
importedListKeys.push(listURL);
|
2019-05-15 20:49:12 +02:00
|
|
|
this.userSettings.externalLists += '\n' + listURL;
|
|
|
|
this.userSettings.externalLists = this.userSettings.externalLists.trim();
|
|
|
|
vAPI.storage.set({ externalLists: this.userSettings.externalLists });
|
|
|
|
this.saveSelectedFilterLists([ listURL ], true);
|
2017-07-03 15:08:46 +02:00
|
|
|
};
|
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
// Final steps:
|
|
|
|
// - reuse existing list metadata if any;
|
|
|
|
// - unregister unreferenced imported filter lists if any.
|
2019-05-15 20:49:12 +02:00
|
|
|
const finalize = ( ) => {
|
2017-01-18 19:17:47 +01:00
|
|
|
// Reuse existing metadata.
|
2019-05-15 20:49:12 +02:00
|
|
|
for ( const assetKey in oldAvailableLists ) {
|
|
|
|
const oldEntry = oldAvailableLists[assetKey];
|
|
|
|
const newEntry = newAvailableLists[assetKey];
|
2017-07-03 15:08:46 +02:00
|
|
|
// List no longer exists. If a stock list, try to convert to
|
|
|
|
// imported list if it was selected.
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( newEntry === undefined ) {
|
2019-05-15 20:49:12 +02:00
|
|
|
this.removeFilterList(assetKey);
|
2017-07-03 15:08:46 +02:00
|
|
|
if ( assetKey.indexOf('://') === -1 ) {
|
|
|
|
customListFromStockList(assetKey);
|
|
|
|
}
|
2017-01-18 19:17:47 +01:00
|
|
|
continue;
|
2015-08-18 17:44:24 +02:00
|
|
|
}
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( oldEntry.entryCount !== undefined ) {
|
|
|
|
newEntry.entryCount = oldEntry.entryCount;
|
2014-08-25 02:52:34 +02:00
|
|
|
}
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( oldEntry.entryUsedCount !== undefined ) {
|
|
|
|
newEntry.entryUsedCount = oldEntry.entryUsedCount;
|
2014-07-25 22:12:20 +02:00
|
|
|
}
|
2015-03-11 04:46:18 +01:00
|
|
|
// This may happen if the list name was pulled from the list
|
|
|
|
// content.
|
2015-04-07 03:26:05 +02:00
|
|
|
// https://github.com/chrisaljoudi/uBlock/issues/982
|
2015-03-11 04:46:18 +01:00
|
|
|
// There is no guarantee the title was successfully extracted from
|
|
|
|
// the list content.
|
2017-01-18 19:17:47 +01:00
|
|
|
if (
|
|
|
|
newEntry.title === '' &&
|
|
|
|
typeof oldEntry.title === 'string' &&
|
|
|
|
oldEntry.title !== ''
|
2015-03-11 04:46:18 +01:00
|
|
|
) {
|
2017-01-18 19:17:47 +01:00
|
|
|
newEntry.title = oldEntry.title;
|
2014-09-26 02:21:21 +02:00
|
|
|
}
|
2014-06-24 00:42:43 +02:00
|
|
|
}
|
2016-01-03 19:58:25 +01:00
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
// Remove unreferenced imported filter lists.
|
2019-05-15 20:49:12 +02:00
|
|
|
const dict = new Set(importedListKeys);
|
|
|
|
for ( const assetKey in newAvailableLists ) {
|
|
|
|
const newEntry = newAvailableLists[assetKey];
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( newEntry.submitter !== 'user' ) { continue; }
|
|
|
|
if ( dict.has(assetKey) ) { continue; }
|
|
|
|
delete newAvailableLists[assetKey];
|
2019-05-15 20:49:12 +02:00
|
|
|
this.assets.unregisterAssetSource(assetKey);
|
|
|
|
this.removeFilterList(assetKey);
|
2016-01-03 19:58:25 +01:00
|
|
|
}
|
2014-06-24 00:42:43 +02:00
|
|
|
};
|
|
|
|
|
2017-01-26 16:17:38 +01:00
|
|
|
// Built-in filter lists loaded.
|
2019-05-15 20:49:12 +02:00
|
|
|
const onBuiltinListsLoaded = entries => {
|
|
|
|
for ( const assetKey in entries ) {
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( entries.hasOwnProperty(assetKey) === false ) { continue; }
|
2019-05-15 20:49:12 +02:00
|
|
|
const entry = entries[assetKey];
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( entry.content !== 'filters' ) { continue; }
|
2018-04-27 14:36:38 +02:00
|
|
|
newAvailableLists[assetKey] = Object.assign({}, entry);
|
2014-07-25 22:12:20 +02:00
|
|
|
}
|
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
// Load set of currently selected filter lists.
|
2019-05-15 20:49:12 +02:00
|
|
|
const listKeySet = new Set(this.selectedFilterLists);
|
|
|
|
for ( const listKey in newAvailableLists ) {
|
2017-01-26 16:17:38 +01:00
|
|
|
if ( newAvailableLists.hasOwnProperty(listKey) ) {
|
|
|
|
newAvailableLists[listKey].off = !listKeySet.has(listKey);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
finalize();
|
|
|
|
callback(newAvailableLists);
|
2017-01-18 19:17:47 +01:00
|
|
|
};
|
2016-01-03 19:58:25 +01:00
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
// Available lists previously computed.
|
2019-05-15 20:49:12 +02:00
|
|
|
const onOldAvailableListsLoaded = bin => {
|
2017-01-18 19:17:47 +01:00
|
|
|
oldAvailableLists = bin && bin.availableFilterLists || {};
|
2019-05-15 20:49:12 +02:00
|
|
|
this.assets.metadata(onBuiltinListsLoaded);
|
2017-01-18 19:17:47 +01:00
|
|
|
};
|
2016-01-03 19:58:25 +01:00
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
// Load previously saved available lists -- these contains data
|
|
|
|
// computed at run-time, we will reuse this data if possible.
|
|
|
|
vAPI.storage.get('availableFilterLists', onOldAvailableListsLoaded);
|
2015-02-24 00:31:29 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2016-01-03 19:58:25 +01:00
|
|
|
// This is used to be re-entrancy resistant.
|
|
|
|
µBlock.loadingFilterLists = false;
|
|
|
|
|
2014-09-08 23:46:58 +02:00
|
|
|
µBlock.loadFilterLists = function(callback) {
|
2016-01-03 19:58:25 +01:00
|
|
|
// Callers are expected to check this first.
|
2018-11-03 12:58:46 +01:00
|
|
|
if ( this.loadingFilterLists ) { return; }
|
2016-01-03 19:58:25 +01:00
|
|
|
this.loadingFilterLists = true;
|
2015-02-24 00:31:29 +01:00
|
|
|
|
2018-12-06 01:18:20 +01:00
|
|
|
const loadedListKeys = [];
|
|
|
|
let filterlistsCount = 0;
|
2014-07-25 22:12:20 +02:00
|
|
|
|
2014-09-08 23:46:58 +02:00
|
|
|
if ( typeof callback !== 'function' ) {
|
|
|
|
callback = this.noopFunc;
|
|
|
|
}
|
|
|
|
|
2019-05-15 20:49:12 +02:00
|
|
|
const onDone = ( ) => {
|
|
|
|
this.staticNetFilteringEngine.freeze();
|
|
|
|
this.staticExtFilteringEngine.freeze();
|
|
|
|
this.redirectEngine.freeze();
|
|
|
|
vAPI.storage.set({ 'availableFilterLists': this.availableFilterLists });
|
2015-02-24 19:48:03 +01:00
|
|
|
|
2017-01-20 21:17:11 +01:00
|
|
|
vAPI.messaging.broadcast({
|
|
|
|
what: 'staticFilteringDataChanged',
|
2019-05-15 20:49:12 +02:00
|
|
|
parseCosmeticFilters: this.userSettings.parseAllABPHideFilters,
|
|
|
|
ignoreGenericCosmeticFilters: this.userSettings.ignoreGenericCosmeticFilters,
|
2017-01-20 21:17:11 +01:00
|
|
|
listKeys: loadedListKeys
|
|
|
|
});
|
2016-09-24 20:36:08 +02:00
|
|
|
|
2014-09-08 23:46:58 +02:00
|
|
|
callback();
|
2015-02-24 00:31:29 +01:00
|
|
|
|
2019-05-15 20:49:12 +02:00
|
|
|
this.selfieManager.destroy();
|
|
|
|
this.lz4Codec.relinquish();
|
2018-12-06 01:18:20 +01:00
|
|
|
|
2019-05-15 20:49:12 +02:00
|
|
|
this.loadingFilterLists = false;
|
2014-07-25 22:12:20 +02:00
|
|
|
};
|
|
|
|
|
2019-05-15 20:49:12 +02:00
|
|
|
const applyCompiledFilters = (assetKey, compiled) => {
|
|
|
|
const snfe = this.staticNetFilteringEngine;
|
|
|
|
const sxfe = this.staticExtFilteringEngine;
|
2018-12-06 01:18:20 +01:00
|
|
|
let acceptedCount = snfe.acceptedCount + sxfe.acceptedCount,
|
2017-12-29 19:31:37 +01:00
|
|
|
discardedCount = snfe.discardedCount + sxfe.discardedCount;
|
2019-05-15 20:49:12 +02:00
|
|
|
this.applyCompiledFilters(compiled, assetKey === this.userFiltersPath);
|
|
|
|
if ( this.availableFilterLists.hasOwnProperty(assetKey) ) {
|
|
|
|
const entry = this.availableFilterLists[assetKey];
|
2017-12-29 19:31:37 +01:00
|
|
|
entry.entryCount = snfe.acceptedCount + sxfe.acceptedCount -
|
|
|
|
acceptedCount;
|
|
|
|
entry.entryUsedCount = entry.entryCount -
|
|
|
|
(snfe.discardedCount + sxfe.discardedCount - discardedCount);
|
2014-06-24 00:42:43 +02:00
|
|
|
}
|
2017-01-20 21:17:11 +01:00
|
|
|
loadedListKeys.push(assetKey);
|
2014-06-24 00:42:43 +02:00
|
|
|
};
|
|
|
|
|
2019-05-15 20:49:12 +02:00
|
|
|
const onCompiledListLoaded = details => {
|
2017-01-18 19:17:47 +01:00
|
|
|
applyCompiledFilters(details.assetKey, details.content);
|
2015-02-24 00:31:29 +01:00
|
|
|
filterlistsCount -= 1;
|
|
|
|
if ( filterlistsCount === 0 ) {
|
|
|
|
onDone();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-05-15 20:49:12 +02:00
|
|
|
const onFilterListsReady = lists => {
|
|
|
|
this.availableFilterLists = lists;
|
2015-02-24 00:31:29 +01:00
|
|
|
|
2019-05-15 20:49:12 +02:00
|
|
|
this.redirectEngine.reset();
|
|
|
|
this.staticExtFilteringEngine.reset();
|
|
|
|
this.staticNetFilteringEngine.reset();
|
|
|
|
this.selfieManager.destroy();
|
|
|
|
this.staticFilteringReverseLookup.resetLists();
|
2014-06-24 00:42:43 +02:00
|
|
|
|
2015-02-24 00:31:29 +01:00
|
|
|
// We need to build a complete list of assets to pull first: this is
|
|
|
|
// because it *may* happens that some load operations are synchronous:
|
|
|
|
// This happens for assets which do not exist, ot assets with no
|
|
|
|
// content.
|
2018-12-06 01:18:20 +01:00
|
|
|
const toLoad = [];
|
|
|
|
for ( const assetKey in lists ) {
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( lists.hasOwnProperty(assetKey) === false ) { continue; }
|
|
|
|
if ( lists[assetKey].off ) { continue; }
|
|
|
|
toLoad.push(assetKey);
|
2015-02-06 07:20:04 +01:00
|
|
|
}
|
2015-02-24 00:31:29 +01:00
|
|
|
filterlistsCount = toLoad.length;
|
|
|
|
if ( filterlistsCount === 0 ) {
|
2016-01-03 19:58:25 +01:00
|
|
|
return onDone();
|
2015-02-24 00:31:29 +01:00
|
|
|
}
|
|
|
|
|
2018-12-06 01:18:20 +01:00
|
|
|
let i = toLoad.length;
|
2015-02-24 00:31:29 +01:00
|
|
|
while ( i-- ) {
|
2019-05-15 20:49:12 +02:00
|
|
|
this.getCompiledFilterList(toLoad[i], onCompiledListLoaded);
|
2014-06-24 00:42:43 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2015-02-24 00:31:29 +01:00
|
|
|
this.getAvailableLists(onFilterListsReady);
|
2015-12-16 03:34:36 +01:00
|
|
|
this.loadRedirectResources();
|
2014-06-24 00:42:43 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
µBlock.getCompiledFilterList = function(assetKey, callback) {
|
2019-05-15 20:49:12 +02:00
|
|
|
const compiledPath = 'compiled/' + assetKey;
|
|
|
|
let rawContent;
|
2014-06-24 00:42:43 +02:00
|
|
|
|
2019-05-15 20:49:12 +02:00
|
|
|
const onCompiledListLoaded2 = details => {
|
2017-05-20 21:32:03 +02:00
|
|
|
if ( details.content === '' ) {
|
2019-05-15 20:49:12 +02:00
|
|
|
details.content = this.compileFilters(
|
2018-12-15 16:46:17 +01:00
|
|
|
rawContent,
|
|
|
|
{ assetKey: assetKey }
|
|
|
|
);
|
2019-05-15 20:49:12 +02:00
|
|
|
this.assets.put(compiledPath, details.content);
|
2017-05-20 21:32:03 +02:00
|
|
|
}
|
|
|
|
rawContent = undefined;
|
2017-01-18 19:17:47 +01:00
|
|
|
details.assetKey = assetKey;
|
2017-05-20 21:32:03 +02:00
|
|
|
callback(details);
|
|
|
|
};
|
|
|
|
|
2019-05-15 20:49:12 +02:00
|
|
|
const onRawListLoaded = details => {
|
2015-06-08 14:09:08 +02:00
|
|
|
if ( details.content === '' ) {
|
2017-05-20 21:32:03 +02:00
|
|
|
details.assetKey = assetKey;
|
2015-06-08 14:09:08 +02:00
|
|
|
callback(details);
|
|
|
|
return;
|
|
|
|
}
|
2019-05-15 20:49:12 +02:00
|
|
|
this.extractFilterListMetadata(assetKey, details.content);
|
2017-05-20 21:32:03 +02:00
|
|
|
// Fectching the raw content may cause the compiled content to be
|
|
|
|
// generated somewhere else in uBO, hence we try one last time to
|
|
|
|
// fetch the compiled content in case it has become available.
|
|
|
|
rawContent = details.content;
|
2019-05-15 20:49:12 +02:00
|
|
|
this.assets.get(compiledPath, onCompiledListLoaded2);
|
2015-02-24 00:31:29 +01:00
|
|
|
};
|
2014-09-25 21:44:18 +02:00
|
|
|
|
2019-05-15 20:49:12 +02:00
|
|
|
const onCompiledListLoaded1 = details => {
|
2015-02-24 00:31:29 +01:00
|
|
|
if ( details.content === '' ) {
|
2019-05-15 20:49:12 +02:00
|
|
|
this.assets.get(assetKey, onRawListLoaded);
|
2015-02-24 00:31:29 +01:00
|
|
|
return;
|
|
|
|
}
|
2017-01-18 19:17:47 +01:00
|
|
|
details.assetKey = assetKey;
|
2015-02-24 00:31:29 +01:00
|
|
|
callback(details);
|
|
|
|
};
|
2014-09-25 21:44:18 +02:00
|
|
|
|
2017-05-20 21:32:03 +02:00
|
|
|
this.assets.get(compiledPath, onCompiledListLoaded1);
|
2015-02-24 00:31:29 +01:00
|
|
|
};
|
2014-09-26 02:21:21 +02:00
|
|
|
|
2015-02-24 00:31:29 +01:00
|
|
|
/******************************************************************************/
|
2014-09-26 02:21:21 +02:00
|
|
|
|
2018-02-23 12:42:17 +01:00
|
|
|
// https://github.com/gorhill/uBlock/issues/3406
|
|
|
|
// Lower minimum update period to 1 day.
|
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
µBlock.extractFilterListMetadata = function(assetKey, raw) {
|
2019-05-15 20:49:12 +02:00
|
|
|
const listEntry = this.availableFilterLists[assetKey];
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( listEntry === undefined ) { return; }
|
|
|
|
// Metadata expected to be found at the top of content.
|
2019-05-15 20:49:12 +02:00
|
|
|
const head = raw.slice(0, 1024);
|
2017-01-18 19:17:47 +01:00
|
|
|
// https://github.com/gorhill/uBlock/issues/313
|
|
|
|
// Always try to fetch the name if this is an external filter list.
|
|
|
|
if ( listEntry.title === '' || listEntry.group === 'custom' ) {
|
2019-05-15 20:49:12 +02:00
|
|
|
const matches = head.match(/(?:^|\n)(?:!|# )[\t ]*Title[\t ]*:([^\n]+)/i);
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( matches !== null ) {
|
2017-05-26 18:46:36 +02:00
|
|
|
// https://bugs.chromium.org/p/v8/issues/detail?id=2869
|
2018-10-23 19:01:08 +02:00
|
|
|
// orphanizeString is to work around String.slice()
|
|
|
|
// potentially causing the whole raw filter list to be held in
|
|
|
|
// memory just because we cut out the title as a substring.
|
|
|
|
listEntry.title = this.orphanizeString(matches[1].trim());
|
2017-01-18 19:17:47 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// Extract update frequency information
|
2019-05-15 20:49:12 +02:00
|
|
|
const matches = head.match(/(?:^|\n)(?:!|# )[\t ]*Expires[\t ]*:[\t ]*(\d+)[\t ]*(h)?/i);
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( matches !== null ) {
|
2018-10-23 19:01:08 +02:00
|
|
|
let v = Math.max(parseInt(matches[1], 10), 1);
|
2018-05-16 20:55:12 +02:00
|
|
|
if ( matches[2] !== undefined ) {
|
2018-05-16 17:01:40 +02:00
|
|
|
v = Math.ceil(v / 24);
|
|
|
|
}
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( v !== listEntry.updateAfter ) {
|
|
|
|
this.assets.registerAssetSource(assetKey, { updateAfter: v });
|
|
|
|
}
|
|
|
|
}
|
2014-09-25 21:44:18 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
2015-02-25 22:33:51 +01:00
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
µBlock.removeCompiledFilterList = function(assetKey) {
|
|
|
|
this.assets.remove('compiled/' + assetKey);
|
|
|
|
};
|
|
|
|
|
|
|
|
µBlock.removeFilterList = function(assetKey) {
|
|
|
|
this.removeCompiledFilterList(assetKey);
|
|
|
|
this.assets.remove(assetKey);
|
2015-02-25 22:33:51 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
2014-09-25 21:44:18 +02:00
|
|
|
|
2018-12-15 16:46:17 +01:00
|
|
|
µBlock.compileFilters = function(rawText, details) {
|
2018-10-23 19:01:08 +02:00
|
|
|
let writer = new this.CompiledLineIO.Writer();
|
2014-06-24 00:42:43 +02:00
|
|
|
|
2018-12-15 16:46:17 +01:00
|
|
|
// Populate the writer with information potentially useful to the
|
|
|
|
// client compilers.
|
|
|
|
if ( details ) {
|
|
|
|
if ( details.assetKey ) {
|
|
|
|
writer.properties.set('assetKey', details.assetKey);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-24 00:42:43 +02:00
|
|
|
// Useful references:
|
|
|
|
// https://adblockplus.org/en/filter-cheatsheet
|
|
|
|
// https://adblockplus.org/en/filters
|
2018-12-15 16:46:17 +01:00
|
|
|
const staticNetFilteringEngine = this.staticNetFilteringEngine;
|
|
|
|
const staticExtFilteringEngine = this.staticExtFilteringEngine;
|
|
|
|
const reIsWhitespaceChar = /\s/;
|
|
|
|
const reMaybeLocalIp = /^[\d:f]/;
|
|
|
|
const reIsLocalhostRedirect = /\s+(?:0\.0\.0\.0|broadcasthost|localhost|local|ip6-\w+)\b/;
|
|
|
|
const reLocalIp = /^(?:0\.0\.0\.0|127\.0\.0\.1|::1|fe80::1%lo0)/;
|
|
|
|
const lineIter = new this.LineIterator(this.processDirectives(rawText));
|
2017-01-18 19:17:47 +01:00
|
|
|
|
|
|
|
while ( lineIter.eot() === false ) {
|
2014-06-24 00:42:43 +02:00
|
|
|
// rhill 2014-04-18: The trim is important here, as without it there
|
|
|
|
// could be a lingering `\r` which would cause problems in the
|
|
|
|
// following parsing code.
|
2018-10-23 19:01:08 +02:00
|
|
|
let line = lineIter.next().trim();
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( line.length === 0 ) { continue; }
|
2015-01-23 17:32:49 +01:00
|
|
|
|
2014-06-24 00:42:43 +02:00
|
|
|
// Strip comments
|
2018-12-15 16:46:17 +01:00
|
|
|
const c = line.charAt(0);
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( c === '!' || c === '[' ) { continue; }
|
2014-06-24 00:42:43 +02:00
|
|
|
|
2014-09-25 21:44:18 +02:00
|
|
|
// Parse or skip cosmetic filters
|
2015-01-23 17:32:49 +01:00
|
|
|
// All cosmetic filters are caught here
|
2017-12-28 19:49:02 +01:00
|
|
|
if ( staticExtFilteringEngine.compile(line, writer) ) { continue; }
|
2014-06-24 00:42:43 +02:00
|
|
|
|
2015-01-23 17:32:49 +01:00
|
|
|
// Whatever else is next can be assumed to not be a cosmetic filter
|
|
|
|
|
|
|
|
// Most comments start in first column
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( c === '#' ) { continue; }
|
2014-06-24 00:42:43 +02:00
|
|
|
|
2015-01-23 17:32:49 +01:00
|
|
|
// Catch comments somewhere on the line
|
|
|
|
// Remove:
|
|
|
|
// ... #blah blah blah
|
|
|
|
// ... # blah blah blah
|
|
|
|
// Don't remove:
|
|
|
|
// ...#blah blah blah
|
|
|
|
// because some ABP filters uses the `#` character (URL fragment)
|
2018-12-15 16:46:17 +01:00
|
|
|
const pos = line.indexOf('#');
|
2015-01-23 17:32:49 +01:00
|
|
|
if ( pos !== -1 && reIsWhitespaceChar.test(line.charAt(pos - 1)) ) {
|
|
|
|
line = line.slice(0, pos).trim();
|
|
|
|
}
|
|
|
|
|
2014-06-24 00:42:43 +02:00
|
|
|
// https://github.com/gorhill/httpswitchboard/issues/15
|
|
|
|
// Ensure localhost et al. don't end up in the ubiquitous blacklist.
|
2015-01-23 17:32:49 +01:00
|
|
|
// With hosts files, we need to remove local IP redirection
|
|
|
|
if ( reMaybeLocalIp.test(c) ) {
|
|
|
|
// Ignore hosts file redirect configuration
|
|
|
|
// 127.0.0.1 localhost
|
|
|
|
// 255.255.255.255 broadcasthost
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( reIsLocalhostRedirect.test(line) ) { continue; }
|
2015-01-23 17:32:49 +01:00
|
|
|
line = line.replace(reLocalIp, '').trim();
|
|
|
|
}
|
2015-03-07 19:20:18 +01:00
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( line.length === 0 ) { continue; }
|
2014-06-24 00:42:43 +02:00
|
|
|
|
2017-12-28 19:49:02 +01:00
|
|
|
staticNetFilteringEngine.compile(line, writer);
|
2015-02-24 00:31:29 +01:00
|
|
|
}
|
|
|
|
|
2017-12-28 19:49:02 +01:00
|
|
|
return writer.toString();
|
2015-02-24 00:31:29 +01:00
|
|
|
};
|
2014-06-24 00:42:43 +02:00
|
|
|
|
2015-02-24 00:31:29 +01:00
|
|
|
/******************************************************************************/
|
2014-06-24 00:42:43 +02:00
|
|
|
|
2016-02-17 15:28:20 +01:00
|
|
|
// https://github.com/gorhill/uBlock/issues/1395
|
2016-02-17 16:04:55 +01:00
|
|
|
// Added `firstparty` argument: to avoid discarding cosmetic filters when
|
2016-02-17 15:28:20 +01:00
|
|
|
// applying 1st-party filters.
|
|
|
|
|
|
|
|
µBlock.applyCompiledFilters = function(rawText, firstparty) {
|
2017-05-25 23:46:59 +02:00
|
|
|
if ( rawText === '' ) { return; }
|
2018-10-23 19:01:08 +02:00
|
|
|
let reader = new this.CompiledLineIO.Reader(rawText);
|
2017-05-25 23:46:59 +02:00
|
|
|
this.staticNetFilteringEngine.fromCompiledContent(reader);
|
2017-12-28 19:49:02 +01:00
|
|
|
this.staticExtFilteringEngine.fromCompiledContent(reader, {
|
|
|
|
skipGenericCosmetic: this.userSettings.ignoreGenericCosmeticFilters,
|
|
|
|
skipCosmetic: !firstparty && !this.userSettings.parseAllABPHideFilters
|
|
|
|
});
|
2014-06-24 00:42:43 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2018-04-05 13:29:15 +02:00
|
|
|
// https://github.com/AdguardTeam/AdguardBrowserExtension/issues/917
|
|
|
|
|
|
|
|
µBlock.processDirectives = function(content) {
|
2019-03-16 14:00:31 +01:00
|
|
|
const reIf = /^!#(if|endif)\b([^\n]*)/gm;
|
2019-05-18 16:31:04 +02:00
|
|
|
const stack = [];
|
|
|
|
const shouldDiscard = ( ) => stack.some(v => v);
|
2019-03-16 14:00:31 +01:00
|
|
|
const parts = [];
|
2019-05-18 16:31:04 +02:00
|
|
|
let beg = 0, discard = false;
|
|
|
|
|
2018-04-05 13:29:15 +02:00
|
|
|
while ( beg < content.length ) {
|
2019-03-16 14:00:31 +01:00
|
|
|
const match = reIf.exec(content);
|
2018-04-05 13:29:15 +02:00
|
|
|
if ( match === null ) { break; }
|
2019-05-18 16:31:04 +02:00
|
|
|
|
|
|
|
switch ( match[1] ) {
|
|
|
|
case 'if':
|
2019-03-16 14:00:31 +01:00
|
|
|
let expr = match[2].trim();
|
2019-05-18 16:31:04 +02:00
|
|
|
const target = expr.charCodeAt(0) === 0x21 /* '!' */;
|
2018-04-05 13:29:15 +02:00
|
|
|
if ( target ) { expr = expr.slice(1); }
|
2019-03-16 14:00:31 +01:00
|
|
|
const token = this.processDirectives.tokens.get(expr);
|
2019-05-18 16:31:04 +02:00
|
|
|
const startDiscard =
|
2018-04-05 13:29:15 +02:00
|
|
|
token !== undefined &&
|
2019-05-18 16:31:04 +02:00
|
|
|
vAPI.webextFlavor.soup.has(token) === target;
|
|
|
|
if ( discard === false && startDiscard ) {
|
2018-04-05 13:29:15 +02:00
|
|
|
parts.push(content.slice(beg, match.index));
|
|
|
|
discard = true;
|
|
|
|
}
|
2019-05-18 16:31:04 +02:00
|
|
|
stack.push(startDiscard);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'endif':
|
|
|
|
stack.pop();
|
|
|
|
const stopDiscard = shouldDiscard() === false;
|
|
|
|
if ( discard && stopDiscard ) {
|
|
|
|
beg = match.index + match[0].length + 1;
|
|
|
|
discard = false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
2018-04-05 13:29:15 +02:00
|
|
|
}
|
|
|
|
}
|
2019-05-18 16:31:04 +02:00
|
|
|
|
|
|
|
if ( stack.length === 0 && parts.length !== 0 ) {
|
2018-04-05 13:29:15 +02:00
|
|
|
parts.push(content.slice(beg));
|
|
|
|
content = parts.join('\n');
|
|
|
|
}
|
|
|
|
return content.trim();
|
|
|
|
};
|
|
|
|
|
|
|
|
µBlock.processDirectives.tokens = new Map([
|
2018-04-11 12:34:13 +02:00
|
|
|
[ 'ext_ublock', 'ublock' ],
|
|
|
|
[ 'env_chromium', 'chromium' ],
|
|
|
|
[ 'env_edge', 'edge' ],
|
|
|
|
[ 'env_firefox', 'firefox' ],
|
|
|
|
[ 'env_mobile', 'mobile' ],
|
|
|
|
[ 'env_safari', 'safari' ],
|
|
|
|
[ 'cap_html_filtering', 'html_filtering' ],
|
|
|
|
[ 'cap_user_stylesheet', 'user_stylesheet' ]
|
2018-04-05 13:29:15 +02:00
|
|
|
]);
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2019-04-20 23:16:49 +02:00
|
|
|
µBlock.loadRedirectResources = function() {
|
|
|
|
return this.redirectEngine.resourcesFromSelfie().then(success => {
|
|
|
|
if ( success === true ) { return; }
|
2015-11-23 13:52:50 +01:00
|
|
|
|
2019-04-20 23:16:49 +02:00
|
|
|
const fetchPromises = [ this.assets.get('ublock-resources') ];
|
2017-03-05 18:54:47 +01:00
|
|
|
|
2019-04-20 23:16:49 +02:00
|
|
|
const userResourcesLocation = this.hiddenSettings.userResourcesLocation;
|
|
|
|
if ( userResourcesLocation !== 'unset' ) {
|
|
|
|
for ( const url of userResourcesLocation.split(/\s+/) ) {
|
|
|
|
fetchPromises.push(this.assets.fetchText(url));
|
|
|
|
}
|
2017-03-05 18:54:47 +01:00
|
|
|
}
|
|
|
|
|
2019-04-20 23:16:49 +02:00
|
|
|
return Promise.all(fetchPromises);
|
|
|
|
}).then(results => {
|
|
|
|
if ( Array.isArray(results) === false ) { return; }
|
2015-11-23 13:52:50 +01:00
|
|
|
|
2019-04-20 23:16:49 +02:00
|
|
|
let content = '';
|
2018-02-15 23:25:38 +01:00
|
|
|
|
2019-04-20 23:16:49 +02:00
|
|
|
for ( const result of results ) {
|
|
|
|
if (
|
|
|
|
result instanceof Object === false ||
|
|
|
|
typeof result.content !== 'string' ||
|
|
|
|
result.content === ''
|
|
|
|
) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
content += '\n\n' + result.content;
|
2018-02-15 23:25:38 +01:00
|
|
|
}
|
2019-04-20 23:16:49 +02:00
|
|
|
|
|
|
|
this.redirectEngine.resourcesFromString(content);
|
2019-02-14 19:33:55 +01:00
|
|
|
});
|
2015-11-23 13:52:50 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2018-11-03 12:58:46 +01:00
|
|
|
µBlock.loadPublicSuffixList = function() {
|
2019-02-10 18:19:05 +01:00
|
|
|
if ( this.hiddenSettings.disableWebAssembly === false ) {
|
|
|
|
publicSuffixList.enableWASM();
|
|
|
|
}
|
|
|
|
|
2019-02-14 19:33:55 +01:00
|
|
|
return this.assets.get(
|
|
|
|
'compiled/' + this.pslAssetKey
|
|
|
|
).then(details =>
|
2019-04-20 15:06:54 +02:00
|
|
|
publicSuffixList.fromSelfie(details.content, µBlock.base64)
|
2019-03-16 14:00:31 +01:00
|
|
|
).catch(reason => {
|
|
|
|
console.info(reason);
|
|
|
|
return false;
|
|
|
|
}).then(success => {
|
|
|
|
if ( success ) { return; }
|
2019-02-14 19:33:55 +01:00
|
|
|
return this.assets.get(this.pslAssetKey, details => {
|
2018-11-03 12:58:46 +01:00
|
|
|
if ( details.content !== '' ) {
|
|
|
|
this.compilePublicSuffixList(details.content);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
2017-01-18 19:17:47 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
µBlock.compilePublicSuffixList = function(content) {
|
|
|
|
publicSuffixList.parse(content, punycode.toASCII);
|
|
|
|
this.assets.put(
|
|
|
|
'compiled/' + this.pslAssetKey,
|
2019-04-20 15:06:54 +02:00
|
|
|
publicSuffixList.toSelfie(µBlock.base64)
|
2017-01-18 19:17:47 +01:00
|
|
|
);
|
2014-06-24 00:42:43 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2014-09-08 23:46:58 +02:00
|
|
|
// This is to be sure the selfie is generated in a sane manner: the selfie will
|
2014-10-17 21:44:19 +02:00
|
|
|
// be generated if the user doesn't change his filter lists selection for
|
2014-09-08 23:46:58 +02:00
|
|
|
// some set time.
|
2014-09-25 21:44:18 +02:00
|
|
|
|
2015-11-29 23:06:58 +01:00
|
|
|
µBlock.selfieManager = (function() {
|
2019-02-14 19:33:55 +01:00
|
|
|
const µb = µBlock;
|
|
|
|
let timer;
|
2014-09-08 23:46:58 +02:00
|
|
|
|
2018-06-01 13:54:31 +02:00
|
|
|
// As of 2018-05-31:
|
2019-02-14 19:33:55 +01:00
|
|
|
// JSON.stringify-ing ourselves results in a better baseline
|
|
|
|
// memory usage at selfie-load time. For some reasons.
|
|
|
|
|
|
|
|
const create = function() {
|
|
|
|
Promise.all([
|
|
|
|
µb.assets.put(
|
|
|
|
'selfie/main',
|
|
|
|
JSON.stringify({
|
|
|
|
magic: µb.systemSettings.selfieMagic,
|
|
|
|
availableFilterLists: µb.availableFilterLists,
|
|
|
|
})
|
|
|
|
),
|
|
|
|
µb.redirectEngine.toSelfie('selfie/redirectEngine'),
|
|
|
|
µb.staticExtFilteringEngine.toSelfie('selfie/staticExtFilteringEngine'),
|
|
|
|
µb.staticNetFilteringEngine.toSelfie('selfie/staticNetFilteringEngine'),
|
|
|
|
]).then(( ) => {
|
|
|
|
µb.lz4Codec.relinquish();
|
2018-08-06 18:34:41 +02:00
|
|
|
});
|
2018-06-01 13:54:31 +02:00
|
|
|
};
|
2015-11-29 23:06:58 +01:00
|
|
|
|
2019-02-14 19:33:55 +01:00
|
|
|
const load = function() {
|
|
|
|
return Promise.all([
|
|
|
|
µb.assets.get('selfie/main').then(details => {
|
|
|
|
if (
|
|
|
|
details instanceof Object === false ||
|
|
|
|
typeof details.content !== 'string' ||
|
|
|
|
details.content === ''
|
|
|
|
) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
let selfie;
|
|
|
|
try {
|
|
|
|
selfie = JSON.parse(details.content);
|
|
|
|
} catch(ex) {
|
|
|
|
}
|
|
|
|
if (
|
|
|
|
selfie instanceof Object === false ||
|
|
|
|
selfie.magic !== µb.systemSettings.selfieMagic
|
|
|
|
) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
µb.availableFilterLists = selfie.availableFilterLists;
|
|
|
|
return true;
|
|
|
|
}),
|
|
|
|
µb.redirectEngine.fromSelfie('selfie/redirectEngine'),
|
|
|
|
µb.staticExtFilteringEngine.fromSelfie('selfie/staticExtFilteringEngine'),
|
|
|
|
µb.staticNetFilteringEngine.fromSelfie('selfie/staticNetFilteringEngine'),
|
|
|
|
]).then(results =>
|
|
|
|
results.reduce((acc, v) => acc && v, true)
|
|
|
|
).catch(reason => {
|
|
|
|
log.info(reason);
|
|
|
|
return false;
|
2018-04-06 22:02:35 +02:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2019-02-14 19:33:55 +01:00
|
|
|
const destroy = function() {
|
|
|
|
if ( timer !== undefined ) {
|
2015-11-29 23:06:58 +01:00
|
|
|
clearTimeout(timer);
|
2019-02-14 19:33:55 +01:00
|
|
|
timer = undefined;
|
2015-11-29 23:06:58 +01:00
|
|
|
}
|
2019-02-14 19:33:55 +01:00
|
|
|
µb.cacheStorage.remove('selfie'); // TODO: obsolete, remove eventually.
|
|
|
|
µb.assets.remove(/^selfie\//);
|
|
|
|
timer = vAPI.setTimeout(( ) => {
|
|
|
|
timer = undefined;
|
|
|
|
create();
|
|
|
|
}, µb.hiddenSettings.selfieAfter * 60000);
|
2018-06-01 13:54:31 +02:00
|
|
|
};
|
2015-11-29 23:06:58 +01:00
|
|
|
|
|
|
|
return {
|
2018-04-06 22:02:35 +02:00
|
|
|
load: load,
|
2015-11-29 23:06:58 +01:00
|
|
|
destroy: destroy
|
|
|
|
};
|
|
|
|
})();
|
2014-09-08 23:46:58 +02:00
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2015-07-27 16:10:34 +02:00
|
|
|
// https://github.com/gorhill/uBlock/issues/531
|
|
|
|
// Overwrite user settings with admin settings if present.
|
|
|
|
//
|
|
|
|
// Admin settings match layout of a uBlock backup. Not all data is
|
|
|
|
// necessarily present, i.e. administrators may removed entries which
|
|
|
|
// values are left to the user's choice.
|
|
|
|
|
2019-02-17 21:40:09 +01:00
|
|
|
µBlock.restoreAdminSettings = function() {
|
|
|
|
return new Promise(resolve => {
|
|
|
|
// >>>> start of executor
|
|
|
|
|
2016-10-15 15:32:33 +02:00
|
|
|
if ( vAPI.adminStorage instanceof Object === false ) {
|
2019-02-17 21:40:09 +01:00
|
|
|
return resolve();
|
2016-10-15 15:32:33 +02:00
|
|
|
}
|
|
|
|
|
2019-02-17 21:40:09 +01:00
|
|
|
vAPI.adminStorage.getItem('adminSettings', json => {
|
|
|
|
let data;
|
2015-10-21 17:53:03 +02:00
|
|
|
if ( typeof json === 'string' && json !== '' ) {
|
|
|
|
try {
|
|
|
|
data = JSON.parse(json);
|
|
|
|
} catch (ex) {
|
|
|
|
console.error(ex);
|
|
|
|
}
|
2015-07-27 16:10:34 +02:00
|
|
|
}
|
|
|
|
|
2019-02-17 21:40:09 +01:00
|
|
|
if ( data instanceof Object === false ) {
|
|
|
|
return resolve();
|
2015-10-21 17:53:03 +02:00
|
|
|
}
|
2015-07-27 16:10:34 +02:00
|
|
|
|
2019-02-17 21:40:09 +01:00
|
|
|
const bin = {};
|
|
|
|
let binNotEmpty = false;
|
2015-07-27 16:10:34 +02:00
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
// Allows an admin to set their own 'assets.json' file, with their own
|
|
|
|
// set of stock assets.
|
|
|
|
if ( typeof data.assetsBootstrapLocation === 'string' ) {
|
|
|
|
bin.assetsBootstrapLocation = data.assetsBootstrapLocation;
|
|
|
|
binNotEmpty = true;
|
|
|
|
}
|
|
|
|
|
2015-10-21 17:53:03 +02:00
|
|
|
if ( typeof data.userSettings === 'object' ) {
|
2019-02-17 21:40:09 +01:00
|
|
|
for ( const name in this.userSettings ) {
|
|
|
|
if ( this.userSettings.hasOwnProperty(name) === false ) {
|
2015-10-21 17:53:03 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if ( data.userSettings.hasOwnProperty(name) === false ) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
bin[name] = data.userSettings[name];
|
|
|
|
binNotEmpty = true;
|
2015-07-27 16:10:34 +02:00
|
|
|
}
|
2015-10-21 17:53:03 +02:00
|
|
|
}
|
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
// 'selectedFilterLists' is an array of filter list tokens. Each token
|
|
|
|
// is a reference to an asset in 'assets.json'.
|
|
|
|
if ( Array.isArray(data.selectedFilterLists) ) {
|
|
|
|
bin.selectedFilterLists = data.selectedFilterLists;
|
|
|
|
binNotEmpty = true;
|
2015-07-27 16:10:34 +02:00
|
|
|
}
|
|
|
|
|
2019-06-25 17:57:14 +02:00
|
|
|
if ( Array.isArray(data.whitelist) ) {
|
|
|
|
bin.netWhitelist = data.whitelist;
|
|
|
|
binNotEmpty = true;
|
|
|
|
} else if ( typeof data.netWhitelist === 'string' ) {
|
|
|
|
bin.netWhitelist = data.netWhitelist.split('\n');
|
2015-10-21 17:53:03 +02:00
|
|
|
binNotEmpty = true;
|
|
|
|
}
|
2015-07-27 16:10:34 +02:00
|
|
|
|
2015-10-21 17:53:03 +02:00
|
|
|
if ( typeof data.dynamicFilteringString === 'string' ) {
|
|
|
|
bin.dynamicFilteringString = data.dynamicFilteringString;
|
|
|
|
binNotEmpty = true;
|
|
|
|
}
|
2015-07-27 16:10:34 +02:00
|
|
|
|
2015-10-21 17:53:03 +02:00
|
|
|
if ( typeof data.urlFilteringString === 'string' ) {
|
|
|
|
bin.urlFilteringString = data.urlFilteringString;
|
|
|
|
binNotEmpty = true;
|
|
|
|
}
|
2015-07-27 16:10:34 +02:00
|
|
|
|
2015-10-21 17:53:03 +02:00
|
|
|
if ( typeof data.hostnameSwitchesString === 'string' ) {
|
|
|
|
bin.hostnameSwitchesString = data.hostnameSwitchesString;
|
|
|
|
binNotEmpty = true;
|
|
|
|
}
|
2015-07-27 16:10:34 +02:00
|
|
|
|
2015-10-21 17:53:03 +02:00
|
|
|
if ( binNotEmpty ) {
|
|
|
|
vAPI.storage.set(bin);
|
|
|
|
}
|
2015-07-27 16:10:34 +02:00
|
|
|
|
2015-10-21 17:53:03 +02:00
|
|
|
if ( typeof data.userFilters === 'string' ) {
|
2019-02-17 21:40:09 +01:00
|
|
|
this.assets.put(this.userFiltersPath, data.userFilters);
|
2015-10-21 17:53:03 +02:00
|
|
|
}
|
2015-07-27 16:10:34 +02:00
|
|
|
|
2019-02-17 21:40:09 +01:00
|
|
|
resolve();
|
|
|
|
});
|
2015-10-21 17:53:03 +02:00
|
|
|
|
2019-02-17 21:40:09 +01:00
|
|
|
// <<<< end of executor
|
|
|
|
});
|
2015-07-27 16:10:34 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2017-11-09 18:53:05 +01:00
|
|
|
// https://github.com/gorhill/uBlock/issues/2344
|
|
|
|
// Support mutliple locales per filter list.
|
|
|
|
|
|
|
|
// https://github.com/gorhill/uBlock/issues/3210
|
|
|
|
// Support ability to auto-enable a filter list based on user agent.
|
|
|
|
|
|
|
|
µBlock.listMatchesEnvironment = function(details) {
|
|
|
|
// Matches language?
|
|
|
|
if ( typeof details.lang === 'string' ) {
|
2019-04-15 00:20:57 +02:00
|
|
|
let re = this.listMatchesEnvironment.reLang;
|
2017-11-15 22:32:52 +01:00
|
|
|
if ( re === undefined ) {
|
2019-04-15 00:20:57 +02:00
|
|
|
const match = /^[a-z]+/.exec(self.navigator.language);
|
|
|
|
if ( match !== null ) {
|
|
|
|
re = new RegExp('\\b' + match[0] + '\\b');
|
|
|
|
this.listMatchesEnvironment.reLang = re;
|
|
|
|
}
|
2017-11-09 18:53:05 +01:00
|
|
|
}
|
2019-04-15 00:20:57 +02:00
|
|
|
if ( re !== undefined && re.test(details.lang) ) { return true; }
|
2017-11-09 18:53:05 +01:00
|
|
|
}
|
|
|
|
// Matches user agent?
|
|
|
|
if ( typeof details.ua === 'string' ) {
|
2019-04-15 00:20:57 +02:00
|
|
|
let re = new RegExp('\\b' + this.escapeRegex(details.ua) + '\\b', 'i');
|
2017-11-09 18:53:05 +01:00
|
|
|
if ( re.test(self.navigator.userAgent) ) { return true; }
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
µBlock.scheduleAssetUpdater = (function() {
|
2019-05-15 20:49:12 +02:00
|
|
|
let timer, next = 0;
|
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
return function(updateDelay) {
|
|
|
|
if ( timer ) {
|
|
|
|
clearTimeout(timer);
|
|
|
|
timer = undefined;
|
2014-09-08 23:46:58 +02:00
|
|
|
}
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( updateDelay === 0 ) {
|
|
|
|
next = 0;
|
2015-02-24 00:31:29 +01:00
|
|
|
return;
|
|
|
|
}
|
2019-05-15 20:49:12 +02:00
|
|
|
const now = Date.now();
|
2017-01-18 19:17:47 +01:00
|
|
|
// Use the new schedule if and only if it is earlier than the previous
|
|
|
|
// one.
|
|
|
|
if ( next !== 0 ) {
|
|
|
|
updateDelay = Math.min(updateDelay, Math.max(next - now, 0));
|
2015-02-24 00:31:29 +01:00
|
|
|
}
|
2017-01-18 19:17:47 +01:00
|
|
|
next = now + updateDelay;
|
2019-05-15 20:49:12 +02:00
|
|
|
timer = vAPI.setTimeout(( ) => {
|
2017-01-18 19:17:47 +01:00
|
|
|
timer = undefined;
|
|
|
|
next = 0;
|
2019-05-15 20:49:12 +02:00
|
|
|
this.assets.updateStart({
|
|
|
|
delay: this.hiddenSettings.autoUpdateAssetFetchPeriod * 1000 ||
|
|
|
|
120000
|
2017-01-18 19:17:47 +01:00
|
|
|
});
|
|
|
|
}, updateDelay);
|
2015-02-24 00:31:29 +01:00
|
|
|
};
|
|
|
|
})();
|
2015-08-18 19:15:58 +02:00
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
µBlock.assetObserver = function(topic, details) {
|
|
|
|
// Do not update filter list if not in use.
|
|
|
|
if ( topic === 'before-asset-updated' ) {
|
2017-05-08 20:00:41 +02:00
|
|
|
if ( details.type === 'filters' ) {
|
|
|
|
if (
|
|
|
|
this.availableFilterLists.hasOwnProperty(details.assetKey) === false ||
|
|
|
|
this.selectedFilterLists.indexOf(details.assetKey) === -1
|
|
|
|
) {
|
2017-12-28 19:49:02 +01:00
|
|
|
return;
|
2017-05-08 20:00:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// https://github.com/gorhill/uBlock/issues/2594
|
|
|
|
if ( details.assetKey === 'ublock-resources' ) {
|
|
|
|
if (
|
|
|
|
this.hiddenSettings.ignoreRedirectFilters === true &&
|
|
|
|
this.hiddenSettings.ignoreScriptInjectFilters === true
|
|
|
|
) {
|
2017-12-28 19:49:02 +01:00
|
|
|
return;
|
2017-05-08 20:00:41 +02:00
|
|
|
}
|
2015-08-18 19:15:58 +02:00
|
|
|
}
|
2017-12-28 19:49:02 +01:00
|
|
|
return true;
|
2015-08-18 19:15:58 +02:00
|
|
|
}
|
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
// Compile the list while we have the raw version in memory
|
|
|
|
if ( topic === 'after-asset-updated' ) {
|
2019-02-14 19:33:55 +01:00
|
|
|
// Skip selfie-related content.
|
|
|
|
if ( details.assetKey.startsWith('selfie/') ) { return; }
|
2019-05-15 20:49:12 +02:00
|
|
|
const cached = typeof details.content === 'string' &&
|
|
|
|
details.content !== '';
|
2017-01-18 19:17:47 +01:00
|
|
|
if ( this.availableFilterLists.hasOwnProperty(details.assetKey) ) {
|
|
|
|
if ( cached ) {
|
2017-01-22 22:05:16 +01:00
|
|
|
if ( this.selectedFilterLists.indexOf(details.assetKey) !== -1 ) {
|
2017-01-18 19:17:47 +01:00
|
|
|
this.extractFilterListMetadata(
|
|
|
|
details.assetKey,
|
|
|
|
details.content
|
|
|
|
);
|
|
|
|
this.assets.put(
|
|
|
|
'compiled/' + details.assetKey,
|
2018-12-15 16:46:17 +01:00
|
|
|
this.compileFilters(
|
|
|
|
details.content,
|
|
|
|
{ assetKey: details.assetKey }
|
|
|
|
)
|
2017-01-18 19:17:47 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
this.removeCompiledFilterList(details.assetKey);
|
2015-09-13 16:26:36 +02:00
|
|
|
}
|
2017-01-18 19:17:47 +01:00
|
|
|
} else if ( details.assetKey === this.pslAssetKey ) {
|
|
|
|
if ( cached ) {
|
|
|
|
this.compilePublicSuffixList(details.content);
|
2015-08-18 19:15:58 +02:00
|
|
|
}
|
2017-01-18 19:17:47 +01:00
|
|
|
} else if ( details.assetKey === 'ublock-resources' ) {
|
2018-02-15 23:25:38 +01:00
|
|
|
this.redirectEngine.invalidateResourcesSelfie();
|
2015-08-18 19:15:58 +02:00
|
|
|
}
|
2017-01-18 19:17:47 +01:00
|
|
|
vAPI.messaging.broadcast({
|
|
|
|
what: 'assetUpdated',
|
|
|
|
key: details.assetKey,
|
|
|
|
cached: cached
|
2015-08-18 19:15:58 +02:00
|
|
|
});
|
2017-05-06 19:19:05 +02:00
|
|
|
// https://github.com/gorhill/uBlock/issues/2585
|
2019-02-14 19:33:55 +01:00
|
|
|
// Whenever an asset is overwritten, the current selfie is quite
|
|
|
|
// likely no longer valid.
|
2017-05-06 19:19:05 +02:00
|
|
|
this.selfieManager.destroy();
|
2017-01-18 19:17:47 +01:00
|
|
|
return;
|
|
|
|
}
|
2015-08-18 19:15:58 +02:00
|
|
|
|
2017-01-22 22:05:16 +01:00
|
|
|
// Update failed.
|
|
|
|
if ( topic === 'asset-update-failed' ) {
|
|
|
|
vAPI.messaging.broadcast({
|
|
|
|
what: 'assetUpdated',
|
|
|
|
key: details.assetKey,
|
|
|
|
failed: true
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-01-18 19:17:47 +01:00
|
|
|
// Reload all filter lists if needed.
|
|
|
|
if ( topic === 'after-assets-updated' ) {
|
|
|
|
if ( details.assetKeys.length !== 0 ) {
|
|
|
|
this.loadFilterLists();
|
|
|
|
}
|
|
|
|
if ( this.userSettings.autoUpdate ) {
|
2017-02-05 13:43:28 +01:00
|
|
|
this.scheduleAssetUpdater(this.hiddenSettings.autoUpdatePeriod * 3600000 || 25200000);
|
2017-01-18 19:17:47 +01:00
|
|
|
} else {
|
|
|
|
this.scheduleAssetUpdater(0);
|
|
|
|
}
|
2017-01-23 15:35:05 +01:00
|
|
|
vAPI.messaging.broadcast({
|
|
|
|
what: 'assetsUpdated',
|
|
|
|
assetKeys: details.assetKeys
|
|
|
|
});
|
2017-01-18 19:17:47 +01:00
|
|
|
return;
|
|
|
|
}
|
2017-01-20 21:17:11 +01:00
|
|
|
|
|
|
|
// New asset source became available, if it's a filter list, should we
|
|
|
|
// auto-select it?
|
|
|
|
if ( topic === 'builtin-asset-source-added' ) {
|
|
|
|
if ( details.entry.content === 'filters' ) {
|
|
|
|
if (
|
|
|
|
details.entry.off !== true ||
|
2017-11-09 18:53:05 +01:00
|
|
|
this.listMatchesEnvironment(details.entry)
|
2017-01-20 21:17:11 +01:00
|
|
|
) {
|
|
|
|
this.saveSelectedFilterLists([ details.assetKey ], true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2015-08-18 19:15:58 +02:00
|
|
|
};
|