uBlock/src/js/tab.js

321 lines
10 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
*/
2015-01-02 19:42:35 +01:00
/* global vAPI, µBlock */
/******************************************************************************/
/******************************************************************************/
(function() {
'use strict';
2014-06-24 00:42:43 +02:00
/******************************************************************************/
2015-01-02 19:42:35 +01:00
var µb = µBlock;
/******************************************************************************/
/******************************************************************************/
// When the DOM content of root frame is loaded, this means the tab
// content has changed.
2015-03-08 16:06:36 +01:00
vAPI.tabs.onNavigation = function(details) {
if ( details.frameId !== 0 ) {
return;
}
2015-01-28 22:19:46 +01:00
var pageStore = µb.bindTabToPageStats(details.tabId, details.url, 'afterNavigate');
// https://github.com/gorhill/uBlock/issues/630
// The hostname of the bound document must always be present in the
// mini-matrix. That's the best place I could find for the fix, all other
// options had bad side-effects or complications.
2015-03-15 18:14:52 +01:00
// TODO: Eventually, we will have to use an API to check whether a scheme
2015-01-28 22:19:46 +01:00
// is supported as I suspect we are going to start to see `ws`, `wss`
// as well soon.
if ( pageStore && details.url.lastIndexOf('http', 0) === 0 ) {
pageStore.hostnameToCountMap[pageStore.pageHostname] = 0;
}
};
2014-09-08 19:45:03 +02:00
2015-03-08 16:06:36 +01:00
/******************************************************************************/
// It may happen the URL in the tab changes, while the page's document
// stays the same (for instance, Google Maps). Without this listener,
// the extension icon won't be properly refreshed.
2015-03-08 16:06:36 +01:00
vAPI.tabs.onUpdated = function(tabId, changeInfo, tab) {
if ( !tab.url || tab.url === '' ) {
return;
}
if ( !changeInfo.url ) {
return;
}
2015-01-02 19:42:35 +01:00
µb.bindTabToPageStats(tabId, changeInfo.url, 'tabUpdated');
};
2014-06-24 00:42:43 +02:00
2015-03-08 16:06:36 +01:00
/******************************************************************************/
vAPI.tabs.onClosed = function(tabId) {
if ( tabId < 0 ) {
return;
}
2015-01-02 19:42:35 +01:00
µb.unbindTabFromPageStats(tabId);
};
2014-10-15 19:14:25 +02:00
2015-03-08 16:06:36 +01:00
/******************************************************************************/
// https://github.com/gorhill/uBlock/issues/297
2015-03-08 16:06:36 +01:00
vAPI.tabs.onPopup = function(details) {
2015-03-09 15:19:00 +01:00
//console.debug('vAPI.tabs.onPopup: details = %o', details);
2015-03-08 16:06:36 +01:00
var pageStore = µb.pageStoreFromTabId(details.openerTabId);
var openerURL = details.openerURL || '';
if ( openerURL === '' && pageStore ) {
openerURL = pageStore.pageURL;
}
if ( openerURL === '' ) {
return;
}
2015-03-08 16:06:36 +01:00
var µburi = µb.URI;
var openerHostname = µburi.hostnameFromURI(openerURL);
var openerDomain = µburi.domainFromHostname(openerHostname);
var targetURL = details.targetURL;
2015-03-27 18:00:55 +01:00
// If the page URL is that of our "blocked page" URL, extract the URL of
// the page which was blocked.
if ( targetURL.lastIndexOf(vAPI.getURL('document-blocked.html'), 0) === 0 ) {
var matches = /details=([^&]+)/.exec(targetURL);
if ( matches !== null ) {
targetURL = JSON.parse(atob(matches[1])).url;
}
}
2015-03-27 19:59:17 +01:00
var context = {
pageHostname: openerHostname,
pageDomain: openerDomain,
rootHostname: openerHostname,
rootDomain: openerDomain,
requestURL: targetURL,
requestHostname: µb.URI.hostnameFromURI(targetURL),
requestType: 'popup'
};
var result = '';
2015-03-27 18:00:55 +01:00
// Check user switch first
if ( µb.hnSwitches.evaluateZ('doBlockAllPopups', openerHostname) ) {
result = 'ub:doBlockAllPopups true';
}
// https://github.com/gorhill/uBlock/issues/323
2015-03-31 22:40:52 +02:00
// https://github.com/gorhill/uBlock/issues/1142
// If popup OR opener URL is whitelisted, do not block the popup
2015-03-31 22:38:47 +02:00
if (
result === '' &&
µb.getNetFilteringSwitch(openerURL) &&
µb.getNetFilteringSwitch(targetURL)
) {
2015-03-08 16:06:36 +01:00
result = µb.staticNetFilteringEngine.matchStringExactType(context, targetURL, 'popup');
}
2014-10-15 19:14:25 +02:00
// https://github.com/gorhill/uBlock/issues/91
2015-03-08 16:06:36 +01:00
if ( pageStore ) {
2015-03-27 19:59:17 +01:00
pageStore.logRequest(context, result);
}
2014-10-15 19:14:25 +02:00
// Not blocked
2015-01-22 19:00:59 +01:00
if ( µb.isAllowResult(result) ) {
return;
}
// Blocked
// It is a popup, block and remove the tab.
2015-03-08 16:06:36 +01:00
µb.unbindTabFromPageStats(details.targetTabId);
vAPI.tabs.remove(details.targetTabId);
return true;
};
vAPI.tabs.registerListeners();
2014-10-15 19:14:25 +02:00
2014-06-24 00:42:43 +02:00
/******************************************************************************/
/******************************************************************************/
// https://github.com/gorhill/httpswitchboard/issues/303
// Some kind of trick going on here:
// Any scheme other than 'http' and 'https' is remapped into a fake
// URL which trick the rest of µBlock into being able to process an
// otherwise unmanageable scheme. µBlock needs web page to have a proper
// hostname to work properly, so just like the 'chromium-behind-the-scene'
// fake domain name, we map unknown schemes into a fake '{scheme}-scheme'
// hostname. This way, for a specific scheme you can create scope with
// rules which will apply only to that scheme.
µb.normalizePageURL = function(tabId, pageURL) {
if ( vAPI.isNoTabId(tabId) ) {
return 'http://behind-the-scene/';
}
2014-06-24 00:42:43 +02:00
var uri = this.URI.set(pageURL);
var scheme = uri.scheme;
if ( scheme === 'https' || scheme === 'http' ) {
2014-06-24 00:42:43 +02:00
return uri.normalizedURI();
}
2015-01-24 18:06:22 +01:00
var url = 'http://' + scheme + '-scheme/';
if ( uri.hostname !== '' ) {
2015-01-24 18:06:22 +01:00
url += uri.hostname + '/';
}
2015-01-24 18:06:22 +01:00
return url;
2014-06-27 23:06:42 +02:00
};
2014-06-24 00:42:43 +02:00
/******************************************************************************/
// Create an entry for the tab if it doesn't exist.
2015-01-02 19:42:35 +01:00
µb.bindTabToPageStats = function(tabId, pageURL, context) {
2014-07-07 01:14:32 +02:00
this.updateBadgeAsync(tabId);
2014-06-24 00:42:43 +02:00
// https://github.com/gorhill/httpswitchboard/issues/303
2014-07-14 17:24:59 +02:00
// Normalize page URL
2015-01-28 22:19:46 +01:00
var normalURL = this.normalizePageURL(tabId, pageURL);
2014-06-24 00:42:43 +02:00
2014-07-14 17:24:59 +02:00
// Do not create a page store for URLs which are of no interests
2015-01-28 22:19:46 +01:00
if ( normalURL === '' ) {
2014-07-14 17:24:59 +02:00
this.unbindTabFromPageStats(tabId);
2014-06-24 00:42:43 +02:00
return null;
}
2015-01-17 13:19:48 +01:00
// Reuse page store if one exists: this allows to guess if a tab is a popup
2014-06-24 00:42:43 +02:00
var pageStore = this.pageStores[tabId];
2014-08-02 17:40:27 +02:00
2015-01-17 13:19:48 +01:00
// Tab is not bound
if ( !pageStore ) {
2015-03-23 20:19:17 +01:00
return this.pageStores[tabId] = this.PageStore.factory(tabId, pageURL, normalURL);
2015-01-17 13:19:48 +01:00
}
// https://github.com/gorhill/uBlock/issues/516
// If context if 'beforeRequest', do not rebind
if ( context === 'beforeRequest' ) {
return pageStore;
}
2015-01-24 18:34:36 +01:00
// Rebind according to context. We rebind even if the URL did not change,
// as maybe the tab was force-reloaded, in which case the page stats must
// be all reset.
2015-03-23 20:19:17 +01:00
pageStore.reuse(pageURL, normalURL, context);
2014-06-24 00:42:43 +02:00
return pageStore;
};
2015-01-02 19:42:35 +01:00
µb.unbindTabFromPageStats = function(tabId) {
2014-07-16 23:20:11 +02:00
//console.debug('µBlock> unbindTabFromPageStats(%d)', tabId);
2014-09-14 22:20:40 +02:00
var pageStore = this.pageStores[tabId];
if ( pageStore !== undefined ) {
pageStore.dispose();
delete this.pageStores[tabId];
}
2014-06-24 00:42:43 +02:00
};
/******************************************************************************/
2015-01-02 19:42:35 +01:00
µb.pageUrlFromTabId = function(tabId) {
2014-06-24 00:42:43 +02:00
var pageStore = this.pageStores[tabId];
return pageStore ? pageStore.pageURL : '';
};
2015-01-02 19:42:35 +01:00
µb.pageUrlFromPageStats = function(pageStats) {
2014-06-24 00:42:43 +02:00
if ( pageStats ) {
return pageStats.pageURL;
}
return '';
};
2015-01-02 19:42:35 +01:00
µb.pageStoreFromTabId = function(tabId) {
2014-06-24 00:42:43 +02:00
return this.pageStores[tabId];
};
/******************************************************************************/
// Permanent page store for behind-the-scene requests. Must never be removed.
µb.pageStores[vAPI.noTabId] = µb.PageStore.factory(
vAPI.noTabId,
2015-03-23 20:19:17 +01:00
'',
µb.normalizePageURL(vAPI.noTabId)
);
2015-01-02 19:42:35 +01:00
/******************************************************************************/
/******************************************************************************/
// Stale page store entries janitor
// https://github.com/gorhill/uBlock/issues/455
var pageStoreJanitorPeriod = 15 * 60 * 1000;
var pageStoreJanitorSampleAt = 0;
var pageStoreJanitorSampleSize = 10;
var pageStoreJanitor = function() {
var vapiTabs = vAPI.tabs;
var tabIds = Object.keys(µb.pageStores).sort();
var checkTab = function(tabId) {
vapiTabs.get(tabId, function(tab) {
if ( !tab ) {
//console.error('tab.js> pageStoreJanitor(): stale page store found:', µb.pageUrlFromTabId(tabId));
µb.unbindTabFromPageStats(tabId);
}
});
};
if ( pageStoreJanitorSampleAt >= tabIds.length ) {
pageStoreJanitorSampleAt = 0;
}
var tabId;
2015-01-02 19:42:35 +01:00
var n = Math.min(pageStoreJanitorSampleAt + pageStoreJanitorSampleSize, tabIds.length);
for ( var i = pageStoreJanitorSampleAt; i < n; i++ ) {
tabId = tabIds[i];
// Do not remove behind-the-scene page store
if ( vAPI.isNoTabId(tabId) ) {
continue;
}
checkTab(tabId);
2015-01-02 19:42:35 +01:00
}
pageStoreJanitorSampleAt = n;
setTimeout(pageStoreJanitor, pageStoreJanitorPeriod);
};
setTimeout(pageStoreJanitor, pageStoreJanitorPeriod);
/******************************************************************************/
2014-06-24 00:42:43 +02:00
/******************************************************************************/
2015-01-02 19:42:35 +01:00
})();
/******************************************************************************/