From 064e7cfebe7cdfdb6518452a2700039e0872b1cc Mon Sep 17 00:00:00 2001 From: Maxim Tsoy Date: Fri, 6 Dec 2024 15:51:27 +0100 Subject: [PATCH 1/4] Add support for filterlist exceptions in CPM --- .../Autoconsent/AutoconsentUserScript.swift | 28 +++++++++++++++++-- 1 file changed, 25 insertions(+), 3 deletions(-) diff --git a/DuckDuckGo/Autoconsent/AutoconsentUserScript.swift b/DuckDuckGo/Autoconsent/AutoconsentUserScript.swift index 97c1370bca..ea4cc34ce7 100644 --- a/DuckDuckGo/Autoconsent/AutoconsentUserScript.swift +++ b/DuckDuckGo/Autoconsent/AutoconsentUserScript.swift @@ -194,6 +194,24 @@ extension AutoconsentUserScript { replyHandler([ "type": "ok" ], nil) // this is just to prevent a Promise rejection } } + + func matchDomainList(domain: String?, domainsList: [String]) -> Bool { + guard let domain = domain else { return false } + let trimmedDomains = domainsList.filter { !$0.trimmingWhitespace().isEmpty } + + // Break domain apart to handle www.* + var tempDomain = domain + while tempDomain.contains(".") { + if trimmedDomains.contains(tempDomain) { + return true + } + + let comps = tempDomain.split(separator: ".") + tempDomain = comps.dropFirst().joined(separator: ".") + } + + return false + } @MainActor func handleInit(message: WKScriptMessage, replyHandler: @escaping (Any?, String?) -> Void) { @@ -236,9 +254,10 @@ extension AutoconsentUserScript { } let remoteConfig = self.config.settings(for: .autoconsent) let disabledCMPs = remoteConfig["disabledCMPs"] as? [String] ?? [] - let enableFilterList = config.isSubfeatureEnabled(AutoconsentSubfeature.filterlist) + let filterlistExceptions = remoteConfig["filterlistExceptions"] as? [String] ?? [] + let enableFilterList = config.isSubfeatureEnabled(AutoconsentSubfeature.filterlist) && !self.matchDomainList(domain: topURLDomain, domainsList: filterlistExceptions) - replyHandler([ + let autoconsentConfig = [ "type": "initResp", "rules": nil, // rules are bundled with the content script atm "config": [ @@ -251,7 +270,10 @@ extension AutoconsentUserScript { "isMainWorld": false, "enableFilterList": enableFilterList ] as [String: Any?] - ] as [String: Any?], nil) + ] as [String: Any?] + Logger.autoconsent.debug("autoconsent config: \(String(describing: autoconsentConfig))") + + replyHandler(autoconsentConfig, nil) } @MainActor From dca35fd3e9d274c862889ee0e274b3f9e78c1b14 Mon Sep 17 00:00:00 2001 From: Maxim Tsoy Date: Fri, 6 Dec 2024 16:11:49 +0100 Subject: [PATCH 2/4] lint fix --- DuckDuckGo/Autoconsent/AutoconsentUserScript.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/DuckDuckGo/Autoconsent/AutoconsentUserScript.swift b/DuckDuckGo/Autoconsent/AutoconsentUserScript.swift index ea4cc34ce7..e48490cc23 100644 --- a/DuckDuckGo/Autoconsent/AutoconsentUserScript.swift +++ b/DuckDuckGo/Autoconsent/AutoconsentUserScript.swift @@ -194,7 +194,7 @@ extension AutoconsentUserScript { replyHandler([ "type": "ok" ], nil) // this is just to prevent a Promise rejection } } - + func matchDomainList(domain: String?, domainsList: [String]) -> Bool { guard let domain = domain else { return false } let trimmedDomains = domainsList.filter { !$0.trimmingWhitespace().isEmpty } From 9adcbee13279c401b1243cff9d93a58587384b5d Mon Sep 17 00:00:00 2001 From: Maxim Tsoy Date: Fri, 6 Dec 2024 20:37:13 +0100 Subject: [PATCH 3/4] Add an integration test for filterlist (disabled) --- .../AutoconsentIntegrationTests.swift | 50 +++++++++++++++++++ 1 file changed, 50 insertions(+) diff --git a/IntegrationTests/AutoconsentIntegrationTests.swift b/IntegrationTests/AutoconsentIntegrationTests.swift index ce5be346e4..e8b465c5b0 100644 --- a/IntegrationTests/AutoconsentIntegrationTests.swift +++ b/IntegrationTests/AutoconsentIntegrationTests.swift @@ -173,6 +173,56 @@ class AutoconsentIntegrationTests: XCTestCase { XCTAssertTrue(isBannerHidden == true) } + @MainActor + func disabled_testFilterlistRule_whenFakeCookieBannerIsDisplayed_bannerIsHidden() async throws { + // enable the feature + CookiePopupProtectionPreferences.shared.isAutoconsentEnabled = true + let url = URL(string: "http://privacy-test-pages.site/features/autoconsent/filterlist.html")! + let tab = self.tabViewModel.tab + // expect `cosmetic` to be published + let cookieConsentManagedPromise = tab.privacyInfoPublisher + .compactMap { + return $0?.$cookieConsentManaged + } + .switchToLatest() + .compactMap { + return $0?.isCosmeticRuleApplied == true ? true : nil + } + .receive(on: DispatchQueue.main) + .timeout(10) + .first() + .promise() + + _=await tab.setUrl(url, source: .link)?.result + + do { + let cookieConsentManaged = try await cookieConsentManagedPromise.value + XCTAssertTrue(cookieConsentManaged == true) + } catch { + struct ErrorWithHTML: Error, LocalizedError, CustomDebugStringConvertible { + let originalError: Error + let html: String + + var errorDescription: String? { + (originalError as CustomDebugStringConvertible).debugDescription + "\nHTML:\n\(html)" + } + var debugDescription: String { + errorDescription! + } + } + let html = try await tab.webView.evaluateJavaScript("document.documentElement.outerHTML") as String? + + if let html { + throw ErrorWithHTML(originalError: error, html: html) + } else { + throw error + } + } + + let isBannerHidden = try await tab.webView.evaluateJavaScript("window.getComputedStyle(banner).opacity === '0'") as Bool? + XCTAssertTrue(isBannerHidden == true) + } + } private extension CookieConsentInfo { From f576daf35d67e5a7659493b02507d34dfb9f99e9 Mon Sep 17 00:00:00 2001 From: Maxim Tsoy Date: Thu, 19 Dec 2024 13:51:29 +0100 Subject: [PATCH 4/4] Enable the filterlist integration test --- IntegrationTests/AutoconsentIntegrationTests.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/IntegrationTests/AutoconsentIntegrationTests.swift b/IntegrationTests/AutoconsentIntegrationTests.swift index e8b465c5b0..aa5fef0b1a 100644 --- a/IntegrationTests/AutoconsentIntegrationTests.swift +++ b/IntegrationTests/AutoconsentIntegrationTests.swift @@ -174,7 +174,7 @@ class AutoconsentIntegrationTests: XCTestCase { } @MainActor - func disabled_testFilterlistRule_whenFakeCookieBannerIsDisplayed_bannerIsHidden() async throws { + func testFilterlistRule_whenFakeCookieBannerIsDisplayed_bannerIsHidden() async throws { // enable the feature CookiePopupProtectionPreferences.shared.isAutoconsentEnabled = true let url = URL(string: "http://privacy-test-pages.site/features/autoconsent/filterlist.html")!