uBlock/src/js/static-net-filtering.js

2272 lines
66 KiB
JavaScript
Raw Normal View History

2014-06-24 00:42:43 +02:00
/*******************************************************************************
2015-03-07 19:20:18 +01:00
µBlock - a browser extension to block requests.
2014-06-24 00:42:43 +02:00
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
*/
/* jshint bitwise: false, esnext: true, boss: true */
/* global punycode, µBlock */
2014-06-24 00:42:43 +02:00
/******************************************************************************/
µBlock.staticNetFilteringEngine = (function(){
2014-06-24 00:42:43 +02:00
2015-03-03 12:09:35 +01:00
'use strict';
2014-06-24 00:42:43 +02:00
/******************************************************************************/
2014-09-14 22:20:40 +02:00
var µb = µBlock;
2014-07-14 17:24:59 +02:00
// fedcba9876543210
2014-09-20 16:44:04 +02:00
// | | | |||
// | | | |||
// | | | |||
// | | | |||
// | | | ||+---- bit 0: [BlockAction | AllowAction]
// | | | |+---- bit 1: `important`
// | | | +---- bit 2-3: party [0 - 3]
// | | +---- bit 4-7: type [0 - 15]
// | +---- bit 8-15: unused
// +---- bit 15: never use! (to ensure valid unicode character)
2015-03-03 12:09:35 +01:00
var BlockAction = 0 << 0;
var AllowAction = 1 << 0;
var ToggleAction = BlockAction ^ AllowAction;
2014-06-24 00:42:43 +02:00
2015-03-03 12:09:35 +01:00
var Important = 1 << 1;
2014-06-24 00:42:43 +02:00
2015-03-03 12:09:35 +01:00
var AnyParty = 0 << 2;
var FirstParty = 1 << 2;
var ThirdParty = 2 << 2;
2014-06-24 00:42:43 +02:00
2015-03-26 00:28:22 +01:00
var AnyType = 0 << 4;
2014-09-21 02:06:55 +02:00
var typeNameToTypeValue = {
2015-03-26 00:28:22 +01:00
'stylesheet': 1 << 4,
'image': 2 << 4,
'object': 3 << 4,
'script': 4 << 4,
'xmlhttprequest': 5 << 4,
'sub_frame': 6 << 4,
'other': 7 << 4,
2015-01-09 03:04:48 +01:00
'cosmetic-filtering': 13 << 4,
2014-09-24 23:38:22 +02:00
'inline-script': 14 << 4,
'popup': 15 << 4
2014-09-21 02:06:55 +02:00
};
2015-01-24 14:21:14 +01:00
var typeOtherValue = typeNameToTypeValue.other;
2014-09-21 02:06:55 +02:00
// All network request types to bitmap
// bring origin to 0 (from 4 -- see typeNameToTypeValue)
// left-shift 1 by the above-calculated value
2015-03-26 00:28:22 +01:00
// subtract 1 to set all type bits
var allNetRequestTypesBitmap = (1 << (typeOtherValue >>> 4)) - 1;
2015-03-03 12:09:35 +01:00
var BlockAnyTypeAnyParty = BlockAction | AnyType | AnyParty;
var BlockAnyType = BlockAction | AnyType;
var BlockAnyParty = BlockAction | AnyParty;
2014-06-24 00:42:43 +02:00
2015-03-03 12:09:35 +01:00
var AllowAnyTypeAnyParty = AllowAction | AnyType | AnyParty;
var AllowAnyType = AllowAction | AnyType;
var AllowAnyParty = AllowAction | AnyParty;
2014-06-24 00:42:43 +02:00
var reHostnameRule = /^[0-9a-z][0-9a-z.-]*[0-9a-z]$/;
2014-09-19 16:59:44 +02:00
var reURLPostHostnameAnchors = /[\/?#]/;
2014-06-24 00:42:43 +02:00
// ABP filters: https://adblockplus.org/en/filters
// regex tester: http://regex101.com/
/******************************************************************************/
2015-02-05 00:06:31 +01:00
// See the following as short-lived registers, used during evaluation. They are
// valid until the next evaluation.
var pageHostnameRegister = '';
var requestHostnameRegister = '';
2015-03-26 00:28:22 +01:00
var filterRegister = null;
var categoryRegister = '';
2015-02-05 00:06:31 +01:00
/******************************************************************************/
var histogram = function() {};
2014-06-24 00:42:43 +02:00
/*
histogram = function(label, categories) {
2014-06-24 00:42:43 +02:00
var h = [],
categoryBucket;
for ( var k in categories ) {
// No need for hasOwnProperty() here: there is no prototype chain.
2014-06-24 00:42:43 +02:00
categoryBucket = categories[k];
for ( var kk in categoryBucket ) {
// No need for hasOwnProperty() here: there is no prototype chain.
2014-06-24 00:42:43 +02:00
filterBucket = categoryBucket[kk];
h.push({
k: k.charCodeAt(0).toString(2) + ' ' + kk,
2014-06-24 00:42:43 +02:00
n: filterBucket instanceof FilterBucket ? filterBucket.filters.length : 1
});
}
}
console.log('Histogram %s', label);
var total = h.length;
h.sort(function(a, b) { return b.n - a.n; });
// Find indices of entries of interest
var target = 2;
for ( var i = 0; i < total; i++ ) {
if ( h[i].n === target ) {
console.log('\tEntries with only %d filter(s) start at index %s (key = "%s")', target, i, h[i].k);
target -= 1;
}
}
h = h.slice(0, 50);
h.forEach(function(v) {
console.log('\tkey=%s count=%d', v.k, v.n);
});
console.log('\tTotal buckets count: %d', total);
};
*/
2014-09-08 23:46:58 +02:00
/******************************************************************************/
// Could be replaced with encodeURIComponent/decodeURIComponent,
// which seems faster on Firefox.
var encode = JSON.stringify;
var decode = JSON.parse;
var cachedParseInt = parseInt;
var atoi = function(s) {
return cachedParseInt(s, 10);
};
var isFirstParty = function(firstPartyDomain, hostname) {
if ( hostname.slice(0 - firstPartyDomain.length) !== firstPartyDomain ) {
return false;
}
// Be sure to not confuse 'example.com' with 'anotherexample.com'
var c = hostname.charAt(hostname.length - firstPartyDomain.length - 1);
return c === '.' || c === '';
};
2015-03-17 14:39:03 +01:00
var alwaysTruePseudoRegex = {
2015-03-26 20:16:48 +01:00
match: { '0': '', index: 0 },
exec: function(s) {
this.match['0'] = s;
return this.match;
},
2015-03-17 14:39:03 +01:00
test: function() {
return true;
}
};
2015-03-05 01:36:09 +01:00
var strToRegex = function(s, anchor) {
2015-03-17 14:39:03 +01:00
// https://github.com/gorhill/uBlock/issues/1038
// Special case: always match.
if ( s === '*' ) {
return alwaysTruePseudoRegex;
}
// https://developer.mozilla.org/en/docs/Web/JavaScript/Guide/Regular_Expressions
2015-03-05 01:36:09 +01:00
var reStr = s.replace(/[.+?^${}()|[\]\\]/g, '\\$&')
2015-03-02 16:41:51 +01:00
.replace(/\*/g, '.*');
2015-03-17 14:39:03 +01:00
2015-03-05 01:36:09 +01:00
if ( anchor < 0 ) {
reStr = '^' + reStr;
} else if ( anchor > 0 ) {
reStr += reStr + '$';
}
2015-03-17 14:39:03 +01:00
2015-03-05 01:36:09 +01:00
//console.debug('µBlock.staticNetFilteringEngine: created RegExp("%s")', reStr);
return new RegExp(reStr);
2015-03-02 16:41:51 +01:00
};
2014-06-24 00:42:43 +02:00
/*******************************************************************************
Filters family tree:
- plain (no wildcard)
- anywhere
- no hostname
- specific hostname
- anchored at start
- no hostname
- specific hostname
- anchored at end
- no hostname
- specific hostname
2014-09-19 16:59:44 +02:00
- anchored within hostname
- no hostname
- specific hostname (not implemented)
2014-06-24 00:42:43 +02:00
2015-03-05 01:36:09 +01:00
- with wildcard(s)
2014-09-19 16:59:44 +02:00
- anchored within hostname
2014-06-24 00:42:43 +02:00
- no hostname
- specific hostname
2015-03-05 01:36:09 +01:00
- all else
2014-06-24 00:42:43 +02:00
- no hostname
- specific hostname
*/
/******************************************************************************/
var FilterPlain = function(s, tokenBeg) {
this.s = s;
this.tokenBeg = tokenBeg;
};
FilterPlain.prototype.match = function(url, tokenBeg) {
return url.substr(tokenBeg - this.tokenBeg, this.s.length) === this.s;
};
2015-02-24 00:31:29 +01:00
FilterPlain.fid = FilterPlain.prototype.fid = 'a';
2014-09-08 23:46:58 +02:00
2014-08-28 15:59:05 +02:00
FilterPlain.prototype.toString = function() {
return this.s;
};
2014-09-08 23:46:58 +02:00
FilterPlain.prototype.toSelfie = function() {
return this.s + '\t' +
this.tokenBeg;
};
2015-02-24 00:31:29 +01:00
FilterPlain.compile = function(details) {
return details.f + '\t' + details.tokenBeg;
};
2014-09-08 23:46:58 +02:00
FilterPlain.fromSelfie = function(s) {
var pos = s.indexOf('\t');
return new FilterPlain(s.slice(0, pos), atoi(s.slice(pos + 1)));
};
2014-08-28 15:59:05 +02:00
/******************************************************************************/
2014-06-24 00:42:43 +02:00
var FilterPlainHostname = function(s, tokenBeg, hostname) {
this.s = s;
this.tokenBeg = tokenBeg;
this.hostname = hostname;
};
FilterPlainHostname.prototype.match = function(url, tokenBeg) {
2015-02-05 00:06:31 +01:00
return pageHostnameRegister.slice(-this.hostname.length) === this.hostname &&
2014-06-24 00:42:43 +02:00
url.substr(tokenBeg - this.tokenBeg, this.s.length) === this.s;
};
2015-02-24 00:31:29 +01:00
FilterPlainHostname.fid = FilterPlainHostname.prototype.fid = 'ah';
2014-09-08 23:46:58 +02:00
2014-08-28 15:59:05 +02:00
FilterPlainHostname.prototype.toString = function() {
return this.s + '$domain=' + this.hostname;
};
2014-09-08 23:46:58 +02:00
FilterPlainHostname.prototype.toSelfie = function() {
return this.s + '\t' +
this.tokenBeg + '\t' +
this.hostname;
};
2015-02-24 00:31:29 +01:00
FilterPlainHostname.compile = function(details, hostname) {
return details.f + '\t' +
details.tokenBeg + '\t' +
hostname;
};
2014-09-08 23:46:58 +02:00
FilterPlainHostname.fromSelfie = function(s) {
var args = s.split('\t');
return new FilterPlainHostname(args[0], atoi(args[1]), args[2]);
};
2014-06-24 00:42:43 +02:00
/******************************************************************************/
var FilterPlainPrefix0 = function(s) {
this.s = s;
};
FilterPlainPrefix0.prototype.match = function(url, tokenBeg) {
return url.substr(tokenBeg, this.s.length) === this.s;
};
2015-02-24 00:31:29 +01:00
FilterPlainPrefix0.fid = FilterPlainPrefix0.prototype.fid = '0a';
2014-09-08 23:46:58 +02:00
2014-08-28 15:59:05 +02:00
FilterPlainPrefix0.prototype.toString = function() {
return this.s;
};
2014-09-08 23:46:58 +02:00
FilterPlainPrefix0.prototype.toSelfie = function() {
return this.s;
};
2015-02-24 00:31:29 +01:00
FilterPlainPrefix0.compile = function(details) {
return details.f;
};
2014-09-08 23:46:58 +02:00
FilterPlainPrefix0.fromSelfie = function(s) {
return new FilterPlainPrefix0(s);
};
2014-08-28 15:59:05 +02:00
/******************************************************************************/
2014-06-24 00:42:43 +02:00
var FilterPlainPrefix0Hostname = function(s, hostname) {
this.s = s;
this.hostname = hostname;
};
FilterPlainPrefix0Hostname.prototype.match = function(url, tokenBeg) {
2015-02-05 00:06:31 +01:00
return pageHostnameRegister.slice(-this.hostname.length) === this.hostname &&
2014-06-24 00:42:43 +02:00
url.substr(tokenBeg, this.s.length) === this.s;
};
2015-02-24 00:31:29 +01:00
FilterPlainPrefix0Hostname.fid = FilterPlainPrefix0Hostname.prototype.fid = '0ah';
2014-09-08 23:46:58 +02:00
2014-08-28 15:59:05 +02:00
FilterPlainPrefix0Hostname.prototype.toString = function() {
return this.s + '$domain=' + this.hostname;
};
2014-09-08 23:46:58 +02:00
FilterPlainPrefix0Hostname.prototype.toSelfie = function() {
return this.s + '\t' +
this.hostname;
};
2015-02-24 00:31:29 +01:00
FilterPlainPrefix0Hostname.compile = function(details, hostname) {
return details.f + '\t' + hostname;
};
2014-09-08 23:46:58 +02:00
FilterPlainPrefix0Hostname.fromSelfie = function(s) {
var pos = s.indexOf('\t');
return new FilterPlainPrefix0Hostname(s.slice(0, pos), s.slice(pos + 1));
};
2014-06-24 00:42:43 +02:00
/******************************************************************************/
var FilterPlainPrefix1 = function(s) {
this.s = s;
};
FilterPlainPrefix1.prototype.match = function(url, tokenBeg) {
return url.substr(tokenBeg - 1, this.s.length) === this.s;
};
2015-02-24 00:31:29 +01:00
FilterPlainPrefix1.fid = FilterPlainPrefix1.prototype.fid = '1a';
2014-09-08 23:46:58 +02:00
2014-08-28 15:59:05 +02:00
FilterPlainPrefix1.prototype.toString = function() {
return this.s;
};
2014-09-08 23:46:58 +02:00
FilterPlainPrefix1.prototype.toSelfie = function() {
return this.s;
};
2015-02-24 00:31:29 +01:00
FilterPlainPrefix1.compile = function(details) {
return details.f;
};
2014-09-08 23:46:58 +02:00
FilterPlainPrefix1.fromSelfie = function(s) {
return new FilterPlainPrefix1(s);
};
2014-08-28 15:59:05 +02:00
/******************************************************************************/
2014-06-24 00:42:43 +02:00
var FilterPlainPrefix1Hostname = function(s, hostname) {
this.s = s;
this.hostname = hostname;
};
FilterPlainPrefix1Hostname.prototype.match = function(url, tokenBeg) {
2015-02-05 00:06:31 +01:00
return pageHostnameRegister.slice(-this.hostname.length) === this.hostname &&
2014-06-24 00:42:43 +02:00
url.substr(tokenBeg - 1, this.s.length) === this.s;
};
2015-02-24 00:31:29 +01:00
FilterPlainPrefix1Hostname.fid = FilterPlainPrefix1Hostname.prototype.fid = '1ah';
2014-09-08 23:46:58 +02:00
2014-08-28 15:59:05 +02:00
FilterPlainPrefix1Hostname.prototype.toString = function() {
return this.s + '$domain=' + this.hostname;
};
2014-09-08 23:46:58 +02:00
FilterPlainPrefix1Hostname.prototype.toSelfie = function() {
return this.s + '\t' +
this.hostname;
};
2015-02-24 00:31:29 +01:00
FilterPlainPrefix1Hostname.compile = function(details, hostname) {
return details.f + '\t' + hostname;
};
2014-09-08 23:46:58 +02:00
FilterPlainPrefix1Hostname.fromSelfie = function(s) {
var pos = s.indexOf('\t');
return new FilterPlainPrefix1Hostname(s.slice(0, pos), s.slice(pos + 1));
};
2014-06-24 00:42:43 +02:00
/******************************************************************************/
var FilterPlainLeftAnchored = function(s) {
this.s = s;
};
FilterPlainLeftAnchored.prototype.match = function(url) {
return url.slice(0, this.s.length) === this.s;
};
2015-02-24 00:31:29 +01:00
FilterPlainLeftAnchored.fid = FilterPlainLeftAnchored.prototype.fid = '|a';
2014-09-08 23:46:58 +02:00
2014-08-28 15:59:05 +02:00
FilterPlainLeftAnchored.prototype.toString = function() {
return '|' + this.s;
};
2014-09-08 23:46:58 +02:00
FilterPlainLeftAnchored.prototype.toSelfie = function() {
return this.s;
};
2015-02-24 00:31:29 +01:00
FilterPlainLeftAnchored.compile = function(details) {
return details.f;
};
2014-09-08 23:46:58 +02:00
FilterPlainLeftAnchored.fromSelfie = function(s) {
return new FilterPlainLeftAnchored(s);
};
2014-08-28 15:59:05 +02:00
/******************************************************************************/
2014-06-24 00:42:43 +02:00
var FilterPlainLeftAnchoredHostname = function(s, hostname) {
this.s = s;
this.hostname = hostname;
};
FilterPlainLeftAnchoredHostname.prototype.match = function(url) {
2015-02-05 00:06:31 +01:00
return pageHostnameRegister.slice(-this.hostname.length) === this.hostname &&
2014-06-24 00:42:43 +02:00
url.slice(0, this.s.length) === this.s;
};
2015-02-24 00:31:29 +01:00
FilterPlainLeftAnchoredHostname.fid = FilterPlainLeftAnchoredHostname.prototype.fid = '|ah';
2014-09-08 23:46:58 +02:00
2014-08-28 15:59:05 +02:00
FilterPlainLeftAnchoredHostname.prototype.toString = function() {
return '|' + this.s + '$domain=' + this.hostname;
};
2014-09-08 23:46:58 +02:00
FilterPlainLeftAnchoredHostname.prototype.toSelfie = function() {
return this.s + '\t' +
this.hostname;
};
2015-02-24 00:31:29 +01:00
FilterPlainLeftAnchoredHostname.compile = function(details, hostname) {
return details.f + '\t' + hostname;
};
2014-09-08 23:46:58 +02:00
FilterPlainLeftAnchoredHostname.fromSelfie = function(s) {
var pos = s.indexOf('\t');
return new FilterPlainLeftAnchoredHostname(s.slice(0, pos), s.slice(pos + 1));
};
2014-06-24 00:42:43 +02:00
/******************************************************************************/
var FilterPlainRightAnchored = function(s) {
this.s = s;
};
FilterPlainRightAnchored.prototype.match = function(url) {
return url.slice(-this.s.length) === this.s;
};
2015-02-24 00:31:29 +01:00
FilterPlainRightAnchored.fid = FilterPlainRightAnchored.prototype.fid = 'a|';
2014-09-08 23:46:58 +02:00
2014-08-28 15:59:05 +02:00
FilterPlainRightAnchored.prototype.toString = function() {
return this.s + '|';
};
2014-09-08 23:46:58 +02:00
FilterPlainRightAnchored.prototype.toSelfie = function() {
return this.s;
};
2015-02-24 00:31:29 +01:00
FilterPlainRightAnchored.compile = function(details) {
return details.f;
};
2014-09-08 23:46:58 +02:00
FilterPlainRightAnchored.fromSelfie = function(s) {
return new FilterPlainRightAnchored(s);
};
2014-08-28 15:59:05 +02:00
/******************************************************************************/
2014-06-24 00:42:43 +02:00
var FilterPlainRightAnchoredHostname = function(s, hostname) {
this.s = s;
this.hostname = hostname;
};
FilterPlainRightAnchoredHostname.prototype.match = function(url) {
2015-02-05 00:06:31 +01:00
return pageHostnameRegister.slice(-this.hostname.length) === this.hostname &&
2014-06-24 00:42:43 +02:00
url.slice(-this.s.length) === this.s;
};
2015-02-24 00:31:29 +01:00
FilterPlainRightAnchoredHostname.fid = FilterPlainRightAnchoredHostname.prototype.fid = 'a|h';
2014-09-08 23:46:58 +02:00
2014-08-28 15:59:05 +02:00
FilterPlainRightAnchoredHostname.prototype.toString = function() {
return this.s + '|$domain=' + this.hostname;
};
2014-09-08 23:46:58 +02:00
FilterPlainRightAnchoredHostname.prototype.toSelfie = function() {
return this.s + '\t' +
this.hostname;
};
2015-02-24 00:31:29 +01:00
FilterPlainRightAnchoredHostname.compile = function(details, hostname) {
return details.f + '\t' + hostname;
};
2014-09-08 23:46:58 +02:00
FilterPlainRightAnchoredHostname.fromSelfie = function(s) {
var pos = s.indexOf('\t');
return new FilterPlainRightAnchoredHostname(s.slice(0, pos), s.slice(pos + 1));
};
2014-06-24 00:42:43 +02:00
/******************************************************************************/
2014-09-19 16:59:44 +02:00
// https://github.com/gorhill/uBlock/issues/235
// The filter is left-anchored somewhere within the hostname part of the URL.
var FilterPlainHnAnchored = function(s) {
this.s = s;
};
FilterPlainHnAnchored.prototype.match = function(url, tokenBeg) {
if ( url.substr(tokenBeg, this.s.length) !== this.s ) {
return false;
}
// Valid only if hostname-valid characters to the left of token
var pos = url.indexOf('://');
return pos !== -1 &&
reURLPostHostnameAnchors.test(url.slice(pos + 3, tokenBeg)) === false;
};
2015-03-05 01:36:09 +01:00
FilterPlainHnAnchored.fid = FilterPlainHnAnchored.prototype.fid = '||a';
2014-09-19 16:59:44 +02:00
FilterPlainHnAnchored.prototype.toString = function() {
return '||' + this.s;
};
FilterPlainHnAnchored.prototype.toSelfie = function() {
return this.s;
};
2015-02-24 00:31:29 +01:00
FilterPlainHnAnchored.compile = function(details) {
return details.f;
};
2014-09-19 16:59:44 +02:00
FilterPlainHnAnchored.fromSelfie = function(s) {
return new FilterPlainHnAnchored(s);
};
// https://www.youtube.com/watch?v=71YS6xDB-E4
/******************************************************************************/
2015-03-05 01:36:09 +01:00
// Generic filter
2014-09-08 23:46:58 +02:00
2015-03-05 01:36:09 +01:00
var FilterGeneric = function(s, anchor) {
this.s = s;
this.anchor = anchor;
this.re = null;
2014-06-24 00:42:43 +02:00
};
2015-03-05 01:36:09 +01:00
FilterGeneric.prototype.match = function(url) {
if ( this.re === null ) {
this.re = strToRegex(this.s, this.anchor);
}
return this.re.test(url);
2014-06-24 00:42:43 +02:00
};
2015-03-05 01:36:09 +01:00
FilterGeneric.fid = FilterGeneric.prototype.fid = '_';
2014-09-08 23:46:58 +02:00
2015-03-05 01:36:09 +01:00
FilterGeneric.prototype.toString = function() {
if ( this.anchor === 0 ) {
return this.s;
}
if ( this.anchor < 0 ) {
return '|' + this.s;
}
return this.s + '|';
2014-08-28 15:59:05 +02:00
};
2015-03-05 01:36:09 +01:00
FilterGeneric.prototype.toSelfie = function() {
return this.s + '\t' + this.anchor;
2014-09-08 23:46:58 +02:00
};
2015-03-05 01:36:09 +01:00
FilterGeneric.compile = function(details) {
return details.f + '\t' + details.anchor;
2015-02-24 00:31:29 +01:00
};
2015-03-05 01:36:09 +01:00
FilterGeneric.fromSelfie = function(s) {
2014-09-08 23:46:58 +02:00
var pos = s.indexOf('\t');
2015-03-05 01:36:09 +01:00
return new FilterGeneric(s.slice(0, pos), parseInt(s.slice(pos + 1), 10));
2014-09-08 23:46:58 +02:00
};
2014-06-24 00:42:43 +02:00
2014-09-08 23:46:58 +02:00
/******************************************************************************/
2015-03-05 01:36:09 +01:00
// Generic filter
2014-08-28 15:59:05 +02:00
2015-03-05 01:36:09 +01:00
var FilterGenericHostname = function(s, anchor, hostname) {
FilterGeneric.call(this, s, anchor);
2014-06-24 00:42:43 +02:00
this.hostname = hostname;
};
2015-03-05 01:36:09 +01:00
FilterGenericHostname.prototype = Object.create(FilterGeneric.prototype);
FilterGenericHostname.prototype.constructor = FilterGenericHostname;
2014-06-24 00:42:43 +02:00
2015-03-05 01:36:09 +01:00
FilterGenericHostname.prototype.match = function(url) {
if ( pageHostnameRegister.slice(-this.hostname.length) !== this.hostname ) {
return false;
}
return FilterGeneric.prototype.match.call(this, url);
2014-06-24 00:42:43 +02:00
};
2015-03-05 01:36:09 +01:00
FilterGenericHostname.fid = FilterGenericHostname.prototype.fid = '_h';
2014-09-08 23:46:58 +02:00
2015-03-05 01:36:09 +01:00
FilterGenericHostname.prototype.toString = function() {
return FilterGeneric.prototype.toString.call(this) + '$domain=' + this.hostname;
2014-08-28 15:59:05 +02:00
};
2015-03-05 01:36:09 +01:00
FilterGenericHostname.prototype.toSelfie = function() {
return FilterGeneric.prototype.toSelfie.call(this) + '\t' + this.hostname;
2014-09-08 23:46:58 +02:00
};
2015-03-05 01:36:09 +01:00
FilterGenericHostname.compile = function(details, hostname) {
return FilterGeneric.compile(details) + '\t' + hostname;
2015-02-24 00:31:29 +01:00
};
2015-03-05 01:36:09 +01:00
FilterGenericHostname.fromSelfie = function(s) {
var fields = s.split('\t');
return new FilterGenericHostname(fields[0], parseInt(fields[1], 10), fields[2]);
2014-09-08 23:46:58 +02:00
};
2014-06-24 00:42:43 +02:00
/******************************************************************************/
2015-03-02 16:41:51 +01:00
// Generic filter: hostname-anchored: it has that extra test to find out
// whether the start of the match falls within the hostname part of the
// URL.
var FilterGenericHnAnchored = function(s) {
this.s = s;
this.re = null;
};
FilterGenericHnAnchored.prototype.match = function(url) {
if ( this.re === null ) {
2015-03-05 01:36:09 +01:00
this.re = strToRegex(this.s, 0);
2015-03-02 16:41:51 +01:00
}
// Quick test first
if ( this.re.test(url) === false ) {
return false;
}
// Valid only if begininning of match is within the hostname
// part of the url
var match = this.re.exec(url);
var pos = url.indexOf('://');
return pos !== -1 &&
reURLPostHostnameAnchors.test(url.slice(pos + 3, match.index)) === false;
};
FilterGenericHnAnchored.fid = FilterGenericHnAnchored.prototype.fid = '||_';
FilterGenericHnAnchored.prototype.toString = function() {
2015-03-02 17:25:45 +01:00
return '||' + this.s;
2015-03-02 16:41:51 +01:00
};
FilterGenericHnAnchored.prototype.toSelfie = function() {
return this.s;
};
FilterGenericHnAnchored.compile = function(details) {
return details.f;
};
FilterGenericHnAnchored.fromSelfie = function(s) {
return new FilterGenericHnAnchored(s);
};
/******************************************************************************/
2015-03-02 22:22:23 +01:00
var FilterGenericHnAnchoredHostname = function(s, hostname) {
FilterGenericHnAnchored.call(this, s);
this.hostname = hostname;
};
FilterGenericHnAnchoredHostname.prototype = Object.create(FilterGenericHnAnchored.prototype);
2015-03-05 01:36:09 +01:00
FilterGenericHnAnchoredHostname.prototype.constructor = FilterGenericHnAnchoredHostname;
2015-03-02 22:22:23 +01:00
FilterGenericHnAnchoredHostname.prototype.match = function(url) {
if ( pageHostnameRegister.slice(-this.hostname.length) !== this.hostname ) {
return false;
}
2015-03-03 09:02:09 +01:00
return FilterGenericHnAnchored.prototype.match.call(this, url);
2015-03-02 22:22:23 +01:00
};
FilterGenericHnAnchoredHostname.fid = FilterGenericHnAnchoredHostname.prototype.fid = '||_h';
FilterGenericHnAnchoredHostname.prototype.toString = function() {
return '||' + this.s + '$domain=' + this.hostname;
};
FilterGenericHnAnchoredHostname.prototype.toSelfie = function() {
return this.s + '\t' + this.hostname;
};
FilterGenericHnAnchoredHostname.compile = function(details, hostname) {
return details.f + '\t' + hostname;
};
FilterGenericHnAnchoredHostname.fromSelfie = function(s) {
var pos = s.indexOf('\t');
return new FilterGenericHnAnchoredHostname(s.slice(0, pos), s.slice(pos + 1));
};
/******************************************************************************/
2015-01-23 17:32:49 +01:00
// Regex-based filters
var FilterRegex = function(s) {
this.re = new RegExp(s);
};
FilterRegex.prototype.match = function(url) {
return this.re.test(url);
};
2015-02-24 00:31:29 +01:00
FilterRegex.fid = FilterRegex.prototype.fid = '//';
2015-01-23 17:32:49 +01:00
FilterRegex.prototype.toString = function() {
return '/' + this.re.source + '/';
};
FilterRegex.prototype.toSelfie = function() {
return this.re.source;
};
2015-02-24 00:31:29 +01:00
FilterRegex.compile = function(details) {
return details.f;
};
2015-01-23 17:32:49 +01:00
FilterRegex.fromSelfie = function(s) {
return new FilterRegex(s);
};
/******************************************************************************/
var FilterRegexHostname = function(s, hostname) {
this.re = new RegExp(s);
this.hostname = hostname;
};
FilterRegexHostname.prototype.match = function(url) {
// test hostname first, it's cheaper than evaluating a regex
2015-02-05 00:06:31 +01:00
return pageHostnameRegister.slice(-this.hostname.length) === this.hostname &&
2015-01-23 17:32:49 +01:00
this.re.test(url);
};
2015-02-24 00:31:29 +01:00
FilterRegexHostname.fid = FilterRegexHostname.prototype.fid = '//h';
2015-01-23 17:32:49 +01:00
FilterRegexHostname.prototype.toString = function() {
return '/' + this.re.source + '/$domain=' + this.hostname;
};
FilterRegexHostname.prototype.toSelfie = function() {
2015-02-24 00:31:29 +01:00
return this.re.source + '\t' + this.hostname;
};
FilterRegexHostname.compile = function(details, hostname) {
return details.f + '\t' + hostname;
2015-01-23 17:32:49 +01:00
};
FilterRegexHostname.fromSelfie = function(s) {
var pos = s.indexOf('\t');
return new FilterRegexHostname(s.slice(0, pos), s.slice(pos + 1));
};
/******************************************************************************/
2014-09-08 23:46:58 +02:00
/******************************************************************************/
2015-02-05 00:06:31 +01:00
// Dictionary of hostnames
2015-02-05 14:45:29 +01:00
//
// FilterHostnameDict is the main reason why uBlock is not equipped to keep
// track of which filter comes from which list, and also why it's not equipped
// to be able to disable a specific filter -- other than through using a
// counter-filter.
//
// On the other hand it is also *one* of the reason uBlock's memory and CPU
// footprint is smaller. Compacting huge list of hostnames into single strings
// saves a lot of memory compared to having one dictionary entry per hostname.
2015-02-05 00:06:31 +01:00
var FilterHostnameDict = function() {
this.h = ''; // short-lived register
this.dict = {};
this.count = 0;
};
// Somewhat arbitrary: I need to come up with hard data to know at which
// point binary search is better than indexOf.
//
// http://jsperf.com/string-indexof-vs-binary-search
// Tuning above performance benchmark, it appears 250 is roughly a good value
// for both Chromium/Firefox.
// Example of benchmark values: '------30', '-----100', etc. -- the
// needle string must always be 8-character long.
FilterHostnameDict.prototype.cutoff = 250;
// Probably not needed under normal circumstances.
FilterHostnameDict.prototype.meltBucket = function(len, bucket) {
var map = {};
if ( bucket.charAt(0) === ' ' ) {
bucket.trim().split(' ').map(function(k) {
map[k] = true;
});
} else {
var offset = 0;
while ( offset < bucket.length ) {
map[bucket.substring(offset, len)] = true;
offset += len;
}
}
return map;
};
2015-02-05 14:45:29 +01:00
// How the key is derived dictates the number and size of buckets:
// - more bits = more buckets = higher memory footprint
// - less bits = less buckets = lower memory footprint
// - binary search mitigates very well the fact that some buckets may grow
// large when fewer bits are used (or when a large number of items are
// stored). Binary search also mitigate to the point of non-issue the
// CPU footprint requirement with large buckets, as far as reference
// benchmark shows.
2015-02-05 00:06:31 +01:00
//
// A hash key capable of better spread while being as fast would be
// just great.
FilterHostnameDict.prototype.makeKey = function(hn) {
var len = hn.length;
if ( len > 255 ) {
len = 255;
}
var i8 = len >>> 3;
var i4 = len >>> 2;
var i2 = len >>> 1;
// http://jsperf.com/makekey-concat-vs-join/3
// Be sure the msb is not set, this will guarantee a valid unicode
// character (because 0xD800-0xDFFF).
return String.fromCharCode(
(hn.charCodeAt( i8) & 0x01) << 14 |
// (hn.charCodeAt( i4 ) & 0x01) << 13 |
(hn.charCodeAt( i4+i8) & 0x01) << 12 |
(hn.charCodeAt(i2 ) & 0x01) << 11 |
(hn.charCodeAt(i2 +i8) & 0x01) << 10 |
// (hn.charCodeAt(i2+i4 ) & 0x01) << 9 |
(hn.charCodeAt(i2+i4+i8) & 0x01) << 8 ,
len
);
};
FilterHostnameDict.prototype.add = function(hn) {
var key = this.makeKey(hn);
var bucket = this.dict[key];
if ( bucket === undefined ) {
bucket = this.dict[key] = {};
bucket[hn] = true;
this.count += 1;
return true;
}
if ( typeof bucket === 'string' ) {
bucket = this.dict[key] = this.meltBucket(hn.len, bucket);
}
2015-02-24 00:31:29 +01:00
if ( bucket.hasOwnProperty(hn) ) {
return false;
2015-02-05 00:06:31 +01:00
}
2015-02-24 00:31:29 +01:00
bucket[hn] = true;
this.count += 1;
return true;
2015-02-05 00:06:31 +01:00
};
FilterHostnameDict.prototype.freeze = function() {
var buckets = this.dict;
var bucket, hostnames, len;
for ( var key in buckets ) {
bucket = buckets[key];
if ( typeof bucket !== 'object' ) {
continue;
}
hostnames = Object.keys(bucket);
len = hostnames[0].length * hostnames.length;
if ( hostnames[0].length * hostnames.length < this.cutoff ) {
buckets[key] = ' ' + hostnames.join(' ') + ' ';
} else {
buckets[key] = hostnames.sort().join('');
}
}
};
FilterHostnameDict.prototype.matchesExactly = function(hn) {
// TODO: Handle IP address
var key = this.makeKey(hn);
var bucket = this.dict[key];
if ( bucket === undefined ) {
return false;
}
if ( typeof bucket === 'object' ) {
2015-02-20 05:25:29 +01:00
return bucket.hasOwnProperty(hn);
2015-02-05 00:06:31 +01:00
}
if ( bucket.charAt(0) === ' ' ) {
return bucket.indexOf(' ' + hn + ' ') !== -1;
}
// binary search
var len = hn.length;
var left = 0;
// http://jsperf.com/or-vs-floor/17
var right = (bucket.length / len + 0.5) | 0;
var i, needle;
while ( left < right ) {
i = left + right >> 1;
needle = bucket.substr( len * i, len );
if ( hn < needle ) {
right = i;
} else if ( hn > needle ) {
left = i + 1;
} else {
return true;
}
}
return false;
};
2015-02-24 00:31:29 +01:00
FilterHostnameDict.prototype.match = function() {
2015-02-05 00:06:31 +01:00
// TODO: mind IP addresses
var pos,
hostname = requestHostnameRegister;
while ( this.matchesExactly(hostname) === false ) {
pos = hostname.indexOf('.');
if ( pos === -1 ) {
this.h = '';
return false;
}
hostname = hostname.slice(pos + 1);
}
this.h = '||' + hostname + '^';
return this;
};
2015-02-24 00:31:29 +01:00
FilterHostnameDict.fid = FilterHostnameDict.prototype.fid = '{h}';
2015-02-05 00:06:31 +01:00
FilterHostnameDict.prototype.toString = function() {
return this.h;
};
FilterHostnameDict.prototype.toSelfie = function() {
return JSON.stringify({
count: this.count,
dict: this.dict
});
};
FilterHostnameDict.fromSelfie = function(s) {
var f = new FilterHostnameDict();
var o = JSON.parse(s);
f.count = o.count;
f.dict = o.dict;
return f;
};
/******************************************************************************/
/******************************************************************************/
// Some buckets can grow quite large, and finding a hit in these buckets
// may end up being expensive. After considering various solutions, the one
// retained is to promote hit filters to a smaller index, so that next time
// they can be looked-up faster.
2014-09-19 16:59:44 +02:00
// key= 10000 ad count=660
// key= 10000 ads count=433
// key= 10001 google count=277
// key=1000000 2mdn count=267
// key= 10000 social count=240
// key= 10001 pagead2 count=166
// key= 10000 twitter count=122
// key= 10000 doubleclick count=118
// key= 10000 facebook count=114
// key= 10000 share count=113
// key= 10000 google count=106
// key= 10001 code count=103
// key= 11000 doubleclick count=100
// key=1010001 g count=100
// key= 10001 js count= 89
// key= 10000 adv count= 88
// key= 10000 youtube count= 61
// key= 10000 plugins count= 60
// key= 10001 partner count= 59
// key= 10000 ico count= 57
// key= 110001 ssl count= 57
// key= 10000 banner count= 53
// key= 10000 footer count= 51
// key= 10000 rss count= 51
2014-09-19 16:59:44 +02:00
/******************************************************************************/
2014-09-08 23:46:58 +02:00
var FilterBucket = function(a, b) {
this.promoted = 0;
this.vip = 16;
2014-10-06 20:02:44 +02:00
this.f = null; // short-lived register
2014-09-08 23:46:58 +02:00
this.filters = [];
if ( a !== undefined ) {
this.filters[0] = a;
if ( b !== undefined ) {
this.filters[1] = b;
}
}
};
FilterBucket.prototype.add = function(a) {
this.filters.push(a);
};
// Promote hit filters so they can be found faster next time.
FilterBucket.prototype.promote = function(i) {
var filters = this.filters;
var pivot = filters.length >>> 1;
while ( i < pivot ) {
pivot >>>= 1;
if ( pivot < this.vip ) {
break;
}
}
if ( i <= pivot ) {
return;
}
var j = this.promoted % pivot;
//console.debug('FilterBucket.promote(): promoted %d to %d', i, j);
var f = filters[j];
filters[j] = filters[i];
filters[i] = f;
this.promoted += 1;
};
2014-09-08 23:46:58 +02:00
FilterBucket.prototype.match = function(url, tokenBeg) {
var filters = this.filters;
var n = filters.length;
for ( var i = 0; i < n; i++ ) {
2014-09-08 23:46:58 +02:00
if ( filters[i].match(url, tokenBeg) !== false ) {
this.f = filters[i];
if ( i >= this.vip ) {
this.promote(i);
}
2014-09-08 23:46:58 +02:00
return true;
}
}
return false;
};
FilterBucket.prototype.fid = '[]';
FilterBucket.prototype.toString = function() {
if ( this.f !== null ) {
return this.f.toString();
}
return '';
};
FilterBucket.prototype.toSelfie = function() {
return this.filters.length.toString();
};
FilterBucket.fromSelfie = function() {
return new FilterBucket();
};
2014-06-24 00:42:43 +02:00
/******************************************************************************/
2015-02-24 00:31:29 +01:00
var getFilterClass = function(details) {
2015-01-23 17:32:49 +01:00
if ( details.isRegex ) {
2015-02-24 00:31:29 +01:00
return FilterRegex;
2015-01-23 17:32:49 +01:00
}
2015-02-24 00:31:29 +01:00
var s = details.f;
2015-03-17 14:39:03 +01:00
if ( s.indexOf('*') !== -1 || details.token === '*' ) {
2015-03-02 16:41:51 +01:00
if ( details.hostnameAnchored ) {
return FilterGenericHnAnchored;
}
2015-03-05 01:36:09 +01:00
return FilterGeneric;
2014-06-24 00:42:43 +02:00
}
if ( details.anchor < 0 ) {
2015-02-24 00:31:29 +01:00
return FilterPlainLeftAnchored;
2014-06-24 00:42:43 +02:00
}
if ( details.anchor > 0 ) {
2015-02-24 00:31:29 +01:00
return FilterPlainRightAnchored;
2014-06-24 00:42:43 +02:00
}
2014-09-19 16:59:44 +02:00
if ( details.hostnameAnchored ) {
2015-02-24 00:31:29 +01:00
return FilterPlainHnAnchored;
2014-09-19 16:59:44 +02:00
}
2015-01-23 17:32:49 +01:00
if ( details.tokenBeg === 0 ) {
2015-02-24 00:31:29 +01:00
return FilterPlainPrefix0;
2014-06-24 00:42:43 +02:00
}
2015-01-23 17:32:49 +01:00
if ( details.tokenBeg === 1 ) {
2015-02-24 00:31:29 +01:00
return FilterPlainPrefix1;
2014-06-24 00:42:43 +02:00
}
2015-02-24 00:31:29 +01:00
return FilterPlain;
2014-06-24 00:42:43 +02:00
};
/******************************************************************************/
2015-02-24 00:31:29 +01:00
var getHostnameBasedFilterClass = function(details) {
2015-01-23 17:32:49 +01:00
if ( details.isRegex ) {
2015-02-24 00:31:29 +01:00
return FilterRegexHostname;
2015-01-23 17:32:49 +01:00
}
2015-02-24 00:31:29 +01:00
var s = details.f;
2015-03-17 14:39:03 +01:00
if ( s.indexOf('*') !== -1 || details.token === '*' ) {
2015-03-02 22:22:23 +01:00
if ( details.hostnameAnchored ) {
return FilterGenericHnAnchoredHostname;
}
2015-03-05 01:36:09 +01:00
return FilterGenericHostname;
2014-06-24 00:42:43 +02:00
}
if ( details.anchor < 0 ) {
2015-02-24 00:31:29 +01:00
return FilterPlainLeftAnchoredHostname;
2014-06-24 00:42:43 +02:00
}
if ( details.anchor > 0 ) {
2015-02-24 00:31:29 +01:00
return FilterPlainRightAnchoredHostname;
2014-06-24 00:42:43 +02:00
}
2015-01-23 17:32:49 +01:00
if ( details.tokenBeg === 0 ) {
2015-02-24 00:31:29 +01:00
return FilterPlainPrefix0Hostname;
2014-06-24 00:42:43 +02:00
}
2015-01-23 17:32:49 +01:00
if ( details.tokenBeg === 1 ) {
2015-02-24 00:31:29 +01:00
return FilterPlainPrefix1Hostname;
2014-06-24 00:42:43 +02:00
}
2015-02-24 00:31:29 +01:00
return FilterPlainHostname;
2014-06-24 00:42:43 +02:00
};
/******************************************************************************/
// Trim leading/trailing char "c"
var trimChar = function(s, c) {
// Remove leading and trailing wildcards
var pos = 0;
while ( s.charAt(pos) === c ) {
pos += 1;
}
s = s.slice(pos);
if ( pos = s.length ) {
while ( s.charAt(pos-1) === c ) {
pos -= 1;
}
s = s.slice(0, pos);
}
return s;
};
2014-09-08 23:46:58 +02:00
/******************************************************************************/
2014-06-24 00:42:43 +02:00
/******************************************************************************/
var FilterParser = function() {
2015-01-23 17:32:49 +01:00
this.reHasWildcard = /[\^\*]/;
this.reHasUppercase = /[A-Z]/;
2015-02-27 00:08:42 +01:00
this.reCleanupHostname = /^\|\|[.*]*/;
this.reIsolateHostname = /^([^\x00-\x24\x26-\x2C\x2F\x3A-\x5E\x60\x7B-\x7F]+)(.*)/;
this.reHasUnicode = /[^\x00-\x7F]/;
2014-06-24 00:42:43 +02:00
this.hostnames = [];
2015-01-23 17:32:49 +01:00
this.notHostnames = [];
2014-08-28 15:59:05 +02:00
this.reset();
2014-06-24 00:42:43 +02:00
};
/******************************************************************************/
FilterParser.prototype.toNormalizedType = {
'stylesheet': 'stylesheet',
'image': 'image',
'object': 'object',
'object-subrequest': 'object',
'script': 'script',
'xmlhttprequest': 'xmlhttprequest',
'subdocument': 'sub_frame',
2014-07-14 17:24:59 +02:00
'other': 'other',
2015-03-26 00:28:22 +01:00
'document': 'main_frame',
'elemhide': 'cosmetic-filtering',
2014-09-24 23:38:22 +02:00
'inline-script': 'inline-script',
2014-07-14 17:24:59 +02:00
'popup': 'popup'
2014-06-24 00:42:43 +02:00
};
/******************************************************************************/
FilterParser.prototype.reset = function() {
this.action = BlockAction;
this.anchor = 0;
this.elemHiding = false;
this.f = '';
this.firstParty = false;
this.fopts = '';
2014-09-19 16:59:44 +02:00
this.hostnameAnchored = false;
this.hostnamePure = false;
2014-08-28 15:59:05 +02:00
this.hostnames.length = 0;
2015-01-23 17:32:49 +01:00
this.notHostnames.length = 0;
this.isRegex = false;
2014-06-24 00:42:43 +02:00
this.thirdParty = false;
2015-01-23 17:32:49 +01:00
this.token = '';
this.tokenBeg = 0;
this.tokenEnd = 0;
this.types = 0;
2014-08-29 21:02:31 +02:00
this.important = 0;
2014-06-24 00:42:43 +02:00
this.unsupported = false;
return this;
};
/******************************************************************************/
// https://github.com/gorhill/uBlock/issues/589
// Be ready to handle multiple negated types
2014-06-24 00:42:43 +02:00
FilterParser.prototype.parseOptType = function(raw, not) {
2015-03-26 00:28:22 +01:00
var typeBit = 1 << ((typeNameToTypeValue[this.toNormalizedType[raw]] >>> 4) - 1);
if ( !not ) {
2015-03-26 00:28:22 +01:00
this.types |= typeBit;
return;
2014-06-24 00:42:43 +02:00
}
2015-02-08 04:20:24 +01:00
// Negated type: set all valid network request type bits to 1
if ( this.types === 0 ) {
this.types = allNetRequestTypesBitmap;
}
2015-03-26 00:28:22 +01:00
this.types &= ~typeBit;
2014-06-24 00:42:43 +02:00
};
/******************************************************************************/
FilterParser.prototype.parseOptParty = function(not) {
if ( not ) {
this.firstParty = true;
} else {
this.thirdParty = true;
}
};
/******************************************************************************/
FilterParser.prototype.parseOptHostnames = function(raw) {
var hostnames = raw.split('|');
2015-01-23 17:32:49 +01:00
var hostname;
2014-06-24 00:42:43 +02:00
for ( var i = 0; i < hostnames.length; i++ ) {
hostname = hostnames[i];
2015-01-23 17:32:49 +01:00
if ( hostname.charAt(0) === '~' ) {
this.notHostnames.push(hostname.slice(1));
} else {
this.hostnames.push(hostname);
}
}
};
/******************************************************************************/
FilterParser.prototype.parseOptions = function(s) {
this.fopts = s;
var opts = s.split(',');
var opt, not;
for ( var i = 0; i < opts.length; i++ ) {
opt = opts[i];
not = opt.charAt(0) === '~';
2014-06-24 00:42:43 +02:00
if ( not ) {
2015-01-23 17:32:49 +01:00
opt = opt.slice(1);
2014-06-24 00:42:43 +02:00
}
2015-01-23 17:32:49 +01:00
if ( opt === 'third-party' ) {
this.parseOptParty(not);
continue;
}
2015-03-26 00:28:22 +01:00
if ( opt === 'elemhide' ) {
if ( this.action !== AllowAction ) {
this.parseOptType('elemhide', false);
this.action = BlockAction;
continue;
}
this.unsupported = true;
break;
}
if ( opt === 'document' ) {
if ( this.action === BlockAction ) {
this.parseOptType('document', false);
continue;
}
this.unsupported = true;
break;
2015-01-23 17:32:49 +01:00
}
if ( this.toNormalizedType.hasOwnProperty(opt) ) {
this.parseOptType(opt, not);
continue;
}
if ( opt.slice(0,7) === 'domain=' ) {
this.parseOptHostnames(opt.slice(7));
continue;
}
if ( opt === 'popup' ) {
this.parseOptType('popup', not);
continue;
2014-06-24 00:42:43 +02:00
}
2015-01-23 17:32:49 +01:00
if ( opt === 'important' ) {
this.important = Important;
continue;
2014-06-24 00:42:43 +02:00
}
2015-01-23 17:32:49 +01:00
this.unsupported = true;
break;
2014-06-24 00:42:43 +02:00
}
};
/******************************************************************************/
2015-02-27 00:08:42 +01:00
FilterParser.prototype.parse = function(raw) {
2014-06-24 00:42:43 +02:00
// important!
this.reset();
2015-02-27 00:08:42 +01:00
var s = raw;
2015-01-23 17:32:49 +01:00
// plain hostname?
2014-09-19 16:59:44 +02:00
if ( reHostnameRule.test(s) ) {
this.f = s;
this.hostnamePure = this.hostnameAnchored = true;
return this;
}
2014-06-24 00:42:43 +02:00
// element hiding filter?
2015-01-23 17:32:49 +01:00
var pos = s.indexOf('#');
if ( pos !== -1 ) {
var c = s.charAt(pos + 1);
if ( c === '#' || c === '@' ) {
console.error('static-net-filtering.js > unexpected cosmetic filters');
this.elemHiding = true;
return this;
}
}
// block or allow filter?
// Important: this must be executed before parsing options
if ( s.lastIndexOf('@@', 0) === 0 ) {
this.action = AllowAction;
s = s.slice(2);
}
2015-01-23 17:32:49 +01:00
// options
pos = s.indexOf('$');
if ( pos !== -1 ) {
this.parseOptions(s.slice(pos + 1));
s = s.slice(0, pos);
2014-06-24 00:42:43 +02:00
}
2015-01-23 17:32:49 +01:00
// regex?
2015-02-14 01:03:43 +01:00
if ( s.charAt(0) === '/' && s.slice(-1) === '/' && s.length > 2 ) {
2015-01-23 17:32:49 +01:00
this.isRegex = true;
this.f = s.slice(1, -1);
2014-09-08 23:46:58 +02:00
return this;
}
2015-02-27 00:08:42 +01:00
// hostname-anchored
2015-01-23 17:32:49 +01:00
if ( s.lastIndexOf('||', 0) === 0 ) {
2014-09-19 16:59:44 +02:00
this.hostnameAnchored = true;
2015-02-27 00:08:42 +01:00
// cleanup: `||example.com`, `||*.example.com^`, `||.example.com/*`
s = s.replace(this.reCleanupHostname, '');
// convert hostname to punycode if needed
if ( this.reHasUnicode.test(s) ) {
var matches = this.reIsolateHostname.exec(s);
if ( matches && matches.length === 3 ) {
s = punycode.toASCII(matches[1]) + matches[2];
//console.debug('µBlock.staticNetFilteringEngine/FilterParser.parse():', raw, '=', s);
}
}
2015-03-26 20:16:48 +01:00
// https://github.com/gorhill/uBlock/issues/1096
if ( s.charAt(0) === '^' ) {
this.unsupported = true;
return this;
}
2014-06-24 00:42:43 +02:00
}
// left-anchored
if ( s.charAt(0) === '|' ) {
this.anchor = -1;
s = s.slice(1);
}
// right-anchored
if ( s.slice(-1) === '|' ) {
this.anchor = 1;
s = s.slice(0, -1);
}
// normalize placeholders
// TODO: transforming `^` into `*` is not a strict interpretation of
// ABP syntax.
2015-01-23 17:32:49 +01:00
if ( this.reHasWildcard.test(s) ) {
s = s.replace(/\^/g, '*').replace(/\*\*+/g, '*');
s = trimChar(s, '*');
}
2014-09-19 16:59:44 +02:00
// nothing left?
if ( s === '' ) {
2015-03-17 14:39:03 +01:00
s = '*';
}
2015-01-23 17:32:49 +01:00
// plain hostname?
2014-09-19 16:59:44 +02:00
this.hostnamePure = this.hostnameAnchored && reHostnameRule.test(s);
2015-01-23 17:32:49 +01:00
// This might look weird but we gain memory footprint by not going through
// toLowerCase(), at least on Chromium. Because copy-on-write?
2014-06-24 00:42:43 +02:00
2015-01-23 17:32:49 +01:00
this.f = this.reHasUppercase.test(s) ? s.toLowerCase() : s;
return this;
};
/******************************************************************************/
2015-03-02 16:41:51 +01:00
// Given a string, find a good token. Tokens which are too generic, i.e. very
// common with a high probability of ending up as a miss, are not
// good. Avoid if possible. This has a *significant* positive impact on
// performance.
// These "bad tokens" are collated manually.
var reHostnameToken = /^[0-9a-z]+/g;
var reGoodToken = /[%0-9a-z]{2,}/g;
var badTokens = {
'com': true,
'http': true,
'https': true,
'icon': true,
'images': true,
'img': true,
'js': true,
'net': true,
'news': true,
'www': true
};
var findFirstGoodToken = function(s) {
reGoodToken.lastIndex = 0;
var matches;
while ( matches = reGoodToken.exec(s) ) {
2015-03-02 22:22:23 +01:00
if ( s.charAt(reGoodToken.lastIndex) === '*' ) {
continue;
}
2015-03-02 16:41:51 +01:00
if ( badTokens.hasOwnProperty(matches[0]) ) {
continue;
}
2015-03-02 22:22:23 +01:00
return matches;
}
// No good token found, try again without minding "bad" tokens
reGoodToken.lastIndex = 0;
while ( matches = reGoodToken.exec(s) ) {
2015-03-02 16:41:51 +01:00
if ( s.charAt(reGoodToken.lastIndex) === '*' ) {
continue;
}
return matches;
}
2015-03-02 22:22:23 +01:00
return null;
2015-03-02 16:41:51 +01:00
};
var findHostnameToken = function(s) {
reHostnameToken.lastIndex = 0;
return reHostnameToken.exec(s);
};
2015-03-02 22:22:23 +01:00
/******************************************************************************/
2015-01-23 17:32:49 +01:00
FilterParser.prototype.makeToken = function() {
if ( this.isRegex ) {
this.token = '*';
return;
2014-06-24 00:42:43 +02:00
}
2015-03-17 14:39:03 +01:00
var s = this.f;
// https://github.com/gorhill/uBlock/issues/1038
// Match any URL.
if ( s === '*' ) {
this.token = '*';
return;
}
2015-01-23 17:32:49 +01:00
var matches;
2015-03-02 16:41:51 +01:00
// Hostname-anchored with no wildcard always have a token index of 0.
2015-03-17 14:39:03 +01:00
if ( this.hostnameAnchored && s.indexOf('*') === -1 ) {
matches = findHostnameToken(s);
2015-01-23 17:32:49 +01:00
if ( !matches || matches[0].length === 0 ) {
return;
2014-06-24 00:42:43 +02:00
}
2015-01-23 17:32:49 +01:00
this.tokenBeg = matches.index;
this.tokenEnd = reHostnameToken.lastIndex;
2015-03-17 14:39:03 +01:00
this.token = s.slice(this.tokenBeg, this.tokenEnd);
2015-01-23 17:32:49 +01:00
return;
2014-06-24 00:42:43 +02:00
}
2015-01-23 17:32:49 +01:00
2015-03-17 14:39:03 +01:00
matches = findFirstGoodToken(s);
2015-03-02 22:22:23 +01:00
if ( matches === null || matches[0].length === 0 ) {
2015-03-17 14:39:03 +01:00
this.token = '*';
2015-01-23 17:32:49 +01:00
return;
}
this.tokenBeg = matches.index;
this.tokenEnd = reGoodToken.lastIndex;
2015-03-17 14:39:03 +01:00
this.token = s.slice(this.tokenBeg, this.tokenEnd);
2014-06-24 00:42:43 +02:00
};
/******************************************************************************/
/******************************************************************************/
var TokenEntry = function() {
this.beg = 0;
this.token = '';
};
/******************************************************************************/
/******************************************************************************/
2014-06-24 00:42:43 +02:00
var FilterContainer = function() {
2014-07-20 21:00:26 +02:00
this.reAnyToken = /[%0-9a-z]+/g;
this.tokens = [];
2014-06-24 00:42:43 +02:00
this.filterParser = new FilterParser();
2014-07-20 21:00:26 +02:00
this.reset();
};
/******************************************************************************/
// Reset all, thus reducing to a minimum memory footprint of the context.
FilterContainer.prototype.reset = function() {
this.frozen = false;
2014-06-24 00:42:43 +02:00
this.processedFilterCount = 0;
this.acceptedCount = 0;
2014-09-08 23:46:58 +02:00
this.rejectedCount = 0;
2014-06-24 00:42:43 +02:00
this.allowFilterCount = 0;
this.blockFilterCount = 0;
this.duplicateCount = 0;
2015-02-24 00:31:29 +01:00
this.duplicateBuster = {};
this.categories = Object.create(null);
2014-07-20 21:00:26 +02:00
this.filterParser.reset();
2015-03-05 01:36:09 +01:00
this.filterCounts = {};
2014-07-20 21:00:26 +02:00
};
2014-06-24 00:42:43 +02:00
2014-07-20 21:00:26 +02:00
/******************************************************************************/
2014-06-24 00:42:43 +02:00
2014-07-20 21:00:26 +02:00
FilterContainer.prototype.freeze = function() {
histogram('allFilters', this.categories);
2015-02-24 00:31:29 +01:00
this.duplicateBuster = {};
2015-02-05 00:06:31 +01:00
var categories = this.categories;
var bucket;
for ( var k in categories ) {
bucket = categories[k]['.'];
if ( bucket !== undefined ) {
bucket.freeze();
}
}
2015-02-24 00:31:29 +01:00
2014-07-20 21:00:26 +02:00
this.filterParser.reset();
this.frozen = true;
2014-06-24 00:42:43 +02:00
};
/******************************************************************************/
2015-02-24 00:31:29 +01:00
FilterContainer.prototype.factories = {
'[]': FilterBucket,
'a': FilterPlain,
'ah': FilterPlainHostname,
'0a': FilterPlainPrefix0,
'0ah': FilterPlainPrefix0Hostname,
'1a': FilterPlainPrefix1,
'1ah': FilterPlainPrefix1Hostname,
'|a': FilterPlainLeftAnchored,
'|ah': FilterPlainLeftAnchoredHostname,
'a|': FilterPlainRightAnchored,
'a|h': FilterPlainRightAnchoredHostname,
2015-03-05 01:36:09 +01:00
'||a': FilterPlainHnAnchored,
2015-02-24 00:31:29 +01:00
'//': FilterRegex,
'//h': FilterRegexHostname,
2015-03-02 16:41:51 +01:00
'{h}': FilterHostnameDict,
2015-03-05 01:36:09 +01:00
'_': FilterGeneric,
'_h': FilterGenericHostname,
2015-03-02 22:22:23 +01:00
'||_': FilterGenericHnAnchored,
'||_h': FilterGenericHnAnchoredHostname
2015-02-24 00:31:29 +01:00
};
/******************************************************************************/
2014-09-08 23:46:58 +02:00
FilterContainer.prototype.toSelfie = function() {
var categoryToSelfie = function(dict) {
var selfie = [];
var bucket, ff, n, i, f;
for ( var k in dict ) {
// No need for hasOwnProperty() here: there is no prototype chain.
2014-09-08 23:46:58 +02:00
// We need to encode the key because there could be a `\n` or '\t'
// character in it, which would trip the code at parse time.
selfie.push('k2\t' + encode(k));
bucket = dict[k];
selfie.push(bucket.fid + '\t' + bucket.toSelfie());
if ( bucket.fid !== '[]' ) {
continue;
}
ff = bucket.filters;
n = ff.length;
for ( i = 0; i < n; i++ ) {
f = ff[i];
selfie.push(f.fid + '\t' + f.toSelfie());
}
}
return selfie.join('\n');
};
var categoriesToSelfie = function(dict) {
var selfie = [];
for ( var k in dict ) {
// No need for hasOwnProperty() here: there is no prototype chain.
2014-09-08 23:46:58 +02:00
// We need to encode the key because there could be a `\n` or '\t'
// character in it, which would trip the code at parse time.
selfie.push('k1\t' + encode(k));
selfie.push(categoryToSelfie(dict[k]));
}
return selfie.join('\n');
};
return {
processedFilterCount: this.processedFilterCount,
acceptedCount: this.acceptedCount,
rejectedCount: this.rejectedCount,
allowFilterCount: this.allowFilterCount,
blockFilterCount: this.blockFilterCount,
duplicateCount: this.duplicateCount,
2015-02-05 00:06:31 +01:00
categories: categoriesToSelfie(this.categories)
2014-09-08 23:46:58 +02:00
};
};
/******************************************************************************/
FilterContainer.prototype.fromSelfie = function(selfie) {
this.frozen = true;
this.processedFilterCount = selfie.processedFilterCount;
this.acceptedCount = selfie.acceptedCount;
this.rejectedCount = selfie.rejectedCount;
this.allowFilterCount = selfie.allowFilterCount;
this.blockFilterCount = selfie.blockFilterCount;
this.duplicateCount = selfie.duplicateCount;
var catKey, tokenKey;
var dict = this.categories, subdict;
var bucket = null;
var rawText = selfie.categories;
var rawEnd = rawText.length;
var lineBeg = 0, lineEnd;
var line, pos, what, factory;
while ( lineBeg < rawEnd ) {
lineEnd = rawText.indexOf('\n', lineBeg);
if ( lineEnd < 0 ) {
lineEnd = rawEnd;
}
line = rawText.slice(lineBeg, lineEnd);
lineBeg = lineEnd + 1;
pos = line.indexOf('\t');
what = line.slice(0, pos);
if ( what === 'k1' ) {
catKey = decode(line.slice(pos + 1));
subdict = dict[catKey] = Object.create(null);
2014-09-08 23:46:58 +02:00
bucket = null;
continue;
}
if ( what === 'k2' ) {
tokenKey = decode(line.slice(pos + 1));
bucket = null;
continue;
}
2015-02-24 00:31:29 +01:00
factory = this.factories[what];
2014-09-08 23:46:58 +02:00
if ( bucket === null ) {
bucket = subdict[tokenKey] = factory.fromSelfie(line.slice(pos + 1));
continue;
}
// When token key is reused, it can't be anything
// else than FilterBucket
bucket.add(factory.fromSelfie(line.slice(pos + 1)));
}
};
/******************************************************************************/
2014-06-24 00:42:43 +02:00
FilterContainer.prototype.makeCategoryKey = function(category) {
2015-02-24 00:31:29 +01:00
return category.toString(16);
2014-06-24 00:42:43 +02:00
};
/******************************************************************************/
2015-02-24 00:31:29 +01:00
FilterContainer.prototype.compile = function(raw, out) {
2014-06-24 00:42:43 +02:00
// ORDER OF TESTS IS IMPORTANT!
// Ignore empty lines
2015-02-01 00:34:46 +01:00
var s = raw.trim();
2015-01-23 17:32:49 +01:00
if ( s.length === 0 ) {
2014-06-24 00:42:43 +02:00
return false;
}
// Ignore comments
2015-01-23 17:32:49 +01:00
var c = s.charAt(0);
if ( c === '[' || c === '!' ) {
2014-06-24 00:42:43 +02:00
return false;
}
var parsed = this.filterParser.parse(s);
2015-01-23 17:32:49 +01:00
// Ignore element-hiding filters
if ( parsed.elemHiding ) {
2014-09-08 23:46:58 +02:00
return false;
}
2015-01-23 17:32:49 +01:00
// Ignore filters with unsupported options
if ( parsed.unsupported ) {
2015-02-01 00:34:46 +01:00
//console.log('static-net-filtering.js > FilterContainer.add(): unsupported filter "%s"', raw);
2014-06-24 00:42:43 +02:00
return false;
}
2014-09-19 16:59:44 +02:00
// Pure hostnames, use more efficient liquid dict
2015-02-05 00:06:31 +01:00
// https://github.com/gorhill/uBlock/issues/665
// Create a dict keyed on request type etc.
2015-02-24 00:31:29 +01:00
if ( parsed.hostnamePure && this.compileHostnameOnlyFilter(parsed, out) ) {
2015-02-05 00:06:31 +01:00
return true;
2014-06-24 00:42:43 +02:00
}
2015-02-24 00:31:29 +01:00
var r = this.compileFilter(parsed, out);
2014-06-24 00:42:43 +02:00
if ( r === false ) {
return false;
}
return true;
};
/******************************************************************************/
2015-02-05 00:06:31 +01:00
// Using fast/compact dictionary when filter is a (or portion of) pure hostname.
2015-02-24 00:31:29 +01:00
FilterContainer.prototype.compileHostnameOnlyFilter = function(parsed, out) {
2015-02-05 00:06:31 +01:00
// Can't fit the filter in a pure hostname dictionary.
if ( parsed.hostnames.length !== 0 || parsed.notHostnames.length !== 0 ) {
2015-02-24 00:31:29 +01:00
return;
2015-02-05 00:06:31 +01:00
}
var party = AnyParty;
if ( parsed.firstParty !== parsed.thirdParty ) {
party = parsed.firstParty ? FirstParty : ThirdParty;
}
var keyShard = parsed.action | parsed.important | party;
2015-03-26 00:28:22 +01:00
var type = parsed.types;
if ( type === 0 ) {
out.push(
'n\v' +
this.makeCategoryKey(keyShard) + '\v' +
'.\v' +
parsed.f
);
return true;
}
2015-02-05 00:06:31 +01:00
var bitOffset = 1;
2015-03-26 00:28:22 +01:00
do {
2015-02-05 00:06:31 +01:00
if ( type & 1 ) {
2015-02-24 00:31:29 +01:00
out.push(
'n\v' +
this.makeCategoryKey(keyShard | (bitOffset << 4)) + '\v' +
'.\v' +
parsed.f
);
2015-02-05 00:06:31 +01:00
}
bitOffset += 1;
type >>>= 1;
2015-03-26 00:28:22 +01:00
} while ( type !== 0 );
2015-02-05 00:06:31 +01:00
return true;
};
/******************************************************************************/
2015-02-24 00:31:29 +01:00
FilterContainer.prototype.compileFilter = function(parsed, out) {
2015-01-23 17:32:49 +01:00
parsed.makeToken();
if ( parsed.token === '' ) {
console.error('static-net-filtering.js > FilterContainer.addFilter("%s"): can\'t tokenize', parsed.f);
2014-06-24 00:42:43 +02:00
return false;
}
2014-08-28 00:39:08 +02:00
2014-09-25 19:26:29 +02:00
var party = AnyParty;
if ( parsed.firstParty !== parsed.thirdParty ) {
party = parsed.firstParty ? FirstParty : ThirdParty;
}
2015-02-24 00:31:29 +01:00
var filterClass;
2015-01-23 17:32:49 +01:00
var i = parsed.hostnames.length;
var j = parsed.notHostnames.length;
2014-09-19 16:59:44 +02:00
2015-01-23 17:32:49 +01:00
// Applies to all domains without exceptions
if ( i === 0 && j === 0 ) {
2015-02-24 00:31:29 +01:00
filterClass = getFilterClass(parsed);
if ( filterClass === null ) {
2014-08-28 00:39:08 +02:00
return false;
}
2015-02-24 00:31:29 +01:00
this.compileToAtomicFilter(filterClass, parsed, party, out);
2015-01-23 17:32:49 +01:00
return true;
}
// Applies to specific domains
if ( i !== 0 ) {
2014-08-28 15:59:05 +02:00
while ( i-- ) {
2015-02-24 00:31:29 +01:00
filterClass = getHostnameBasedFilterClass(parsed);
if ( filterClass === null ) {
2014-06-24 00:42:43 +02:00
return false;
}
2015-02-24 00:31:29 +01:00
this.compileToAtomicFilter(filterClass, parsed, party, out, parsed.hostnames[i]);
2014-06-24 00:42:43 +02:00
}
2015-01-23 17:32:49 +01:00
}
// No exceptions
if ( j === 0 ) {
2014-08-28 00:39:08 +02:00
return true;
}
2015-01-23 17:32:49 +01:00
// Case:
// - applies everywhere except to specific domains
// Example:
// - ||adm.fwmrm.net/p/msnbc_live/$object-subrequest,third-party,domain=~msnbc.msn.com|~www.nbcnews.com
if ( i === 0 ) {
2015-02-24 00:31:29 +01:00
filterClass = getFilterClass(parsed);
if ( filterClass === null ) {
2015-01-23 17:32:49 +01:00
return false;
}
// https://github.com/gorhill/uBlock/issues/251
// Apply third-party option if it is present
2015-02-24 00:31:29 +01:00
this.compileToAtomicFilter(filterClass, parsed, party, out);
2014-08-28 00:39:08 +02:00
}
2014-09-25 19:26:29 +02:00
2015-01-23 17:32:49 +01:00
// Cases:
// - applies everywhere except to specific domains
// - applies to specific domains except other specific domains
// Example:
// - /^https?\:\/\/(?!(...)\/)/$script,third-party,xmlhttprequest,domain=photobucket.com|~secure.photobucket.com
2014-09-25 19:26:29 +02:00
2015-01-23 17:32:49 +01:00
// Reverse purpose of filter
parsed.action ^= ToggleAction;
while ( j-- ) {
2015-02-24 00:31:29 +01:00
filterClass = getHostnameBasedFilterClass(parsed);
if ( filterClass === null ) {
2015-01-23 17:32:49 +01:00
return false;
}
// https://github.com/gorhill/uBlock/issues/191#issuecomment-53654024
// If it is a block filter, we need to reverse the order of
// evaluation.
if ( parsed.action === BlockAction ) {
parsed.important = Important;
}
2015-02-24 00:31:29 +01:00
this.compileToAtomicFilter(filterClass, parsed, party, out, parsed.notHostnames[j]);
2015-01-23 17:32:49 +01:00
}
2014-06-24 00:42:43 +02:00
return true;
};
/******************************************************************************/
2015-02-24 00:31:29 +01:00
FilterContainer.prototype.compileToAtomicFilter = function(filterClass, parsed, party, out, hostname) {
2014-08-29 21:02:31 +02:00
var bits = parsed.action | parsed.important | party;
2015-03-26 00:28:22 +01:00
var type = parsed.types;
if ( type === 0 ) {
out.push(
'n\v' +
this.makeCategoryKey(bits) + '\v' +
parsed.token + '\v' +
filterClass.fid + '\v' +
filterClass.compile(parsed, hostname)
);
return;
}
2015-02-05 00:06:31 +01:00
var bitOffset = 1;
2015-03-26 00:28:22 +01:00
do {
2015-02-05 00:06:31 +01:00
if ( type & 1 ) {
2015-02-24 00:31:29 +01:00
out.push(
'n\v' +
this.makeCategoryKey(bits | (bitOffset << 4)) + '\v' +
parsed.token + '\v' +
filterClass.fid + '\v' +
filterClass.compile(parsed, hostname)
);
}
bitOffset += 1;
type >>>= 1;
2015-03-26 00:28:22 +01:00
} while ( type !== 0 );
2014-06-24 00:42:43 +02:00
};
/******************************************************************************/
2015-02-24 00:31:29 +01:00
FilterContainer.prototype.fromCompiledContent = function(text, lineBeg) {
var lineEnd;
var textEnd = text.length;
var line, fields, bucket, entry, factory, filter;
while ( lineBeg < textEnd ) {
if ( text.charAt(lineBeg) !== 'n' ) {
return lineBeg;
}
lineEnd = text.indexOf('\n', lineBeg);
if ( lineEnd === -1 ) {
lineEnd = textEnd;
}
line = text.slice(lineBeg + 2, lineEnd);
fields = line.split('\v');
lineBeg = lineEnd + 1;
this.acceptedCount += 1;
bucket = this.categories[fields[0]];
if ( bucket === undefined ) {
bucket = this.categories[fields[0]] = Object.create(null);
}
entry = bucket[fields[1]];
if ( fields[1] === '.' ) {
if ( entry === undefined ) {
entry = bucket['.'] = new FilterHostnameDict();
}
if ( entry.add(fields[2]) === false ) {
this.duplicateCount += 1;
}
continue;
}
if ( this.duplicateBuster.hasOwnProperty(line) ) {
this.duplicateCount += 1;
continue;
}
this.duplicateBuster[line] = true;
factory = this.factories[fields[2]];
2015-03-05 01:36:09 +01:00
2015-03-06 03:17:09 +01:00
// For development purpose
2015-03-05 01:36:09 +01:00
//if ( this.filterCounts.hasOwnProperty(fields[2]) === false ) {
// this.filterCounts[fields[2]] = 1;
//} else {
// this.filterCounts[fields[2]]++;
//}
2015-02-24 00:31:29 +01:00
filter = factory.fromSelfie(fields[3]);
if ( entry === undefined ) {
bucket[fields[1]] = filter;
continue;
}
if ( entry.fid === '[]' ) {
entry.add(filter);
continue;
}
bucket[fields[1]] = new FilterBucket(entry, filter);
2014-06-24 00:42:43 +02:00
}
2015-02-24 00:31:29 +01:00
return textEnd;
2014-06-24 00:42:43 +02:00
};
/******************************************************************************/
// Since the addition of the `important` evaluation, this means it is now
// likely that the url will have to be scanned more than once. So this is
// to ensure we do it once only, and reuse results.
FilterContainer.prototype.tokenize = function(url) {
var tokens = this.tokens;
2014-06-24 00:42:43 +02:00
var re = this.reAnyToken;
var matches, tokenEntry;
re.lastIndex = 0;
var i = 0;
while ( matches = re.exec(url) ) {
tokenEntry = tokens[i];
if ( tokenEntry === undefined ) {
tokenEntry = tokens[i] = new TokenEntry();
}
tokenEntry.beg = matches.index;
tokenEntry.token = matches[0];
i += 1;
2015-03-29 18:35:13 +02:00
// https://github.com/gorhill/uBlock/issues/1118
// Crazy case... but I guess we have to expect the worst...
if ( i === 2048 ) {
break;
}
}
2015-03-29 18:35:13 +02:00
// Sentinel
tokenEntry = tokens[i];
if ( tokenEntry === undefined ) {
tokenEntry = tokens[i] = new TokenEntry();
}
tokenEntry.token = '';
};
/******************************************************************************/
FilterContainer.prototype.matchTokens = function(bucket, url) {
2015-02-05 00:06:31 +01:00
// Hostname-only filters
var f = bucket['.'];
if ( f !== undefined && f.match() !== false ) {
return f;
}
var tokens = this.tokens;
2015-02-05 00:06:31 +01:00
var tokenEntry, token;
var i = 0;
for (;;) {
tokenEntry = tokens[i++];
token = tokenEntry.token;
if ( token === '' ) {
break;
}
f = bucket[token];
if ( f !== undefined && f.match(url, tokenEntry.beg) !== false ) {
return f;
2014-06-24 00:42:43 +02:00
}
}
2015-01-23 17:32:49 +01:00
// Regex-based filters
f = bucket['*'];
if ( f !== undefined && f.match(url) !== false ) {
return f;
}
2014-06-24 00:42:43 +02:00
return false;
};
/******************************************************************************/
// Specialized handlers
2014-07-30 03:10:00 +02:00
// https://github.com/gorhill/uBlock/issues/116
// Some type of requests are exceptional, they need custom handling,
// not the generic handling.
FilterContainer.prototype.matchStringExactType = function(context, requestURL, requestType) {
var url = requestURL.toLowerCase();
2015-02-05 00:06:31 +01:00
// These registers will be used by various filters
pageHostnameRegister = context.pageHostname || '';
requestHostnameRegister = µb.URI.hostnameFromURI(requestURL);
var party = isFirstParty(context.pageDomain, requestHostnameRegister) ? FirstParty : ThirdParty;
2015-01-21 01:39:13 +01:00
// Be prepared to support unknown types
2015-03-26 00:28:22 +01:00
var type = typeNameToTypeValue[requestType] || 0;
if ( type === 0 ) {
return '';
}
var categories = this.categories;
2015-01-23 17:32:49 +01:00
var bf = false, bucket;
// Tokenize only once
this.tokenize(url);
2014-08-29 21:02:31 +02:00
// https://github.com/gorhill/uBlock/issues/139
// Test against important block filters
if ( bucket = categories[this.makeCategoryKey(BlockAnyParty | Important | type)] ) {
bf = this.matchTokens(bucket, url);
if ( bf !== false ) {
return 'sb:' + bf.toString();
}
}
if ( bucket = categories[this.makeCategoryKey(BlockAction | Important | type | party)] ) {
bf = this.matchTokens(bucket, url);
if ( bf !== false ) {
return 'sb:' + bf.toString();
}
2014-08-29 21:02:31 +02:00
}
// Test against block filters
if ( bucket = categories[this.makeCategoryKey(BlockAnyParty | type)] ) {
bf = this.matchTokens(bucket, url);
}
if ( bf === false ) {
if ( bucket = categories[this.makeCategoryKey(BlockAction | type | party)] ) {
bf = this.matchTokens(bucket, url);
}
}
2014-08-29 21:02:31 +02:00
// If there is no block filter, no need to test against allow filters
2014-08-28 15:59:05 +02:00
if ( bf === false ) {
2014-09-14 22:20:40 +02:00
return '';
}
// Test against allow filters
var af;
if ( bucket = categories[this.makeCategoryKey(AllowAnyParty | type)] ) {
af = this.matchTokens(bucket, url);
if ( af !== false ) {
return 'sa:' + af.toString();
}
}
if ( bucket = categories[this.makeCategoryKey(AllowAction | type | party)] ) {
af = this.matchTokens(bucket, url);
if ( af !== false ) {
return 'sa:' + af.toString();
}
}
return 'sb:' + bf.toString();
};
/******************************************************************************/
FilterContainer.prototype.matchString = function(context) {
2015-01-17 13:53:19 +01:00
// https://github.com/gorhill/uBlock/issues/519
// Use exact type match for anything beyond `other`
2015-01-21 01:39:13 +01:00
// Also, be prepared to support unknown types
2015-01-24 14:21:14 +01:00
var type = typeNameToTypeValue[context.requestType] || typeOtherValue;
2015-03-27 18:00:55 +01:00
if ( type > typeOtherValue ) {
2015-01-17 13:53:19 +01:00
return this.matchStringExactType(context, context.requestURL, context.requestType);
}
2014-06-24 00:42:43 +02:00
// https://github.com/gorhill/httpswitchboard/issues/239
// Convert url to lower case:
// `match-case` option not supported, but then, I saw only one
// occurrence of it in all the supported lists (bulgaria list).
var url = context.requestURL.toLowerCase();
2014-06-24 00:42:43 +02:00
// The logic here is simple:
//
// block = !whitelisted && blacklisted
// or equivalent
// allow = whitelisted || !blacklisted
2014-06-28 17:40:26 +02:00
// Statistically, hits on a URL in order of likelihood:
// 1. No hit
// 2. Hit on a block filter
// 3. Hit on an allow filter
//
// High likelihood of "no hit" means to optimize we need to reduce as much
// as possible the number of filters to test.
//
// Then, because of the order of probabilities, we should test only
// block filters first, and test allow filters if and only if there is a
2014-06-28 17:40:26 +02:00
// hit on a block filter. Since there is a high likelihood of no hit,
// testing allow filter by default is likely wasted work, hence allow
2014-06-28 17:41:49 +02:00
// filters are tested *only* if there is a (unlikely) hit on a block
// filter.
2014-06-24 00:42:43 +02:00
2015-02-05 00:06:31 +01:00
// These registers will be used by various filters
pageHostnameRegister = context.pageHostname || '';
requestHostnameRegister = context.requestHostname;
2015-02-05 00:06:31 +01:00
var party = isFirstParty(context.pageDomain, context.requestHostname) ? FirstParty : ThirdParty;
var filterClasses = this.categories;
var bucket;
// Tokenize only once
this.tokenize(url);
2015-02-05 00:06:31 +01:00
var bf = false;
2015-01-21 14:59:23 +01:00
2014-08-29 21:02:31 +02:00
// https://github.com/gorhill/uBlock/issues/139
// Test against important block filters.
// The purpose of the `important` option is to reverse the order of
// evaluation. Normally, it is "evaluate block then evaluate allow", with
// the `important` property it is "evaluate allow then evaluate block".
2015-02-05 00:06:31 +01:00
if ( bucket = filterClasses[this.makeCategoryKey(BlockAnyTypeAnyParty | Important)] ) {
bf = this.matchTokens(bucket, url);
if ( bf !== false ) {
return 'sb:' + bf.toString() + '$important';
}
}
2015-02-05 00:06:31 +01:00
if ( bucket = filterClasses[this.makeCategoryKey(BlockAnyType | Important | party)] ) {
bf = this.matchTokens(bucket, url);
if ( bf !== false ) {
return 'sb:' + bf.toString() + '$important';
}
}
2015-02-05 00:06:31 +01:00
if ( bucket = filterClasses[this.makeCategoryKey(BlockAnyParty | Important | type)] ) {
bf = this.matchTokens(bucket, url);
if ( bf !== false ) {
return 'sb:' + bf.toString() + '$important';
}
}
2015-02-05 00:06:31 +01:00
if ( bucket = filterClasses[this.makeCategoryKey(BlockAction | Important | type | party)] ) {
bf = this.matchTokens(bucket, url);
if ( bf !== false ) {
return 'sb:' + bf.toString() + '$important';
}
2014-08-29 21:02:31 +02:00
}
2014-06-24 00:42:43 +02:00
// Test against block filters
2014-08-28 15:59:05 +02:00
if ( bf === false ) {
2015-02-05 00:06:31 +01:00
if ( bucket = filterClasses[this.makeCategoryKey(BlockAnyTypeAnyParty)] ) {
bf = this.matchTokens(bucket, url);
}
}
if ( bf === false ) {
2015-02-05 00:06:31 +01:00
if ( bucket = filterClasses[this.makeCategoryKey(BlockAnyType | party)] ) {
bf = this.matchTokens(bucket, url);
}
}
if ( bf === false ) {
2015-02-05 00:06:31 +01:00
if ( bucket = filterClasses[this.makeCategoryKey(BlockAnyParty | type)] ) {
bf = this.matchTokens(bucket, url);
}
}
if ( bf === false ) {
2015-02-05 00:06:31 +01:00
if ( bucket = filterClasses[this.makeCategoryKey(BlockAction | type | party)] ) {
bf = this.matchTokens(bucket, url);
}
2014-06-24 00:42:43 +02:00
}
// If there is no block filter, no need to test against allow filters
2014-08-28 15:59:05 +02:00
if ( bf === false ) {
2014-09-14 22:20:40 +02:00
return '';
2014-06-24 00:42:43 +02:00
}
// Test against allow filters
var af;
2015-02-05 00:06:31 +01:00
if ( bucket = filterClasses[this.makeCategoryKey(AllowAnyTypeAnyParty)] ) {
af = this.matchTokens(bucket, url);
if ( af !== false ) {
return 'sa:' + af.toString();
}
}
2015-02-05 00:06:31 +01:00
if ( bucket = filterClasses[this.makeCategoryKey(AllowAnyType | party)] ) {
af = this.matchTokens(bucket, url);
if ( af !== false ) {
return 'sa:' + af.toString();
}
}
2015-02-05 00:06:31 +01:00
if ( bucket = filterClasses[this.makeCategoryKey(AllowAnyParty | type)] ) {
af = this.matchTokens(bucket, url);
if ( af !== false ) {
return 'sa:' + af.toString();
}
}
2015-02-05 00:06:31 +01:00
if ( bucket = filterClasses[this.makeCategoryKey(AllowAction | type | party)] ) {
af = this.matchTokens(bucket, url);
if ( af !== false ) {
return 'sa:' + af.toString();
}
2014-06-24 00:42:43 +02:00
}
return 'sb:' + bf.toString();
2014-06-24 00:42:43 +02:00
};
/******************************************************************************/
FilterContainer.prototype.getFilterCount = function() {
2015-02-24 00:31:29 +01:00
return this.acceptedCount - this.duplicateCount;
2014-06-24 00:42:43 +02:00
};
/******************************************************************************/
return new FilterContainer();
/******************************************************************************/
})();