2020-02-27 13:45:29 +01:00
|
|
|
/**
|
|
|
|
* Request Analyzer
|
|
|
|
* Belongs to Decentraleyes.
|
|
|
|
*
|
|
|
|
* @author Thomas Rientjes
|
|
|
|
* @since 2016-04-11
|
2020-04-14 07:43:25 +02:00
|
|
|
*
|
|
|
|
* @author nobody42
|
|
|
|
* @since 2020-02-26
|
|
|
|
*
|
2020-02-27 13:45:29 +01:00
|
|
|
* @license MPL 2.0
|
|
|
|
*
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
|
|
* You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
*/
|
|
|
|
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Request Analyzer
|
|
|
|
*/
|
|
|
|
|
|
|
|
var requestAnalyzer = {};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Public Methods
|
|
|
|
*/
|
|
|
|
|
|
|
|
requestAnalyzer.isValidCandidate = function (requestDetails, tabDetails) {
|
|
|
|
|
|
|
|
let initiatorDomain, isWhitelisted;
|
2020-05-02 06:58:40 +02:00
|
|
|
let fontawesome = new RegExp('\(font-awesome|fontawesome)');
|
2020-02-27 13:45:29 +01:00
|
|
|
|
|
|
|
initiatorDomain = helpers.extractDomainFromUrl(tabDetails.url, true);
|
|
|
|
|
|
|
|
if (initiatorDomain === null) {
|
|
|
|
initiatorDomain = Address.EXAMPLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
isWhitelisted = requestAnalyzer.whitelistedDomains[initiatorDomain];
|
|
|
|
|
|
|
|
if (isWhitelisted) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-05-02 06:58:40 +02:00
|
|
|
// Font Awesome injections in Chromium deactivated (https://gitlab.com/nobody42/localcdn/-/issues/67)
|
|
|
|
if(BrowserType.CHROMIUM && fontawesome.test(requestDetails.url)) {
|
|
|
|
console.warn('[ LocalCDN ] Font Awesome is not fully supported by your browser.');
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-02-27 13:45:29 +01:00
|
|
|
// Only requests of type GET can be valid candidates.
|
|
|
|
return requestDetails.method === WebRequest.GET;
|
|
|
|
};
|
|
|
|
|
|
|
|
requestAnalyzer.getLocalTarget = function (requestDetails) {
|
|
|
|
|
|
|
|
let destinationUrl, destinationHost, destinationPath, hostMappings, basePath, resourceMappings;
|
|
|
|
|
|
|
|
destinationUrl = new URL(requestDetails.url);
|
|
|
|
|
|
|
|
destinationHost = destinationUrl.host;
|
|
|
|
destinationPath = destinationUrl.pathname;
|
|
|
|
|
|
|
|
// Use the proper mappings for the targeted host.
|
|
|
|
hostMappings = mappings[destinationHost];
|
|
|
|
|
|
|
|
// Resource mapping files are never locally available.
|
|
|
|
if (Resource.MAPPING_EXPRESSION.test(destinationPath)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
basePath = requestAnalyzer._matchBasePath(hostMappings, destinationPath);
|
|
|
|
resourceMappings = hostMappings[basePath];
|
|
|
|
|
|
|
|
if (!resourceMappings) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return either the local target's path or false.
|
|
|
|
return requestAnalyzer._findLocalTarget(resourceMappings, basePath, destinationHost, destinationPath);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Private Methods
|
|
|
|
*/
|
|
|
|
|
|
|
|
requestAnalyzer._matchBasePath = function (hostMappings, channelPath) {
|
|
|
|
|
|
|
|
for (let basePath of Object.keys(hostMappings)) {
|
|
|
|
|
|
|
|
if (channelPath.startsWith(basePath)) {
|
|
|
|
return basePath;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
|
|
|
requestAnalyzer._findLocalTarget = function (resourceMappings, basePath, channelHost, channelPath) {
|
|
|
|
|
2020-04-14 07:42:56 +02:00
|
|
|
let resourcePath, versionNumber, resourcePattern, filename, shorthandResource;
|
2020-02-27 13:45:29 +01:00
|
|
|
|
2020-03-29 10:23:16 +02:00
|
|
|
chrome.storage.local.get(Setting.LOGGING, function (items) {
|
|
|
|
requestAnalyzer.logging = items.enableLogging;
|
|
|
|
});
|
2020-02-27 13:45:29 +01:00
|
|
|
|
2020-03-29 10:23:16 +02:00
|
|
|
resourcePath = channelPath.replace(basePath, '');
|
2020-02-27 13:45:29 +01:00
|
|
|
versionNumber = resourcePath.match(Resource.VERSION_EXPRESSION);
|
|
|
|
resourcePattern = resourcePath.replace(versionNumber, Resource.VERSION_PLACEHOLDER);
|
|
|
|
|
2020-04-14 07:42:56 +02:00
|
|
|
shorthandResource = shorthands.specialFiles(channelHost, channelPath);
|
|
|
|
if (shorthandResource) {
|
|
|
|
return shorthandResource;
|
2020-04-05 08:55:04 +02:00
|
|
|
}
|
|
|
|
|
2020-02-27 13:45:29 +01:00
|
|
|
for (let resourceMold of Object.keys(resourceMappings)) {
|
|
|
|
|
|
|
|
if (resourcePattern.startsWith(resourceMold)) {
|
|
|
|
|
|
|
|
let targetPath, hostShorthands, version;
|
|
|
|
|
|
|
|
targetPath = resourceMappings[resourceMold].path;
|
|
|
|
targetPath = targetPath.replace(Resource.VERSION_PLACEHOLDER, versionNumber);
|
|
|
|
|
2020-03-20 08:14:31 +01:00
|
|
|
// Replace the requested version with the latest depending on major version
|
2020-03-26 09:41:14 +01:00
|
|
|
version = helpers.setLastVersion(targetPath, versionNumber).toString();
|
2020-03-16 14:50:13 +01:00
|
|
|
targetPath = targetPath.replace(versionNumber, version);
|
|
|
|
|
2020-02-27 13:45:29 +01:00
|
|
|
hostShorthands = shorthands[channelHost];
|
|
|
|
|
|
|
|
if (hostShorthands && hostShorthands[targetPath]) {
|
|
|
|
|
|
|
|
let shorthand = hostShorthands[targetPath];
|
|
|
|
|
|
|
|
targetPath = shorthand.path;
|
|
|
|
version = shorthand.version;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
version = versionNumber && versionNumber[0] || targetPath.match(Resource.VERSION_EXPRESSION);
|
|
|
|
}
|
|
|
|
|
2020-03-24 08:03:08 +01:00
|
|
|
// Get bundle name
|
2020-03-25 15:31:33 +01:00
|
|
|
let bundle = helpers.determineBundle(channelPath);
|
2020-03-24 08:03:08 +01:00
|
|
|
if(bundle !== '') {
|
|
|
|
filename = channelPath.split('/').pop();
|
2020-04-05 20:55:19 +02:00
|
|
|
targetPath = ( RegExp('.*\.css$').test(filename) ) ? targetPath + filename : targetPath + filename + 'm';
|
2020-03-24 08:03:08 +01:00
|
|
|
}
|
|
|
|
|
2020-03-29 10:23:16 +02:00
|
|
|
if (requestAnalyzer.logging) {
|
|
|
|
console.log('[ LocalCDN ] Replaced resource: ' + targetPath);
|
|
|
|
}
|
2020-02-27 13:45:29 +01:00
|
|
|
// Prepare and return a local target.
|
|
|
|
return {
|
|
|
|
'source': channelHost,
|
2020-03-26 09:41:14 +01:00
|
|
|
'version': version,
|
2020-03-24 08:03:08 +01:00
|
|
|
'path': targetPath,
|
|
|
|
'bundle': bundle
|
2020-02-27 13:45:29 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
2020-03-29 10:23:16 +02:00
|
|
|
if (requestAnalyzer.logging) {
|
|
|
|
console.warn('[ LocalCDN ] Missing resource: ' + channelHost + channelPath);
|
|
|
|
}
|
2020-02-27 13:45:29 +01:00
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
|
|
|
requestAnalyzer._applyWhitelistedDomains = function () {
|
|
|
|
|
|
|
|
chrome.storage.local.get(Setting.WHITELISTED_DOMAINS, function (items) {
|
|
|
|
requestAnalyzer.whitelistedDomains = items.whitelistedDomains || {};
|
|
|
|
});
|
|
|
|
};
|
2020-05-25 22:30:44 +02:00
|
|
|
requestAnalyzer._applyManipulateDOMDomains = function () {
|
|
|
|
|
|
|
|
chrome.storage.local.get(Setting.DOMAINS_MANIPULATE_DOM, function (items) {
|
|
|
|
requestAnalyzer.manipulateDOMDomains = items.manipulateDOMDomains || {};
|
|
|
|
});
|
|
|
|
};
|
2020-02-27 13:45:29 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Initializations
|
|
|
|
*/
|
|
|
|
|
|
|
|
requestAnalyzer.whitelistedDomains = {};
|
|
|
|
requestAnalyzer._applyWhitelistedDomains();
|
|
|
|
|
2020-05-25 22:30:44 +02:00
|
|
|
requestAnalyzer.domainsManipulateDOMlist = {};
|
|
|
|
requestAnalyzer._applyManipulateDOMDomains();
|
2020-02-27 13:45:29 +01:00
|
|
|
/**
|
|
|
|
* Event Handlers
|
|
|
|
*/
|
|
|
|
|
|
|
|
chrome.storage.onChanged.addListener(requestAnalyzer._applyWhitelistedDomains);
|
2020-05-25 22:30:44 +02:00
|
|
|
chrome.storage.onChanged.addListener(requestAnalyzer._applyManipulateDOMDomains);
|