uBlock/platform/mv3/safari/patch-ruleset.js
Raymond Hill ff8e63e31d
draft
2026-03-07 11:47:02 -05:00

142 lines
5.7 KiB
JavaScript

/*******************************************************************************
uBlock Origin - a comprehensive, efficient content blocker
Copyright (C) 2025-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
*/
const restrSeparator = '(?:[^%.0-9a-z_-]|$)';
const rePlainChars = /[.+?${}()|[\]\\]/g;
const reSeparators = /\^/g;
const reDanglingAsterisks = /^\*+|\*+$/g;
const reAsterisks = /\*+/g;
const restrHostnameAnchor1 = '^[a-z-]+://(?:[^/?#]+\\.)?';
const restrHostnameAnchor2 = '^[a-z-]+://(?:[^/?#]+)?';
function regexFilterFromUrlFilter(urlFilter) {
let regexFilter = urlFilter;
if ( urlFilter.startsWith('||') ) {
regexFilter = regexFilter.slice(2);
} else if ( urlFilter.startsWith('|') ) {
regexFilter = regexFilter.slice(1);
}
if ( urlFilter.endsWith('|') ) {
regexFilter = regexFilter.slice(0, -1);
}
regexFilter = regexFilter.replace(rePlainChars, '\\$&')
.replace(reSeparators, restrSeparator)
.replace(reDanglingAsterisks, '')
.replace(reAsterisks, '.*?');
if ( urlFilter.startsWith('||') ) {
regexFilter = (
regexFilter.startsWith('\\.') ?
restrHostnameAnchor2 :
restrHostnameAnchor1
) + regexFilter;
} else if ( urlFilter.startsWith('|') ) {
regexFilter = `^${regexFilter}`;
}
if ( urlFilter.endsWith('|') ) {
regexFilter = `${regexFilter}$`;
}
console.info(`converting "${urlFilter}" to "${regexFilter}"`);
return regexFilter;
}
// https://github.com/uBlockOrigin/uBOL-home/issues/539
function patchRuleForIssue539(rule) {
const { condition } = rule;
if ( Array.isArray(condition.requestDomains) === false ) { return; }
if ( Array.isArray(condition.initiatorDomains) ) { return; }
if ( Array.isArray(condition.excludedRequestDomains) ) {
if ( Array.isArray(condition.excludedInitiatorDomains) ) { return; }
}
if ( Array.isArray(condition.resourceTypes) === false ) { return; }
if ( condition.resourceTypes.length !== 1 ) { return; }
if ( condition.resourceTypes.includes('main_frame') === false ) { return; }
if ( condition.regexFilter === undefined ) { return; }
condition.initiatorDomains = condition.requestDomains;
delete condition.requestDomains;
if ( Array.isArray(condition.excludedRequestDomains) ) {
condition.excludedInitiatorDomains = condition.excludedRequestDomains;
delete condition.excludedRequestDomains;
}
}
// https://github.com/WebKit/WebKit/blob/6cef2858442a4012b783876efd7dd8c0c5669cf9/Source/WebKit/UIProcess/Extensions/Cocoa/_WKWebExtensionDeclarativeNetRequestRule.mm#L1134
function patchRuleWithTo(rule) {
const { condition } = rule;
if ( Array.isArray(condition.requestDomains) === false ) { return; }
const { urlFilter } = condition;
if ( urlFilter === undefined ) { return; }
if ( urlFilter.includes('*') === false ) { return; }
if ( urlFilter.includes('^') ) { return; }
condition.regexFilter = regexFilterFromUrlFilter(urlFilter);
condition.urlFilter = undefined;
}
function patchRule(rule, out) {
const copy = structuredClone(rule);
const condition = copy.condition;
if ( copy.action.type === 'modifyHeaders' ) { return; }
if ( Array.isArray(copy.condition.responseHeaders) ) { return; }
// https://github.com/uBlockOrigin/uBOL-home/issues/476#issuecomment-3299309478
// https://github.com/uBlockOrigin/uBOL-home/issues/608
if ( copy.action.redirect?.transform?.queryTransform?.removeParams ) {
const resourceTypes = condition.resourceTypes;
if ( resourceTypes?.includes('main_frame') ) {
condition.resourceTypes = resourceTypes.filter(a => a !== 'main_frame' && a !== 'image');
if ( condition.resourceTypes.length === 0 ) { return; }
}
}
if ( Array.isArray(condition.initiatorDomains) ) {
condition.domains = condition.initiatorDomains;
delete condition.initiatorDomains;
}
if ( Array.isArray(condition.excludedInitiatorDomains) ) {
condition.excludedDomains = condition.excludedInitiatorDomains;
delete condition.excludedInitiatorDomains;
}
patchRuleForIssue539(copy);
patchRuleWithTo(copy);
// https://github.com/uBlockOrigin/uBOL-home/issues/434
let { urlFilter } = condition;
if ( urlFilter?.endsWith('^') ) {
urlFilter = urlFilter.slice(0, -1);
const match = /^(.*?\/\/|\|\|)/.exec(urlFilter);
const pattern = match
? urlFilter.slice(match[0].length)
: urlFilter;
if ( /[^\w.%*-]/.test(pattern) ) {
const extra = structuredClone(copy);
extra.condition.urlFilter = `${urlFilter}|`;
out.push(extra);
console.log(`\tAdd ${extra.condition.urlFilter}`);
}
}
out.push(copy);
return copy;
}
export function patchRuleset(ruleset) {
const out = [];
for ( const rule of ruleset ) {
if ( patchRule(rule, out) ) { continue; }
console.log(`\tReject ${JSON.stringify(rule)}`);
}
return out;
}