2014-06-24 00:42:43 +02:00
|
|
|
/*******************************************************************************
|
|
|
|
|
2015-12-29 17:34:41 +01:00
|
|
|
uBlock Origin - a browser extension to block requests.
|
2016-08-13 22:42:58 +02:00
|
|
|
Copyright (C) 2014-2016 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
|
|
|
|
*/
|
|
|
|
|
2014-10-19 13:11:27 +02:00
|
|
|
'use strict';
|
2014-06-24 00:42:43 +02:00
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2015-12-29 17:34:41 +01:00
|
|
|
// A standalone URL tokenizer will allow us to use URL tokens in more than
|
|
|
|
// just static filtering engine. This opens the door to optimize other
|
|
|
|
// filtering engine parts aside static filtering. This also allows:
|
|
|
|
// - Tokenize only on demand.
|
|
|
|
// - To potentially avoid tokenizing when same URL is fed to tokenizer.
|
|
|
|
// - Benchmarking shows this to be a common occurrence.
|
|
|
|
|
|
|
|
µBlock.urlTokenizer = {
|
|
|
|
setURL: function(url) {
|
|
|
|
if ( url !== this._urlIn ) {
|
|
|
|
this._urlIn = url;
|
|
|
|
this._urlOut = url.toLowerCase();
|
|
|
|
this._tokenized = false;
|
|
|
|
}
|
|
|
|
return this._urlOut;
|
|
|
|
},
|
|
|
|
|
|
|
|
// Tokenize on demand.
|
|
|
|
getTokens: function() {
|
|
|
|
if ( this._tokenized === false ) {
|
2016-10-11 17:53:28 +02:00
|
|
|
if ( this._gcAfter === undefined ) {
|
|
|
|
this._gcAfter = Date.now() + 1499;
|
|
|
|
}
|
2015-12-29 17:34:41 +01:00
|
|
|
this._tokenize();
|
|
|
|
this._tokenized = true;
|
|
|
|
}
|
|
|
|
return this._tokens;
|
|
|
|
},
|
|
|
|
|
|
|
|
isTokenized: function() {
|
|
|
|
return this._tokens !== null && this._tokens[0].token !== '';
|
|
|
|
},
|
|
|
|
|
|
|
|
_Entry: function() {
|
|
|
|
this.beg = 0;
|
2016-10-11 17:53:28 +02:00
|
|
|
this.token = undefined;
|
2015-12-29 17:34:41 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
// https://github.com/chrisaljoudi/uBlock/issues/1118
|
|
|
|
// We limit to a maximum number of tokens.
|
|
|
|
_init: function() {
|
|
|
|
this._tokens = new Array(2048);
|
|
|
|
for ( var i = 0; i < 2048; i++ ) {
|
|
|
|
this._tokens[i] = new this._Entry();
|
|
|
|
}
|
|
|
|
this._init = null;
|
|
|
|
},
|
|
|
|
|
|
|
|
_tokenize: function() {
|
|
|
|
var tokens = this._tokens,
|
|
|
|
re = this._reAnyToken,
|
|
|
|
url = this._urlOut;
|
|
|
|
var matches, entry;
|
|
|
|
re.lastIndex = 0;
|
|
|
|
for ( var i = 0; i < 2047; i++ ) {
|
|
|
|
matches = re.exec(url);
|
2016-10-11 17:53:28 +02:00
|
|
|
if ( matches === null ) { break; }
|
2015-12-29 17:34:41 +01:00
|
|
|
entry = tokens[i];
|
|
|
|
entry.beg = matches.index;
|
|
|
|
entry.token = matches[0];
|
|
|
|
}
|
|
|
|
tokens[i].token = ''; // Sentinel
|
2016-10-11 17:53:28 +02:00
|
|
|
// Could no-longer-used-but-still-referenced string fragments
|
|
|
|
// contribute to memory fragmentation in the long-term? The code below
|
|
|
|
// is to address this: left over unused string fragments are removed at
|
|
|
|
// regular interval.
|
|
|
|
if ( Date.now() < this._gcAfter ) { return; }
|
|
|
|
this._gcAfter = undefined;
|
|
|
|
for ( i += 1; i < 2047; i++ ) {
|
|
|
|
entry = tokens[i];
|
|
|
|
if ( entry.token === undefined ) { break; }
|
|
|
|
entry.token = undefined;
|
|
|
|
}
|
2015-12-29 17:34:41 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
_urlIn: '',
|
|
|
|
_urlOut: '',
|
|
|
|
_tokenized: false,
|
|
|
|
_tokens: null,
|
2016-10-11 17:53:28 +02:00
|
|
|
_reAnyToken: /[%0-9a-z]+/g,
|
|
|
|
_gcAfter: undefined
|
2015-12-29 17:34:41 +01:00
|
|
|
};
|
2014-06-24 00:42:43 +02:00
|
|
|
|
2015-12-29 17:34:41 +01:00
|
|
|
µBlock.urlTokenizer._init();
|
2014-08-20 02:41:52 +02:00
|
|
|
|
2014-06-24 00:42:43 +02:00
|
|
|
/******************************************************************************/
|
|
|
|
|
2015-12-29 17:34:41 +01:00
|
|
|
µBlock.formatCount = function(count) {
|
2014-08-20 02:41:52 +02:00
|
|
|
if ( typeof count !== 'number' ) {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
var s = count.toFixed(0);
|
|
|
|
if ( count >= 1000 ) {
|
|
|
|
if ( count < 10000 ) {
|
2014-12-24 14:11:22 +01:00
|
|
|
s = '>' + s.slice(0,1) + 'k';
|
2014-08-20 02:41:52 +02:00
|
|
|
} else if ( count < 100000 ) {
|
2014-12-24 14:11:22 +01:00
|
|
|
s = s.slice(0,2) + 'k';
|
2014-08-20 02:41:52 +02:00
|
|
|
} else if ( count < 1000000 ) {
|
2014-12-24 14:11:22 +01:00
|
|
|
s = s.slice(0,3) + 'k';
|
2014-08-20 02:41:52 +02:00
|
|
|
} else if ( count < 10000000 ) {
|
|
|
|
s = s.slice(0,1) + 'M';
|
|
|
|
} else {
|
|
|
|
s = s.slice(0,-6) + 'M';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return s;
|
2014-06-24 00:42:43 +02:00
|
|
|
};
|
|
|
|
|
2014-08-20 15:24:16 +02:00
|
|
|
// https://www.youtube.com/watch?v=DyvzfyqYm_s
|
2014-08-20 02:41:52 +02:00
|
|
|
|
|
|
|
/******************************************************************************/
|
2016-08-13 22:42:58 +02:00
|
|
|
|
2016-10-13 19:25:57 +02:00
|
|
|
µBlock.dateNowToSensibleString = function() {
|
|
|
|
var now = new Date(Date.now() - (new Date()).getTimezoneOffset() * 60000);
|
|
|
|
return now.toISOString().replace(/\.\d+Z$/, '')
|
|
|
|
.replace(/:/g, '.')
|
|
|
|
.replace('T', '_');
|
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2016-08-13 22:42:58 +02:00
|
|
|
µBlock.LineIterator = function(text, offset) {
|
|
|
|
this.text = text;
|
2016-09-12 16:22:25 +02:00
|
|
|
this.textLen = this.text.length;
|
2016-08-13 22:42:58 +02:00
|
|
|
this.offset = offset || 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
µBlock.LineIterator.prototype.next = function() {
|
|
|
|
var lineEnd = this.text.indexOf('\n', this.offset);
|
|
|
|
if ( lineEnd === -1 ) {
|
|
|
|
lineEnd = this.text.indexOf('\r', this.offset);
|
|
|
|
if ( lineEnd === -1 ) {
|
2016-09-12 16:22:25 +02:00
|
|
|
lineEnd = this.textLen;
|
2016-08-13 22:42:58 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
var line = this.text.slice(this.offset, lineEnd);
|
|
|
|
this.offset = lineEnd + 1;
|
|
|
|
return line;
|
|
|
|
};
|
|
|
|
|
2017-03-11 19:55:47 +01:00
|
|
|
µBlock.LineIterator.prototype.rewind = function() {
|
|
|
|
if ( this.offset <= 1 ) {
|
|
|
|
this.offset = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
var lineEnd = this.text.lastIndexOf('\n', this.offset - 2);
|
|
|
|
if ( lineEnd !== -1 ) {
|
|
|
|
this.offset = lineEnd + 1;
|
|
|
|
} else {
|
|
|
|
lineEnd = this.text.lastIndexOf('\r', this.offset - 2);
|
|
|
|
this.offset = lineEnd !== -1 ? lineEnd + 1 : 0;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-08-13 22:42:58 +02:00
|
|
|
µBlock.LineIterator.prototype.eot = function() {
|
2016-09-12 16:22:25 +02:00
|
|
|
return this.offset >= this.textLen;
|
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
|
|
|
// The field iterator is less CPU-intensive than when using native
|
|
|
|
// String.split().
|
|
|
|
|
|
|
|
µBlock.FieldIterator = function(sep) {
|
|
|
|
this.text = '';
|
|
|
|
this.sep = sep;
|
|
|
|
this.sepLen = sep.length;
|
|
|
|
this.offset = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
µBlock.FieldIterator.prototype.first = function(text) {
|
|
|
|
this.text = text;
|
|
|
|
this.offset = 0;
|
|
|
|
return this.next();
|
|
|
|
};
|
|
|
|
|
|
|
|
µBlock.FieldIterator.prototype.next = function() {
|
|
|
|
var end = this.text.indexOf(this.sep, this.offset);
|
|
|
|
if ( end === -1 ) {
|
|
|
|
end = this.text.length;
|
|
|
|
}
|
|
|
|
var field = this.text.slice(this.offset, end);
|
|
|
|
this.offset = end + this.sepLen;
|
|
|
|
return field;
|
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2017-03-11 19:55:47 +01:00
|
|
|
µBlock.mapToArray = typeof Array.from === 'function'
|
|
|
|
? Array.from
|
|
|
|
: function(map) {
|
|
|
|
var out = [],
|
|
|
|
entries = map.entries(),
|
|
|
|
entry;
|
|
|
|
for (;;) {
|
|
|
|
entry = entries.next();
|
|
|
|
if ( entry.done ) { break; }
|
|
|
|
out.push([ entry.value[0], entry.value[1] ]);
|
|
|
|
}
|
|
|
|
return out;
|
|
|
|
};
|
2016-09-12 16:22:25 +02:00
|
|
|
|
|
|
|
µBlock.mapFromArray = function(arr) {
|
|
|
|
return new Map(arr);
|
|
|
|
};
|
|
|
|
|
2017-03-11 19:55:47 +01:00
|
|
|
µBlock.setToArray = typeof Array.from === 'function'
|
|
|
|
? Array.from
|
|
|
|
: function(dict) {
|
|
|
|
var out = [],
|
|
|
|
entries = dict.values(),
|
|
|
|
entry;
|
|
|
|
for (;;) {
|
|
|
|
entry = entries.next();
|
|
|
|
if ( entry.done ) { break; }
|
|
|
|
out.push(entry.value);
|
|
|
|
}
|
|
|
|
return out;
|
|
|
|
};
|
2016-09-12 16:22:25 +02:00
|
|
|
|
|
|
|
µBlock.setFromArray = function(arr) {
|
|
|
|
return new Set(arr);
|
2016-08-13 22:42:58 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
2016-09-16 23:41:17 +02:00
|
|
|
|
|
|
|
µBlock.openNewTab = function(details) {
|
|
|
|
if ( details.url.startsWith('logger-ui.html') ) {
|
|
|
|
if ( details.shiftKey ) {
|
2016-09-17 01:12:16 +02:00
|
|
|
this.changeUserSettings(
|
|
|
|
'alwaysDetachLogger',
|
|
|
|
!this.userSettings.alwaysDetachLogger
|
|
|
|
);
|
2016-09-16 23:41:17 +02:00
|
|
|
}
|
|
|
|
details.popup = this.userSettings.alwaysDetachLogger;
|
|
|
|
}
|
|
|
|
vAPI.tabs.open(details);
|
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
2017-01-27 19:44:52 +01:00
|
|
|
|
|
|
|
// https://github.com/gorhill/uBlock/issues/2344
|
|
|
|
|
|
|
|
µBlock.matchCurrentLanguage = function(s) {
|
|
|
|
if ( typeof s !== 'string' ) { return false; }
|
|
|
|
if ( this.matchCurrentLanguage.reLang === undefined ) {
|
|
|
|
this.matchCurrentLanguage.reLang = new RegExp('\\b' + self.navigator.language.slice(0, 2) + '\\b');
|
|
|
|
}
|
|
|
|
return this.matchCurrentLanguage.reLang.test(s);
|
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|