From 2a5a444482b4b2248ecb025cf9bc7d672052e993 Mon Sep 17 00:00:00 2001 From: Raymond Hill Date: Thu, 15 Feb 2024 14:34:50 -0500 Subject: [PATCH] Mind that multiple `uritransform` may apply to a single request Related issue: https://github.com/uBlockOrigin/uBlock-issues/issues/3125 --- src/js/static-net-filtering.js | 51 ++++++++++++++++++++-------------- 1 file changed, 30 insertions(+), 21 deletions(-) diff --git a/src/js/static-net-filtering.js b/src/js/static-net-filtering.js index 8e6b3a4bc2533..9189c01bc9604 100644 --- a/src/js/static-net-filtering.js +++ b/src/js/static-net-filtering.js @@ -5289,28 +5289,37 @@ FilterContainer.prototype.redirectRequest = function(redirectEngine, fctxt) { FilterContainer.prototype.transformRequest = function(fctxt) { const directives = this.matchAndFetchModifiers(fctxt, 'uritransform'); if ( directives === undefined ) { return; } - const directive = directives[directives.length-1]; - if ( (directive.bits & ALLOW_REALM) !== 0 ) { return directives; } - if ( directive.refs instanceof Object === false ) { return; } - const { refs } = directive; - if ( refs.$cache === null ) { - refs.$cache = sfp.parseReplaceValue(refs.value); - } - const cache = refs.$cache; - if ( cache === undefined ) { return; } const redirectURL = new URL(fctxt.url); - const before = `${redirectURL.pathname}${redirectURL.search}${redirectURL.hash}`; - if ( cache.re.test(before) !== true ) { return; } - const after = before.replace(cache.re, cache.replacement); - if ( after === before ) { return; } - const hashPos = after.indexOf('#'); - redirectURL.hash = hashPos !== -1 ? after.slice(hashPos) : ''; - const afterMinusHash = hashPos !== -1 ? after.slice(0, hashPos) : after; - const searchPos = afterMinusHash.indexOf('?'); - redirectURL.search = searchPos !== -1 ? afterMinusHash.slice(searchPos) : ''; - redirectURL.pathname = searchPos !== -1 ? after.slice(0, searchPos) : after; - fctxt.redirectURL = redirectURL.href; - return directives; + const out = []; + for ( const directive of directives ) { + if ( (directive.bits & ALLOW_REALM) !== 0 ) { + out.push(directive); + continue; + } + const { refs } = directive; + if ( refs instanceof Object === false ) { continue; } + if ( refs.$cache === null ) { + refs.$cache = sfp.parseReplaceValue(refs.value); + } + const cache = refs.$cache; + if ( cache === undefined ) { continue; } + const before = `${redirectURL.pathname}${redirectURL.search}${redirectURL.hash}`; + if ( cache.re.test(before) !== true ) { continue; } + const after = before.replace(cache.re, cache.replacement); + if ( after === before ) { continue; } + const hashPos = after.indexOf('#'); + redirectURL.hash = hashPos !== -1 ? after.slice(hashPos) : ''; + const afterMinusHash = hashPos !== -1 ? after.slice(0, hashPos) : after; + const searchPos = afterMinusHash.indexOf('?'); + redirectURL.search = searchPos !== -1 ? afterMinusHash.slice(searchPos) : ''; + redirectURL.pathname = searchPos !== -1 ? after.slice(0, searchPos) : after; + out.push(directive); + } + if ( out.length === 0 ) { return; } + if ( redirectURL.href !== fctxt.url ) { + fctxt.redirectURL = redirectURL.href; + } + return out; }; function parseRedirectRequestValue(directive) {