From d11a3f2fa3533fff9e3074b2e0e39a9efa3a5101 Mon Sep 17 00:00:00 2001 From: Raymond Hill Date: Tue, 13 Sep 2022 16:15:22 -0400 Subject: [PATCH] Fix broken :not() operator when forcing parsing as procedural Related issue: - https://github.com/uBlockOrigin/uBlock-issues/issues/2262 --- src/js/static-filtering-parser.js | 33 +++++++++++++++++++++++++++++++ 1 file changed, 33 insertions(+) diff --git a/src/js/static-filtering-parser.js b/src/js/static-filtering-parser.js index b29113c8a..9ffaa5722 100644 --- a/src/js/static-filtering-parser.js +++ b/src/js/static-filtering-parser.js @@ -1534,6 +1534,9 @@ Parser.prototype.SelectorCompiler = class { compileProceduralSelector(raw, asProcedural = false) { const compiled = this.compileProcedural(raw, true, asProcedural); if ( compiled !== undefined ) { + if ( asProcedural ) { + this.optimizeCompiledProcedural(compiled); + } compiled.raw = this.decompileProcedural(compiled); } return compiled; @@ -1688,6 +1691,31 @@ Parser.prototype.SelectorCompiler = class { return s; } + optimizeCompiledProcedural(compiled) { + if ( typeof compiled === 'string' ) { return; } + if ( Array.isArray(compiled.tasks) === false ) { return; } + const tasks = []; + let selector = compiled.selector; + for ( const task of compiled.tasks ) { + switch ( task[0] ) { + case ':not': + case ':if-not': + this.optimizeCompiledProcedural(task[1]); + if ( tasks.length === 0 && typeof task[1] === 'string' ) { + selector += `:not(${task[1]})`; + break; + } + tasks.push(task); + break; + default: + tasks.push(task); + break; + } + } + compiled.selector = selector; + compiled.tasks = tasks.length !== 0 ? tasks : undefined; + } + // https://github.com/gorhill/uBlock/issues/2793#issuecomment-333269387 // Normalize (somewhat) the stringified version of procedural // cosmetic filters -- this increase the likelihood of detecting @@ -1696,6 +1724,7 @@ Parser.prototype.SelectorCompiler = class { // The normalized string version is what is reported in the logger, // by design. decompileProcedural(compiled) { + if ( typeof compiled === 'string' ) { return compiled; } const tasks = compiled.tasks || []; const raw = [ compiled.selector ]; for ( const task of tasks ) { @@ -1736,6 +1765,10 @@ Parser.prototype.SelectorCompiler = class { break; case ':not': case ':if-not': + if ( typeof(task[1]) === 'string' ) { + raw.push(`:not(${task[1]})`); + break; + } raw.push(`:not(${this.decompileProcedural(task[1])})`); break; case ':spath':