uBlock/src/js/ublock.js

330 lines
10 KiB
JavaScript
Raw Normal View History

2014-06-24 00:42:43 +02:00
/*******************************************************************************
µBlock - a Chromium browser extension to block requests.
Copyright (C) 2014 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
*/
2014-12-12 23:59:47 +01:00
/* global vAPI, µBlock */
/******************************************************************************/
(function(){
'use strict';
2014-06-24 00:42:43 +02:00
/******************************************************************************/
var µb = µBlock;
2014-08-02 17:40:27 +02:00
/******************************************************************************/
2014-08-27 08:01:10 +02:00
// https://github.com/gorhill/uBlock/issues/405
// Be more flexible with whitelist syntax
var matchWhitelistException = function(url, exception) {
// Exception is a plain hostname
if ( exception.indexOf('/') === -1 ) {
return µb.URI.hostnameFromURI(url).slice(-exception.length) === exception;
}
// Match URL exactly
if ( exception.indexOf('*') === -1 ) {
return url === exception;
2014-06-24 00:42:43 +02:00
}
// Regex escape code inspired from:
// "Is there a RegExp.escape function in Javascript?"
// http://stackoverflow.com/a/3561711
var reStr = exception.replace(whitelistDirectiveEscape, '\\$&')
.replace(whitelistDirectiveEscapeAsterisk, '.*');
var re = new RegExp(reStr);
return re.test(url);
};
2014-08-02 17:40:27 +02:00
// Any special regexp char will be escaped
var whitelistDirectiveEscape = /[-\/\\^$+?.()|[\]{}]/g;
// All `*` will be expanded into `.*`
var whitelistDirectiveEscapeAsterisk = /\*/g;
// Probably manually entered whitelist directive
var isHandcraftedWhitelistDirective = function(directive) {
if ( directive.indexOf('/') === -1 ) {
return false;
2014-08-02 17:40:27 +02:00
}
return directive.indexOf('*') !== -1 || directive.slice(0, 4) !== 'http';
};
/******************************************************************************/
2014-08-02 17:40:27 +02:00
µBlock.getNetFilteringSwitch = function(url) {
var buckets, i;
var hostname = this.URI.hostnameFromURI(url);
var pos = url.indexOf('#');
url = pos !== -1 ? url.slice(0, pos) : url;
var netWhitelist = this.netWhitelist;
for (;;) {
if ( netWhitelist.hasOwnProperty(hostname) ) {
buckets = netWhitelist[hostname];
i = buckets.length;
while ( i-- ) {
if ( matchWhitelistException(url, buckets[i]) ) {
// console.log('"%s" matche url "%s"', buckets[i], keyURL);
return false;
}
}
}
pos = hostname.indexOf('.');
if ( pos === -1 ) {
break;
2014-06-24 00:42:43 +02:00
}
hostname = hostname.slice(pos + 1);
2014-06-24 00:42:43 +02:00
}
return true;
};
/******************************************************************************/
2014-08-02 17:40:27 +02:00
µBlock.toggleNetFilteringSwitch = function(url, scope, newState) {
var currentState = this.getNetFilteringSwitch(url);
2014-06-24 00:42:43 +02:00
if ( newState === undefined ) {
newState = !currentState;
}
if ( newState === currentState ) {
return currentState;
}
2014-08-02 17:40:27 +02:00
var hostname = this.URI.hostnameFromURI(url);
var pos = url.indexOf('#');
url = pos !== -1 ? url.slice(0, pos) : url;
var directive = scope === 'page' ? url : hostname;
2014-08-02 17:40:27 +02:00
var netWhitelist = this.netWhitelist;
var buckets;
2014-06-24 00:42:43 +02:00
// Add to exception list
if ( newState === false ) {
if ( netWhitelist.hasOwnProperty(hostname) === false ) {
buckets = netWhitelist[hostname] = [];
}
buckets.push(directive);
2014-08-02 17:40:27 +02:00
this.saveWhitelist();
2014-06-24 00:42:43 +02:00
return true;
}
2014-12-08 18:37:35 +01:00
// Remove from exception list whatever causes current URL to be whitelisted
var i;
for (;;) {
if ( netWhitelist.hasOwnProperty(hostname) ) {
buckets = netWhitelist[hostname];
i = buckets.length;
while ( i-- ) {
directive = buckets[i];
if ( !matchWhitelistException(url, directive) ) {
continue;
}
buckets.splice(i, 1);
// If it is a directive which can't be created easily through
// the user interface, keep it around as a commented out
// directive
if ( isHandcraftedWhitelistDirective(directive) ) {
netWhitelist['#'].push('# ' + directive);
}
}
if ( buckets.length === 0 ) {
delete netWhitelist[hostname];
}
2014-06-24 00:42:43 +02:00
}
pos = hostname.indexOf('.');
if ( pos === -1 ) {
break;
}
hostname = hostname.slice(pos + 1);
2014-08-02 17:40:27 +02:00
}
this.saveWhitelist();
return true;
2014-06-24 00:42:43 +02:00
};
/******************************************************************************/
2014-12-08 18:37:35 +01:00
2014-06-24 01:23:36 +02:00
// For now we will use the net exception list
2014-08-02 17:40:27 +02:00
µBlock.getCosmeticFilteringSwitch = function(url, domain) {
return this.getNetFilteringSwitch(url, domain);
};
/******************************************************************************/
µBlock.stringFromWhitelist = function(whitelist) {
2014-08-02 17:40:27 +02:00
var r = {};
var i, bucket;
for ( var key in whitelist ) {
if ( whitelist.hasOwnProperty(key) === false ) {
2014-08-02 17:40:27 +02:00
continue;
}
bucket = whitelist[key];
i = bucket.length;
while ( i-- ) {
2014-08-02 17:40:27 +02:00
r[bucket[i]] = true;
2014-06-24 01:23:36 +02:00
}
}
2014-08-02 17:40:27 +02:00
return Object.keys(r).sort(function(a,b){return a.localeCompare(b);}).join('\n');
2014-06-24 01:23:36 +02:00
};
/******************************************************************************/
2014-08-02 17:40:27 +02:00
µBlock.whitelistFromString = function(s) {
var whitelist = {
'#': []
};
var reInvalidHostname = /[^a-z0-9.\-\[\]:]/;
var reHostnameExtractor = /([a-z0-9\[][a-z0-9.\-:]*[a-z0-9\]])\/(?:[^\x00-\x20\/]|$)[^\x00-\x20]*$/;
2014-08-02 17:40:27 +02:00
var lines = s.split(/[\n\r]+/);
var line, matches, key, directive;
2014-08-02 17:40:27 +02:00
for ( var i = 0; i < lines.length; i++ ) {
line = lines[i].trim();
// Don't throw out commented out lines: user might want to fix them
if ( line.charAt(0) === '#' ) {
key = '#';
directive = line;
}
// Plain hostname
else if ( line.indexOf('/') === -1 ) {
if ( reInvalidHostname.test(line) ) {
key = '#';
directive = '# ' + line;
} else {
key = directive = line;
}
2014-08-02 17:40:27 +02:00
}
// URL, possibly wildcarded: there MUST be at least one hostname
// label (or else it would be just impossible to make an efficient
// dict.
else {
matches = reHostnameExtractor.exec(line);
if ( !matches || matches.length !== 2 ) {
key = '#';
directive = '# ' + line;
} else {
key = matches[1];
directive = line;
}
}
// Be sure this stays fixed:
// https://github.com/gorhill/uBlock/issues/185
if ( whitelist.hasOwnProperty(key) === false ) {
whitelist[key] = [];
2014-08-02 17:40:27 +02:00
}
whitelist[key].push(directive);
2014-08-02 17:40:27 +02:00
}
return whitelist;
2014-06-24 00:42:43 +02:00
};
/******************************************************************************/
2014-06-25 00:29:55 +02:00
// Return all settings if none specified.
2014-06-24 00:42:43 +02:00
µBlock.changeUserSettings = function(name, value) {
2014-06-25 00:29:55 +02:00
if ( name === undefined ) {
2014-09-28 18:05:46 +02:00
return this.userSettings;
2014-06-25 00:29:55 +02:00
}
2014-06-24 00:42:43 +02:00
if ( typeof name !== 'string' || name === '' ) {
return;
}
// Do not allow an unknown user setting to be created
2014-09-28 18:05:46 +02:00
if ( this.userSettings[name] === undefined ) {
2014-06-24 00:42:43 +02:00
return;
}
if ( value === undefined ) {
2014-09-28 18:05:46 +02:00
return this.userSettings[name];
2014-06-24 00:42:43 +02:00
}
// Pre-change
switch ( name ) {
default:
break;
2014-06-24 00:42:43 +02:00
}
// Change
2014-09-28 18:05:46 +02:00
this.userSettings[name] = value;
2014-06-24 00:42:43 +02:00
// Post-change
switch ( name ) {
2014-09-28 18:05:46 +02:00
case 'contextMenuEnabled':
this.contextMenu.toggle(value);
break;
case 'experimentalEnabled':
if ( typeof this.mirrors === 'object' ) {
this.mirrors.toggle(value);
}
2014-09-28 18:05:46 +02:00
break;
default:
break;
2014-06-24 00:42:43 +02:00
}
2014-09-28 18:05:46 +02:00
this.saveUserSettings();
2014-06-24 00:42:43 +02:00
};
/******************************************************************************/
µBlock.transposeType = function(type, path) {
2014-09-20 16:44:04 +02:00
if ( type !== 'other' ) {
return type;
}
var pos = path.lastIndexOf('.');
if ( pos === -1 ) {
return type;
}
2014-09-20 16:47:30 +02:00
var ext = path.slice(pos) + '.';
2014-10-06 20:02:44 +02:00
if ( '.css.eot.ttf.otf.svg.woff.woff2.'.indexOf(ext) !== -1 ) {
2014-09-20 16:44:04 +02:00
return 'stylesheet';
}
2014-09-20 16:47:30 +02:00
if ( '.ico.png.gif.jpg.jpeg.'.indexOf(ext) !== -1 ) {
2014-09-20 16:44:04 +02:00
return 'image';
2014-06-24 00:42:43 +02:00
}
return type;
};
/******************************************************************************/
2014-09-28 18:05:46 +02:00
µBlock.elementPickerExec = function(tabId, targetElement) {
this.elementPickerTarget = targetElement || '';
vAPI.tabs.injectScript(tabId, { file: 'js/element-picker.js' });
2014-09-28 18:05:46 +02:00
};
/******************************************************************************/
2014-10-06 20:02:44 +02:00
µBlock.toggleDynamicFilter = function(details) {
var changed = false;
if ( details.block ) {
changed = this.netFilteringEngine.dynamicFilterBlock(details.hostname, details.requestType, details.firstParty);
} else {
changed = this.netFilteringEngine.dynamicFilterUnblock(details.hostname, details.requestType, details.firstParty);
}
if ( changed ) {
this.userSettings.dynamicFilteringSelfie = this.netFilteringEngine.selfieFromDynamicFilters();
2014-10-07 14:59:35 +02:00
this.XAL.keyvalSetOne('dynamicFilteringSelfie', this.userSettings.dynamicFilteringSelfie);
2014-10-06 20:02:44 +02:00
}
};
2014-12-12 23:59:47 +01:00
/******************************************************************************/
})();