2017-09-02 04:11:33 -06:00
|
|
|
/*******************************************************************************
|
|
|
|
|
|
|
|
uBlock Origin - a browser extension to block requests.
|
2018-12-13 10:30:54 -07:00
|
|
|
Copyright (C) 2017-present Raymond Hill
|
2017-09-02 04:11:33 -06: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
|
|
|
|
*/
|
|
|
|
|
|
|
|
// For background page
|
|
|
|
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2019-06-30 08:09:27 -06:00
|
|
|
(( ) => {
|
2019-01-28 14:12:26 -07:00
|
|
|
// https://github.com/uBlockOrigin/uBlock-issues/issues/407
|
|
|
|
if ( vAPI.webextFlavor.soup.has('firefox') === false ) { return; }
|
2017-09-02 04:11:33 -06:00
|
|
|
|
|
|
|
// https://github.com/gorhill/uBlock/issues/2950
|
2018-04-05 05:29:15 -06:00
|
|
|
// Firefox 56 does not normalize URLs to ASCII, uBO must do this itself.
|
2017-09-02 04:11:33 -06:00
|
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=945240
|
2019-06-30 08:09:27 -06:00
|
|
|
const evalMustPunycode = ( ) => {
|
2018-04-05 05:29:15 -06:00
|
|
|
return vAPI.webextFlavor.soup.has('firefox') &&
|
|
|
|
vAPI.webextFlavor.major < 57;
|
2018-04-04 10:42:01 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
let mustPunycode = evalMustPunycode();
|
|
|
|
|
|
|
|
// The real actual webextFlavor value may not be set in stone, so listen
|
|
|
|
// for possible future changes.
|
2018-12-16 08:51:25 -07:00
|
|
|
window.addEventListener('webextFlavor', ( ) => {
|
2018-04-04 10:42:01 -06:00
|
|
|
mustPunycode = evalMustPunycode();
|
|
|
|
}, { once: true });
|
2017-09-02 04:11:33 -06:00
|
|
|
|
2018-12-16 08:51:25 -07:00
|
|
|
const punycode = self.punycode;
|
|
|
|
const reAsciiHostname = /^https?:\/\/[0-9a-z_.:@-]+[/?#]/;
|
|
|
|
const parsedURL = new URL('about:blank');
|
2017-09-02 04:11:33 -06:00
|
|
|
|
2019-06-30 08:09:27 -06:00
|
|
|
// Related issues:
|
|
|
|
// - https://github.com/gorhill/uBlock/issues/1327
|
|
|
|
// - https://github.com/uBlockOrigin/uBlock-issues/issues/128
|
|
|
|
// - https://bugzilla.mozilla.org/show_bug.cgi?id=1503721
|
2017-09-02 04:11:33 -06:00
|
|
|
|
2019-06-30 08:09:27 -06:00
|
|
|
// Extend base class to normalize as per platform.
|
2017-09-02 04:11:33 -06:00
|
|
|
|
2019-06-30 08:09:27 -06:00
|
|
|
vAPI.Net = class extends vAPI.Net {
|
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
this.pendingRequests = [];
|
2019-11-20 08:45:17 -07:00
|
|
|
this.cnames = new Map([ [ '', '' ] ]);
|
2019-11-19 10:05:33 -07:00
|
|
|
this.cnameAliasList = null;
|
|
|
|
this.cnameIgnoreList = null;
|
|
|
|
this.url = new URL(vAPI.getURL('/'));
|
|
|
|
this.cnameMaxTTL = 60;
|
|
|
|
this.cnameTimer = undefined;
|
|
|
|
}
|
|
|
|
setOptions(options) {
|
|
|
|
super.setOptions(options);
|
|
|
|
this.cnameAliasList = this.regexFromStrList(options.cnameAliasList);
|
|
|
|
this.cnameIgnoreList = this.regexFromStrList(options.cnameIgnoreList);
|
|
|
|
this.cnameIgnore1stParty = options.cnameIgnore1stParty === true;
|
|
|
|
this.cnameMaxTTL = options.cnameMaxTTL || 120;
|
2019-11-20 08:45:17 -07:00
|
|
|
this.cnames.clear(); this.cnames.set('', '');
|
2017-09-02 04:11:33 -06:00
|
|
|
}
|
2019-06-30 08:09:27 -06:00
|
|
|
normalizeDetails(details) {
|
|
|
|
if ( mustPunycode && !reAsciiHostname.test(details.url) ) {
|
|
|
|
parsedURL.href = details.url;
|
|
|
|
details.url = details.url.replace(
|
|
|
|
parsedURL.hostname,
|
|
|
|
punycode.toASCII(parsedURL.hostname)
|
|
|
|
);
|
|
|
|
}
|
2017-09-02 04:11:33 -06:00
|
|
|
|
2019-06-30 08:09:27 -06:00
|
|
|
const type = details.type;
|
2018-12-16 08:51:25 -07:00
|
|
|
|
2019-06-30 08:09:27 -06:00
|
|
|
if ( type === 'imageset' ) {
|
|
|
|
details.type = 'image';
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// https://github.com/uBlockOrigin/uBlock-issues/issues/345
|
|
|
|
// Re-categorize an embedded object as a `sub_frame` if its
|
|
|
|
// content type is that of a HTML document.
|
|
|
|
if ( type === 'object' && Array.isArray(details.responseHeaders) ) {
|
|
|
|
for ( const header of details.responseHeaders ) {
|
|
|
|
if ( header.name.toLowerCase() === 'content-type' ) {
|
|
|
|
if ( header.value.startsWith('text/html') ) {
|
|
|
|
details.type = 'sub_frame';
|
|
|
|
}
|
|
|
|
break;
|
2018-12-16 08:51:25 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-06-30 08:09:27 -06:00
|
|
|
denormalizeTypes(types) {
|
|
|
|
if ( types.length === 0 ) {
|
|
|
|
return Array.from(this.validTypes);
|
2017-09-02 04:11:33 -06:00
|
|
|
}
|
2019-06-30 08:09:27 -06:00
|
|
|
const out = new Set();
|
|
|
|
for ( const type of types ) {
|
|
|
|
if ( this.validTypes.has(type) ) {
|
|
|
|
out.add(type);
|
|
|
|
}
|
|
|
|
if ( type === 'image' && this.validTypes.has('imageset') ) {
|
|
|
|
out.add('imageset');
|
|
|
|
}
|
|
|
|
if ( type === 'sub_frame' ) {
|
|
|
|
out.add('object');
|
|
|
|
}
|
2017-09-02 04:11:33 -06:00
|
|
|
}
|
2019-06-30 08:09:27 -06:00
|
|
|
return Array.from(out);
|
2017-09-02 04:11:33 -06:00
|
|
|
}
|
2019-11-19 10:05:33 -07:00
|
|
|
processCanonicalName(cname, details) {
|
|
|
|
this.url.href = details.url;
|
|
|
|
details.cnameOf = this.url.hostname;
|
|
|
|
this.url.hostname = cname;
|
|
|
|
details.url = this.url.href;
|
|
|
|
return super.onBeforeSuspendableRequest(details);
|
|
|
|
}
|
|
|
|
recordCanonicalName(hn, record) {
|
|
|
|
let cname =
|
|
|
|
typeof record.canonicalName === 'string' &&
|
|
|
|
record.canonicalName !== hn
|
|
|
|
? record.canonicalName
|
|
|
|
: '';
|
|
|
|
if (
|
|
|
|
cname !== '' &&
|
|
|
|
this.cnameIgnore1stParty &&
|
|
|
|
vAPI.domainFromHostname(cname) === vAPI.domainFromHostname(hn)
|
|
|
|
) {
|
|
|
|
cname = '';
|
|
|
|
}
|
|
|
|
if (
|
|
|
|
cname !== '' &&
|
|
|
|
this.cnameIgnoreList !== null &&
|
|
|
|
this.cnameIgnoreList.test(cname)
|
|
|
|
) {
|
|
|
|
cname = '';
|
|
|
|
}
|
|
|
|
this.cnames.set(hn, cname);
|
|
|
|
if ( this.cnameTimer === undefined ) {
|
|
|
|
this.cnameTimer = self.setTimeout(
|
|
|
|
( ) => {
|
|
|
|
this.cnameTimer = undefined;
|
2019-11-20 08:45:17 -07:00
|
|
|
this.cnames.clear(); this.cnames.set('', '');
|
2019-11-19 10:05:33 -07:00
|
|
|
},
|
|
|
|
this.cnameMaxTTL * 60000
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return cname;
|
|
|
|
}
|
|
|
|
regexFromStrList(list) {
|
|
|
|
if (
|
|
|
|
typeof list !== 'string' ||
|
|
|
|
list.length === 0 ||
|
2019-11-19 14:48:53 -07:00
|
|
|
list === 'unset' ||
|
|
|
|
browser.dns instanceof Object === false
|
2019-11-19 10:05:33 -07:00
|
|
|
) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
if ( list === '*' ) {
|
|
|
|
return /^./;
|
|
|
|
}
|
|
|
|
return new RegExp(
|
|
|
|
'(?:^|\.)(?:' +
|
|
|
|
list.trim()
|
|
|
|
.split(/\s+/)
|
|
|
|
.map(a => a.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'))
|
|
|
|
.join('|') +
|
|
|
|
')$'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
onBeforeSuspendableRequest(details) {
|
|
|
|
let r = super.onBeforeSuspendableRequest(details);
|
|
|
|
if ( r !== undefined ) { return r; }
|
|
|
|
if ( this.cnameAliasList === null ) { return; }
|
2019-11-20 08:45:17 -07:00
|
|
|
const hn = vAPI.hostnameFromNetworkURL(details.url);
|
2019-11-19 10:05:33 -07:00
|
|
|
let cname = this.cnames.get(hn);
|
|
|
|
if ( cname === '' ) { return; }
|
|
|
|
if ( cname !== undefined ) {
|
|
|
|
return this.processCanonicalName(cname, details);
|
|
|
|
}
|
|
|
|
if ( this.cnameAliasList.test(hn) === false ) {
|
|
|
|
this.cnames.set(hn, '');
|
|
|
|
return;
|
|
|
|
}
|
2019-11-21 10:04:19 -07:00
|
|
|
return browser.dns.resolve(hn, [ 'canonical_name' ]).then(
|
|
|
|
rec => {
|
|
|
|
const cname = this.recordCanonicalName(hn, rec);
|
|
|
|
if ( cname === '' ) { return; }
|
|
|
|
return this.processCanonicalName(cname, details);
|
|
|
|
},
|
|
|
|
( ) => {
|
|
|
|
this.cnames.set(hn, '');
|
|
|
|
}
|
|
|
|
);
|
2019-11-19 10:05:33 -07:00
|
|
|
}
|
2019-06-30 08:09:27 -06:00
|
|
|
suspendOneRequest(details) {
|
|
|
|
const pending = {
|
|
|
|
details: Object.assign({}, details),
|
|
|
|
resolve: undefined,
|
|
|
|
promise: undefined
|
|
|
|
};
|
2019-10-07 06:13:37 -06:00
|
|
|
pending.promise = new Promise(resolve => {
|
2019-06-30 08:09:27 -06:00
|
|
|
pending.resolve = resolve;
|
|
|
|
});
|
|
|
|
this.pendingRequests.push(pending);
|
|
|
|
return pending.promise;
|
|
|
|
}
|
2019-11-19 10:05:33 -07:00
|
|
|
unsuspendAllRequests() {
|
2019-06-30 08:09:27 -06:00
|
|
|
const pendingRequests = this.pendingRequests;
|
|
|
|
this.pendingRequests = [];
|
|
|
|
for ( const entry of pendingRequests ) {
|
2019-11-19 10:05:33 -07:00
|
|
|
entry.resolve(this.onBeforeSuspendableRequest(entry.details));
|
2018-12-23 15:59:31 -07:00
|
|
|
}
|
2019-06-30 08:09:27 -06:00
|
|
|
}
|
|
|
|
canSuspend() {
|
|
|
|
return true;
|
|
|
|
}
|
2018-12-23 15:59:31 -07:00
|
|
|
};
|
|
|
|
})();
|
|
|
|
|
|
|
|
/******************************************************************************/
|