2024-07-29 12:54:46 -06:00
|
|
|
/*******************************************************************************
|
|
|
|
|
|
|
|
uBlock Origin Lite - a comprehensive, MV3-compliant content blocker
|
|
|
|
Copyright (C) 2024-present 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
|
|
|
|
*/
|
|
|
|
|
|
|
|
import { browser, dnr } from './ext.js';
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
|
|
|
export const isSideloaded = (( ) => {
|
|
|
|
if ( dnr.onRuleMatchedDebug instanceof Object === false ) { return false; }
|
|
|
|
const { id } = browser.runtime;
|
|
|
|
// https://addons.mozilla.org/en-US/firefox/addon/ublock-origin-lite/
|
|
|
|
if ( id === 'uBOLite@raymondhill.net' ) { return false; }
|
|
|
|
// https://chromewebstore.google.com/detail/ddkjiahejlhfcafbddmgiahcphecmpfh
|
|
|
|
if ( id === 'ddkjiahejlhfcafbddmgiahcphecmpfh' ) { return false; }
|
|
|
|
// https://microsoftedge.microsoft.com/addons/detail/cimighlppcgcoapaliogpjjdehbnofhn
|
|
|
|
if ( id === 'cimighlppcgcoapaliogpjjdehbnofhn' ) { return false; }
|
|
|
|
return true;
|
|
|
|
})();
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
|
|
|
export const ubolLog = (...args) => {
|
|
|
|
// Do not pollute dev console in stable releases.
|
|
|
|
if ( isSideloaded !== true ) { return; }
|
|
|
|
console.info('[uBOL]', ...args);
|
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
|
|
|
export const getMatchedRules = (( ) => {
|
|
|
|
const noopFn = ( ) => Promise.resolve([]);
|
|
|
|
if ( isSideloaded !== true ) { return noopFn; }
|
|
|
|
if ( dnr.onRuleMatchedDebug instanceof Object === false ) { return noopFn; }
|
|
|
|
|
|
|
|
const rulesets = new Map();
|
|
|
|
const bufferSize = 256;
|
|
|
|
const matchedRules = new Array(bufferSize);
|
|
|
|
matchedRules.fill(null);
|
|
|
|
let writePtr = 0;
|
|
|
|
|
|
|
|
const pruneLongLists = list => {
|
|
|
|
if ( list.length <= 21 ) { return list; }
|
|
|
|
return [ ...list.slice(0, 10), '...', ...list.slice(-10) ];
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
const getRuleset = async rulesetId => {
|
|
|
|
if ( rulesets.has(rulesetId) ) {
|
|
|
|
return rulesets.get(rulesetId);
|
|
|
|
}
|
|
|
|
let rules;
|
|
|
|
if ( rulesetId === dnr.DYNAMIC_RULESET_ID ) {
|
|
|
|
rules = await dnr.getDynamicRules().catch(( ) => undefined);
|
|
|
|
} else {
|
|
|
|
const response = await fetch(`/rulesets/main/${rulesetId}.json`).catch(( ) => undefined);
|
|
|
|
if ( response === undefined ) { return; }
|
|
|
|
rules = await response.json().catch(( ) => undefined);
|
|
|
|
}
|
|
|
|
if ( Array.isArray(rules) === false ) { return; }
|
|
|
|
const ruleset = new Map();
|
|
|
|
for ( const rule of rules ) {
|
|
|
|
const condition = rule.condition;
|
|
|
|
if ( condition ) {
|
|
|
|
if ( condition.requestDomains ) {
|
|
|
|
condition.requestDomains = pruneLongLists(condition.requestDomains);
|
|
|
|
}
|
|
|
|
if ( condition.initiatorDomains ) {
|
|
|
|
condition.initiatorDomains = pruneLongLists(condition.initiatorDomains);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const ruleId = rule.id;
|
2024-08-06 12:55:30 -06:00
|
|
|
rule.id = `${rulesetId}/${ruleId}`;
|
2024-07-29 12:54:46 -06:00
|
|
|
ruleset.set(ruleId, rule);
|
|
|
|
}
|
|
|
|
rulesets.set(rulesetId, ruleset);
|
|
|
|
return ruleset;
|
|
|
|
};
|
|
|
|
|
|
|
|
const getRuleDetails = async ruleInfo => {
|
|
|
|
const { rulesetId, ruleId } = ruleInfo.rule;
|
|
|
|
const ruleset = await getRuleset(rulesetId);
|
|
|
|
if ( ruleset === undefined ) { return; }
|
|
|
|
return { request: ruleInfo.request, rule: ruleset.get(ruleId) };
|
|
|
|
};
|
|
|
|
|
|
|
|
dnr.onRuleMatchedDebug.addListener(ruleInfo => {
|
|
|
|
matchedRules[writePtr] = ruleInfo;
|
|
|
|
writePtr = (writePtr + 1) % bufferSize;
|
|
|
|
});
|
|
|
|
|
|
|
|
return async tabId => {
|
|
|
|
const promises = [];
|
|
|
|
for ( let i = 0; i < bufferSize; i++ ) {
|
|
|
|
const j = (writePtr + i) % bufferSize;
|
|
|
|
const ruleInfo = matchedRules[j];
|
|
|
|
if ( ruleInfo === null ) { continue; }
|
|
|
|
if ( ruleInfo.request.tabId !== tabId ) { continue; }
|
|
|
|
const promise = getRuleDetails(ruleInfo);
|
|
|
|
if ( promise === undefined ) { continue; }
|
|
|
|
promises.unshift(promise);
|
|
|
|
}
|
|
|
|
return Promise.all(promises);
|
|
|
|
};
|
|
|
|
})();
|
|
|
|
|
|
|
|
/******************************************************************************/
|