From cb0877d0fdf3a06a80893a23277f81acb0481002 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Mariusz=20=C5=9Apiewak?= Date: Fri, 8 Mar 2024 16:30:26 +0100 Subject: [PATCH 1/9] Revert "Report Apple Ad attribution using pixel (#2510)" (#2562) --- Core/Pixel.swift | 9 - Core/PixelEvent.swift | 5 - Core/UserDefaultsPropertyWrapper.swift | 3 +- DuckDuckGo.xcodeproj/project.pbxproj | 36 --- .../AdAttribution/AdAttributionFetcher.swift | 159 -------------- .../AdAttributionPixelReporter.swift | 96 -------- .../AdAttributionReporterStorage.swift | 38 ---- DuckDuckGo/AppDelegate.swift | 8 - .../AdAttributionFetcherTests.swift | 183 ---------------- .../AdAttributionPixelReporterTests.swift | 206 ------------------ 10 files changed, 1 insertion(+), 742 deletions(-) delete mode 100644 DuckDuckGo/AdAttribution/AdAttributionFetcher.swift delete mode 100644 DuckDuckGo/AdAttribution/AdAttributionPixelReporter.swift delete mode 100644 DuckDuckGo/AdAttribution/AdAttributionReporterStorage.swift delete mode 100644 DuckDuckGoTests/AdAttributionFetcherTests.swift delete mode 100644 DuckDuckGoTests/AdAttributionPixelReporterTests.swift diff --git a/Core/Pixel.swift b/Core/Pixel.swift index edf6d6611b..33b3ca2214 100644 --- a/Core/Pixel.swift +++ b/Core/Pixel.swift @@ -127,15 +127,6 @@ public struct PixelParameters { public static let returnUserErrorCode = "error_code" public static let returnUserOldATB = "old_atb" public static let returnUserNewATB = "new_atb" - - // Ad Attribution - public static let adAttributionOrgID = "org_id" - public static let adAttributionCampaignID = "campaign_id" - public static let adAttributionConversionType = "conversion_type" - public static let adAttributionAdGroupID = "ad_group_id" - public static let adAttributionCountryOrRegion = "country_or_region" - public static let adAttributionKeywordID = "keyword_id" - public static let adAttributionAdID = "ad_id" } public struct PixelValues { diff --git a/Core/PixelEvent.swift b/Core/PixelEvent.swift index 8925cb5716..a4e38df468 100644 --- a/Core/PixelEvent.swift +++ b/Core/PixelEvent.swift @@ -541,8 +541,6 @@ extension Pixel { case appRatingPromptFetchError - case appleAdAttribution - case userBehaviorReloadTwice case userBehaviorReloadAndRestart case userBehaviorReloadAndFireButton @@ -1059,9 +1057,6 @@ extension Pixel.Event { case .debugReturnUserUpdateATB: return "m_debug_return_user_update_atb" case .appRatingPromptFetchError: return "m_d_app_rating_prompt_fetch_error" - - // MARK: - Apple Ad Attribution - case .appleAdAttribution: return "m_apple-ad-attribution" // MARK: - User behavior case .userBehaviorReloadTwice: return "m_reload-twice" diff --git a/Core/UserDefaultsPropertyWrapper.swift b/Core/UserDefaultsPropertyWrapper.swift index f0515137b2..9ab2131af7 100644 --- a/Core/UserDefaultsPropertyWrapper.swift +++ b/Core/UserDefaultsPropertyWrapper.swift @@ -124,10 +124,9 @@ public struct UserDefaultsWrapper { case subscriptionIsActive = "com.duckduckgo.ios.subscruption.isActive" - case appleAdAttributionReportCompleted = "com.duckduckgo.ios.appleAdAttributionReport.completed" - case didRefreshTimestamp = "com.duckduckgo.ios.userBehavior.didRefreshTimestamp" case didBurnTimestamp = "com.duckduckgo.ios.userBehavior.didBurnTimestamp" + } private let key: Key diff --git a/DuckDuckGo.xcodeproj/project.pbxproj b/DuckDuckGo.xcodeproj/project.pbxproj index 118bd1271b..ed21691428 100644 --- a/DuckDuckGo.xcodeproj/project.pbxproj +++ b/DuckDuckGo.xcodeproj/project.pbxproj @@ -309,12 +309,7 @@ 4BFB911B29B7D9530014D4B7 /* AppTrackingProtectionStoringModelPerformanceTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4BFB911A29B7D9530014D4B7 /* AppTrackingProtectionStoringModelPerformanceTests.swift */; }; 6AC6DAB328804F97002723C0 /* BarsAnimator.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6AC6DAB228804F97002723C0 /* BarsAnimator.swift */; }; 6AC98419288055C1005FA9CA /* BarsAnimatorTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6AC98418288055C1005FA9CA /* BarsAnimatorTests.swift */; }; - 6FD1BAE42B87A107000C475C /* AdAttributionPixelReporter.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FD1BAE12B87A107000C475C /* AdAttributionPixelReporter.swift */; }; - 6FD1BAE52B87A107000C475C /* AdAttributionReporterStorage.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FD1BAE22B87A107000C475C /* AdAttributionReporterStorage.swift */; }; - 6FD1BAE62B87A107000C475C /* AdAttributionFetcher.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FD1BAE32B87A107000C475C /* AdAttributionFetcher.swift */; }; - 6FD3AEE32B8F4EEB0060FCCC /* AdAttributionPixelReporterTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FD3AEE12B8DFBB80060FCCC /* AdAttributionPixelReporterTests.swift */; }; 6FDA1FB32B59584400AC962A /* AddressDisplayHelper.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FDA1FB22B59584400AC962A /* AddressDisplayHelper.swift */; }; - 6FF915822B88E07A0042AC87 /* AdAttributionFetcherTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FF915802B88E0750042AC87 /* AdAttributionFetcherTests.swift */; }; 83004E802193BB8200DA013C /* WKNavigationExtension.swift in Sources */ = {isa = PBXBuildFile; fileRef = 83004E7F2193BB8200DA013C /* WKNavigationExtension.swift */; }; 83004E862193E5ED00DA013C /* TabViewControllerBrowsingMenuExtension.swift in Sources */ = {isa = PBXBuildFile; fileRef = 83004E852193E5ED00DA013C /* TabViewControllerBrowsingMenuExtension.swift */; }; 83004E882193E8C700DA013C /* TabViewControllerLongPressMenuExtension.swift in Sources */ = {isa = PBXBuildFile; fileRef = 83004E872193E8C700DA013C /* TabViewControllerLongPressMenuExtension.swift */; }; @@ -1404,12 +1399,7 @@ 6AC6DAB228804F97002723C0 /* BarsAnimator.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BarsAnimator.swift; sourceTree = ""; }; 6AC98418288055C1005FA9CA /* BarsAnimatorTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BarsAnimatorTests.swift; sourceTree = ""; }; 6FB030C7234331B400A10DB9 /* Configuration.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; name = Configuration.xcconfig; path = Configuration/Configuration.xcconfig; sourceTree = ""; }; - 6FD1BAE12B87A107000C475C /* AdAttributionPixelReporter.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = AdAttributionPixelReporter.swift; path = AdAttribution/AdAttributionPixelReporter.swift; sourceTree = ""; }; - 6FD1BAE22B87A107000C475C /* AdAttributionReporterStorage.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = AdAttributionReporterStorage.swift; path = AdAttribution/AdAttributionReporterStorage.swift; sourceTree = ""; }; - 6FD1BAE32B87A107000C475C /* AdAttributionFetcher.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = AdAttributionFetcher.swift; path = AdAttribution/AdAttributionFetcher.swift; sourceTree = ""; }; - 6FD3AEE12B8DFBB80060FCCC /* AdAttributionPixelReporterTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AdAttributionPixelReporterTests.swift; sourceTree = ""; }; 6FDA1FB22B59584400AC962A /* AddressDisplayHelper.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AddressDisplayHelper.swift; sourceTree = ""; }; - 6FF915802B88E0750042AC87 /* AdAttributionFetcherTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AdAttributionFetcherTests.swift; sourceTree = ""; }; 83004E7F2193BB8200DA013C /* WKNavigationExtension.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = WKNavigationExtension.swift; sourceTree = ""; }; 83004E832193E14C00DA013C /* UIAlertControllerExtension.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = UIAlertControllerExtension.swift; path = ../Core/UIAlertControllerExtension.swift; sourceTree = ""; }; 83004E852193E5ED00DA013C /* TabViewControllerBrowsingMenuExtension.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TabViewControllerBrowsingMenuExtension.swift; sourceTree = ""; }; @@ -3555,25 +3545,6 @@ name = VPN; sourceTree = ""; }; - 6FD1BAE02B87A0E8000C475C /* AdAttribution */ = { - isa = PBXGroup; - children = ( - 6FD1BAE32B87A107000C475C /* AdAttributionFetcher.swift */, - 6FD1BAE12B87A107000C475C /* AdAttributionPixelReporter.swift */, - 6FD1BAE22B87A107000C475C /* AdAttributionReporterStorage.swift */, - ); - name = AdAttribution; - sourceTree = ""; - }; - 6FF9157F2B88E04F0042AC87 /* AdAttribution */ = { - isa = PBXGroup; - children = ( - 6FF915802B88E0750042AC87 /* AdAttributionFetcherTests.swift */, - 6FD3AEE12B8DFBB80060FCCC /* AdAttributionPixelReporterTests.swift */, - ); - name = AdAttribution; - sourceTree = ""; - }; 830FA79B1F8E81FB00FCE105 /* ContentBlocker */ = { isa = PBXGroup; children = ( @@ -3759,7 +3730,6 @@ CB48D32F2B90CE8500631D8B /* UserBehaviorMonitor */, EE3B98EA2A9634CC002F63A0 /* DuckDuckGoAlpha.entitlements */, CB258D1129A4F1BB00DEBA24 /* Configuration */, - 6FD1BAE02B87A0E8000C475C /* AdAttribution */, 1E908BED29827C480008C8F3 /* Autoconsent */, 3157B43627F4C8380042D3D7 /* Favicons */, AA4D6A8023DE4973007E8790 /* AppIcon */, @@ -4966,7 +4936,6 @@ F12D98401F266B30003C2EE3 /* DuckDuckGo */ = { isa = PBXGroup; children = ( - 6FF9157F2B88E04F0042AC87 /* AdAttribution */, CB48D3342B90CEBD00631D8B /* UserBehaviorMonitor */, F17669A21E411D63003D3222 /* Application */, 026F08B629B7DC130079B9DF /* AppTrackingProtection */, @@ -6516,7 +6485,6 @@ 319A371028299A850079FBCE /* PasswordHider.swift in Sources */, 982C87C42255559A00919035 /* UITableViewCellExtension.swift in Sources */, B623C1C42862CD670043013E /* WKDownloadSession.swift in Sources */, - 6FD1BAE42B87A107000C475C /* AdAttributionPixelReporter.swift in Sources */, EEFD562F2A65B6CA00DAEC48 /* NetworkProtectionInviteViewModel.swift in Sources */, 1E8AD1D927C4FEC100ABA377 /* DownloadsListSectioningHelper.swift in Sources */, 1E4DCF4827B6A35400961E25 /* DownloadsListModel.swift in Sources */, @@ -6693,7 +6661,6 @@ 31CB4251273AF50700FA0F3F /* SpeechRecognizerProtocol.swift in Sources */, 319A37172829C8AD0079FBCE /* UITableViewExtension.swift in Sources */, 85EE7F59224673C5000FE757 /* WebContainerNavigationController.swift in Sources */, - 6FD1BAE52B87A107000C475C /* AdAttributionReporterStorage.swift in Sources */, D68A21462B7EC16200BB372E /* SubscriptionExternalLinkViewModel.swift in Sources */, F4C9FBF528340DDA002281CC /* AutofillInterfaceEmailTruncator.swift in Sources */, 1E016AB42949FEB500F21625 /* OmniBarNotificationViewModel.swift in Sources */, @@ -6914,7 +6881,6 @@ 1E4FAA6627D8DFC800ADC5B3 /* CompleteDownloadRowViewModel.swift in Sources */, D6D95CE12B6D52DA00960317 /* RootPresentationMode.swift in Sources */, 83004E862193E5ED00DA013C /* TabViewControllerBrowsingMenuExtension.swift in Sources */, - 6FD1BAE62B87A107000C475C /* AdAttributionFetcher.swift in Sources */, EE01EB402AFBD0000096AAC9 /* NetworkProtectionVPNSettingsViewModel.swift in Sources */, EE72CA852A862D000043B5B3 /* NetworkProtectionDebugViewController.swift in Sources */, C18ED43A2AB6F77600BF3805 /* AutofillSettingsEnableFooterView.swift in Sources */, @@ -6993,7 +6959,6 @@ CBDD5DDF29A6736A00832877 /* APIHeadersTests.swift in Sources */, 986B45D0299E30A50089D2D7 /* BookmarkEntityTests.swift in Sources */, B6AD9E3828D4512E0019CDE9 /* EmbeddedTrackerDataTests.swift in Sources */, - 6FF915822B88E07A0042AC87 /* AdAttributionFetcherTests.swift in Sources */, 1E722729292EB24D003B5F53 /* AppSettingsMock.swift in Sources */, 8536A1C8209AF2410050739E /* MockVariantManager.swift in Sources */, C1B7B53428944EFA0098FD6A /* CoreDataTestUtilities.swift in Sources */, @@ -7070,7 +7035,6 @@ 9847C00527A41A0A00DB07AA /* WebViewTestHelper.swift in Sources */, 3170048227A9504F00C03F35 /* DownloadMocks.swift in Sources */, 317045C02858C6B90016ED1F /* AutofillInterfaceEmailTruncatorTests.swift in Sources */, - 6FD3AEE32B8F4EEB0060FCCC /* AdAttributionPixelReporterTests.swift in Sources */, 987130C6294AAB9F00AB05E0 /* BookmarkListViewModelTests.swift in Sources */, F1134ED21F40EF3A00B73467 /* JsonTestDataLoader.swift in Sources */, 4B83397129AC18C9003F7EA9 /* AppTrackingProtectionStoringModelTests.swift in Sources */, diff --git a/DuckDuckGo/AdAttribution/AdAttributionFetcher.swift b/DuckDuckGo/AdAttribution/AdAttributionFetcher.swift deleted file mode 100644 index a9d3d31a80..0000000000 --- a/DuckDuckGo/AdAttribution/AdAttributionFetcher.swift +++ /dev/null @@ -1,159 +0,0 @@ -// -// AdAttributionFetcher.swift -// DuckDuckGo -// -// Copyright © 2024 DuckDuckGo. All rights reserved. -// -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. -// - -import AdServices -import Common - -protocol AdAttributionFetcher { - func fetch() async -> AdServicesAttributionResponse? -} - -/// Fetches ad attribution data for from Apple. -/// -/// DuckDuckGo uses the AdServices framework to fetch and monitor anonymous install attribution data from Apple. No personally identifiable data is involved. -/// DuckDuckGo does not use the App Tracking Transparency framework at any point, and only uses the “standard” attribution payload. -/// See https://developer.apple.com/documentation/adservices/aaattribution/attributiontoken()#Attribution-payload-descriptions for details. -struct DefaultAdAttributionFetcher: AdAttributionFetcher { - - typealias TokenGetter = () throws -> String - - private let tokenGetter: TokenGetter - private let urlSession: URLSession - private let retryInterval: TimeInterval - - init(tokenGetter: @escaping TokenGetter = Self.fetchAttributionToken, - urlSession: URLSession = .shared, - retryInterval: TimeInterval = .seconds(5)) { - self.tokenGetter = tokenGetter - self.urlSession = urlSession - self.retryInterval = retryInterval - } - - func fetch() async -> AdServicesAttributionResponse? { - guard #available(iOS 14.3, *) else { - return nil - } - - var lastToken: String? - - for _ in 0.. AdServicesAttributionResponse { - let request = createAttributionDataRequest(with: token) - let (data, response) = try await urlSession.data(for: request) - - guard let response = response as? HTTPURLResponse else { - throw AdAttributionFetcherError.invalidResponse - } - - switch response.statusCode { - case 200: - let decoder = JSONDecoder() - let decoded = try decoder.decode(AdServicesAttributionResponse.self, from: data) - - return decoded - case 400: - throw AdAttributionFetcherError.invalidToken - case 404: - throw AdAttributionFetcherError.invalidResponse - default: - throw AdAttributionFetcherError.unknown - } - } - - private func createAttributionDataRequest(with token: String) -> URLRequest { - var request = URLRequest(url: Constant.attributionServiceURL) - request.httpMethod = "POST" - request.setValue("text/plain", forHTTPHeaderField: "Content-Type") - request.httpBody = token.data(using: .utf8) - - return request - } - - private struct Constant { - static let attributionServiceURL = URL(string: "https://api-adservices.apple.com/api/v1/")! - static let maxRetries = 3 - } -} - -extension AdAttributionFetcher { - static func fetchAttributionToken() throws -> String { - if #available(iOS 14.3, *) { - return try AAAttribution.attributionToken() - } else { - throw AdAttributionFetcherError.attributionUnsupported - } - } -} - -struct AdServicesAttributionResponse: Decodable { - let attribution: Bool - let orgId: Int? - let campaignId: Int? - let conversionType: String? - let adGroupId: Int? - let countryOrRegion: String? - let keywordId: Int? - let adId: Int? -} - -enum AdAttributionFetcherError: Error { - case attributionUnsupported - case invalidResponse - case invalidToken - case unknown - - var allowsRetry: Bool { - switch self { - case .invalidToken, .invalidResponse: - return true - case .unknown, .attributionUnsupported: - return false - } - } -} diff --git a/DuckDuckGo/AdAttribution/AdAttributionPixelReporter.swift b/DuckDuckGo/AdAttribution/AdAttributionPixelReporter.swift deleted file mode 100644 index 4ea0faa969..0000000000 --- a/DuckDuckGo/AdAttribution/AdAttributionPixelReporter.swift +++ /dev/null @@ -1,96 +0,0 @@ -// -// AdAttributionPixelReporter.swift -// DuckDuckGo -// -// Copyright © 2024 DuckDuckGo. All rights reserved. -// -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. -// - -import Foundation -import Core - -protocol PixelFiring { - static func fire(pixel: Pixel.Event, withAdditionalParameters params: [String: String]) async throws -} - -final class AdAttributionPixelReporter { - - static var shared = AdAttributionPixelReporter() - - private var fetcherStorage: AdAttributionReporterStorage - private let attributionFetcher: AdAttributionFetcher - private let pixelFiring: PixelFiring.Type - - init(fetcherStorage: AdAttributionReporterStorage = UserDefaultsAdAttributionReporterStorage(), - attributionFetcher: AdAttributionFetcher = DefaultAdAttributionFetcher(), - pixelFiring: PixelFiring.Type = Pixel.self) { - self.fetcherStorage = fetcherStorage - self.attributionFetcher = attributionFetcher - self.pixelFiring = pixelFiring - } - - @discardableResult - func reportAttributionIfNeeded() async -> Bool { - guard await fetcherStorage.wasAttributionReportSuccessful == false else { - return false - } - - if let attributionData = await self.attributionFetcher.fetch() { - if attributionData.attribution { - let parameters = self.pixelParametersForAttribution(attributionData) - do { - try await pixelFiring.fire(pixel: .appleAdAttribution, withAdditionalParameters: parameters) - } catch { - return false - } - } - - await fetcherStorage.markAttributionReportSuccessful() - - return true - } - - return false - } - - private func pixelParametersForAttribution(_ attribution: AdServicesAttributionResponse) -> [String: String] { - var params: [String: String] = [:] - - params[PixelParameters.adAttributionAdGroupID] = attribution.adGroupId.map(String.init) - params[PixelParameters.adAttributionOrgID] = attribution.orgId.map(String.init) - params[PixelParameters.adAttributionCampaignID] = attribution.campaignId.map(String.init) - params[PixelParameters.adAttributionConversionType] = attribution.conversionType - params[PixelParameters.adAttributionAdGroupID] = attribution.adGroupId.map(String.init) - params[PixelParameters.adAttributionCountryOrRegion] = attribution.countryOrRegion - params[PixelParameters.adAttributionKeywordID] = attribution.keywordId.map(String.init) - params[PixelParameters.adAttributionAdID] = attribution.adId.map(String.init) - - return params - } -} - -extension Pixel: PixelFiring { - static func fire(pixel: Event, withAdditionalParameters params: [String: String]) async throws { - - try await withCheckedThrowingContinuation { (continuation: CheckedContinuation) in - Pixel.fire(pixel: pixel, withAdditionalParameters: params) { error in - if let error { - continuation.resume(throwing: error) - } else { - continuation.resume() - } - } - } - } -} diff --git a/DuckDuckGo/AdAttribution/AdAttributionReporterStorage.swift b/DuckDuckGo/AdAttribution/AdAttributionReporterStorage.swift deleted file mode 100644 index c0085e05f9..0000000000 --- a/DuckDuckGo/AdAttribution/AdAttributionReporterStorage.swift +++ /dev/null @@ -1,38 +0,0 @@ -// -// AdAttributionReporterStorage.swift -// DuckDuckGo -// -// Copyright © 2024 DuckDuckGo. All rights reserved. -// -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. -// - -import Core -import Foundation - -protocol AdAttributionReporterStorage { - var wasAttributionReportSuccessful: Bool { get async } - - func markAttributionReportSuccessful() async -} - -final class UserDefaultsAdAttributionReporterStorage: AdAttributionReporterStorage { - @MainActor - @UserDefaultsWrapper(key: .appleAdAttributionReportCompleted, defaultValue: false) - var wasAttributionReportSuccessful: Bool - - @MainActor - func markAttributionReportSuccessful() async { - wasAttributionReportSuccessful = true - } -} diff --git a/DuckDuckGo/AppDelegate.swift b/DuckDuckGo/AppDelegate.swift index 3bd12868ed..1f712dd2c6 100644 --- a/DuckDuckGo/AppDelegate.swift +++ b/DuckDuckGo/AppDelegate.swift @@ -335,8 +335,6 @@ class AppDelegate: UIResponder, UIApplicationDelegate { clearDebugWaitlistState() - reportAdAttribution() - AppDependencyProvider.shared.userBehaviorMonitor.handleAction(.reopenApp) return true @@ -389,12 +387,6 @@ class AppDelegate: UIResponder, UIApplicationDelegate { } #endif - private func reportAdAttribution() { - Task.detached(priority: .background) { - await AdAttributionPixelReporter.shared.reportAttributionIfNeeded() - } - } - func applicationDidBecomeActive(_ application: UIApplication) { guard !testing else { return } diff --git a/DuckDuckGoTests/AdAttributionFetcherTests.swift b/DuckDuckGoTests/AdAttributionFetcherTests.swift deleted file mode 100644 index 28b9c94541..0000000000 --- a/DuckDuckGoTests/AdAttributionFetcherTests.swift +++ /dev/null @@ -1,183 +0,0 @@ -// -// AdAttributionFetcherTests.swift -// DuckDuckGo -// -// Copyright © 2024 DuckDuckGo. All rights reserved. -// -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. -// - -import XCTest - -@testable import DuckDuckGo -@testable import TestUtils - -final class AdAttributionFetcherTests: XCTestCase { - - private let mockSession: URLSession = { - let configuration = URLSessionConfiguration.default - configuration.protocolClasses = [MockURLProtocol.self] - let session = URLSession(configuration: configuration) - - return session - }() - - override func setUpWithError() throws { - MockURLProtocol.requestHandler = MockURLProtocol.defaultHandler - } - - override func tearDownWithError() throws { - MockURLProtocol.requestHandler = nil - } - - func testMakesRequestWithToken() async throws { - let testToken = "foo" - let sut = DefaultAdAttributionFetcher(tokenGetter: { testToken }, urlSession: mockSession, retryInterval: .leastNonzeroMagnitude) - - _ = await sut.fetch() - - let requestStream = try XCTUnwrap(MockURLProtocol.lastRequest?.httpBodyStream) - let requestBody = try Data(reading: requestStream) - - XCTAssertEqual(String(data: requestBody, encoding: .utf8), testToken) - } - - func testRetriesRequest() async throws { - let testToken = "foo" - let sut = DefaultAdAttributionFetcher(tokenGetter: { testToken }, urlSession: mockSession, retryInterval: .leastNonzeroMagnitude) - let retryExpectation = XCTestExpectation() - retryExpectation.expectedFulfillmentCount = 3 - retryExpectation.assertForOverFulfill = true - - MockURLProtocol.requestHandler = { request in - retryExpectation.fulfill() - let handler = MockURLProtocol.handler(with: 404) - return try handler(request) - } - - _ = await sut.fetch() - - let requestStream = try XCTUnwrap(MockURLProtocol.lastRequest?.httpBodyStream) - let requestBody = try Data(reading: requestStream) - - XCTAssertEqual(String(data: requestBody, encoding: .utf8), testToken) - - await fulfillment(of: [retryExpectation]) - } - - func testRefreshesTokenOnRetry() async throws { - let retryExpectation = XCTestExpectation() - retryExpectation.expectedFulfillmentCount = 3 - retryExpectation.assertForOverFulfill = true - - let refreshExpectation = XCTestExpectation() - - let testToken = "foo" - let sut = DefaultAdAttributionFetcher(tokenGetter: { - refreshExpectation.fulfill() - return testToken - }, urlSession: mockSession, retryInterval: .leastNonzeroMagnitude) - - MockURLProtocol.requestHandler = { request in - retryExpectation.fulfill() - let handler = MockURLProtocol.handler(with: 400) - return try handler(request) - } - - _ = await sut.fetch() - - let requestStream = try XCTUnwrap(MockURLProtocol.lastRequest?.httpBodyStream) - let requestBody = try Data(reading: requestStream) - - XCTAssertEqual(String(data: requestBody, encoding: .utf8), testToken) - - await fulfillment(of: [retryExpectation]) - } - - func testDoesNotRetry_WhenUnrecoverable() async throws { - let testToken = "foo" - let sut = DefaultAdAttributionFetcher(tokenGetter: { testToken }, urlSession: mockSession, retryInterval: .leastNonzeroMagnitude) - let noRetryExpectation = XCTestExpectation() - noRetryExpectation.expectedFulfillmentCount = 1 - noRetryExpectation.assertForOverFulfill = true - - MockURLProtocol.requestHandler = { request in - noRetryExpectation.fulfill() - let handler = MockURLProtocol.handler(with: 500) - return try handler(request) - } - - _ = await sut.fetch() - - let requestStream = try XCTUnwrap(MockURLProtocol.lastRequest?.httpBodyStream) - let requestBody = try Data(reading: requestStream) - - XCTAssertEqual(String(data: requestBody, encoding: .utf8), testToken) - - await fulfillment(of: [noRetryExpectation]) - } - - func testRespectsRetryInterval() async throws { - let testToken = "foo" - let sut = DefaultAdAttributionFetcher(tokenGetter: { testToken }, urlSession: mockSession, retryInterval: .milliseconds(30)) - - MockURLProtocol.requestHandler = { request in - let handler = MockURLProtocol.handler(with: 404) - return try handler(request) - } - - let startTime = Date() - _ = await sut.fetch() - - XCTAssertGreaterThanOrEqual(Date().timeIntervalSince(startTime), .milliseconds(90)) - } -} - -private extension MockURLProtocol { - typealias RequestHandler = (URLRequest) throws -> (HTTPURLResponse, Data?) - - static func handler(with statusCode: Int, data: Data? = nil) -> RequestHandler { - return { request in - (HTTPURLResponse(url: request.url!, statusCode: statusCode, httpVersion: nil, headerFields: nil)!, data) - } - } - - static let defaultHandler = handler(with: 300) -} - -private extension Data { - init(reading input: InputStream, size: Int = 1024) throws { - self.init() - input.open() - defer { - input.close() - } - - let bufferSize = size - let buffer = UnsafeMutablePointer.allocate(capacity: bufferSize) - defer { - buffer.deallocate() - } - while input.hasBytesAvailable { - let read = input.read(buffer, maxLength: bufferSize) - if read < 0 { - // Stream error occured - throw input.streamError! - } else if read == 0 { - // EOF - break - } - self.append(buffer, count: read) - } - } -} diff --git a/DuckDuckGoTests/AdAttributionPixelReporterTests.swift b/DuckDuckGoTests/AdAttributionPixelReporterTests.swift deleted file mode 100644 index 97eef8546e..0000000000 --- a/DuckDuckGoTests/AdAttributionPixelReporterTests.swift +++ /dev/null @@ -1,206 +0,0 @@ -// -// AdAttributionPixelReporterTests.swift -// DuckDuckGo -// -// Copyright © 2024 DuckDuckGo. All rights reserved. -// -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. -// - -import Core -import XCTest - -@testable import DuckDuckGo - -final class AdAttributionPixelReporterTests: XCTestCase { - - private var attributionFetcher: AdAttributionFetcherMock! - private var fetcherStorage: AdAttributionReporterStorageMock! - - override func setUpWithError() throws { - attributionFetcher = AdAttributionFetcherMock() - fetcherStorage = AdAttributionReporterStorageMock() - } - - override func tearDownWithError() throws { - attributionFetcher = nil - fetcherStorage = nil - PixelFiringMock.tearDown() - } - - func testReportsAttribution() async { - let sut = createSUT() - attributionFetcher.fetchResponse = AdServicesAttributionResponse(attribution: true) - - let result = await sut.reportAttributionIfNeeded() - - XCTAssertEqual(PixelFiringMock.lastPixel, .appleAdAttribution) - XCTAssertTrue(result) - } - - func testReportsOnce() async { - let sut = createSUT() - attributionFetcher.fetchResponse = AdServicesAttributionResponse(attribution: true) - - await fetcherStorage.markAttributionReportSuccessful() - let result = await sut.reportAttributionIfNeeded() - - XCTAssertNil(PixelFiringMock.lastPixel) - XCTAssertFalse(result) - } - - func testPixelname() async { - let sut = createSUT() - attributionFetcher.fetchResponse = AdServicesAttributionResponse(attribution: true) - - let result = await sut.reportAttributionIfNeeded() - - XCTAssertEqual(PixelFiringMock.lastPixel?.name, "m_apple-ad-attribution") - XCTAssertTrue(result) - } - - func testPixelAttributesNaming() async throws { - let sut = createSUT() - attributionFetcher.fetchResponse = AdServicesAttributionResponse(attribution: true) - - await sut.reportAttributionIfNeeded() - - let pixelAttributes = try XCTUnwrap(PixelFiringMock.lastParams) - - XCTAssertEqual(pixelAttributes["org_id"], "1") - XCTAssertEqual(pixelAttributes["campaign_id"], "2") - XCTAssertEqual(pixelAttributes["conversion_type"], "conversionType") - XCTAssertEqual(pixelAttributes["ad_group_id"], "3") - XCTAssertEqual(pixelAttributes["country_or_region"], "countryOrRegion") - XCTAssertEqual(pixelAttributes["keyword_id"], "4") - XCTAssertEqual(pixelAttributes["ad_id"], "5") - } - - func testPixelAttributes_WhenPartialAttributionData() async throws { - let sut = createSUT() - attributionFetcher.fetchResponse = AdServicesAttributionResponse( - attribution: true, - orgId: 1, - campaignId: 2, - conversionType: "conversionType", - adGroupId: nil, - countryOrRegion: nil, - keywordId: nil, - adId: nil - ) - - await sut.reportAttributionIfNeeded() - - let pixelAttributes = try XCTUnwrap(PixelFiringMock.lastParams) - - XCTAssertEqual(pixelAttributes["org_id"], "1") - XCTAssertEqual(pixelAttributes["campaign_id"], "2") - XCTAssertEqual(pixelAttributes["conversion_type"], "conversionType") - XCTAssertNil(pixelAttributes["ad_group_id"]) - XCTAssertNil(pixelAttributes["country_or_region"]) - XCTAssertNil(pixelAttributes["keyword_id"]) - XCTAssertNil(pixelAttributes["ad_id"]) - } - - func testPixelNotFiredAndMarksReport_WhenAttributionFalse() async { - let sut = createSUT() - attributionFetcher.fetchResponse = AdServicesAttributionResponse(attribution: false) - - let result = await sut.reportAttributionIfNeeded() - - XCTAssertNil(PixelFiringMock.lastPixel) - XCTAssertTrue(fetcherStorage.wasAttributionReportSuccessful) - XCTAssertTrue(result) - } - - func testPixelNotFiredAndReportNotMarked_WhenAttributionUnavailable() async { - let sut = createSUT() - attributionFetcher.fetchResponse = nil - - let result = await sut.reportAttributionIfNeeded() - - XCTAssertNil(PixelFiringMock.lastPixel) - XCTAssertFalse(fetcherStorage.wasAttributionReportSuccessful) - XCTAssertFalse(result) - } - - func testDoesNotMarkSuccessful_WhenPixelFiringFailed() async { - let sut = createSUT() - attributionFetcher.fetchResponse = AdServicesAttributionResponse(attribution: true) - PixelFiringMock.expectedFireError = NSError(domain: "PixelFailure", code: 1) - - let result = await sut.reportAttributionIfNeeded() - - XCTAssertFalse(fetcherStorage.wasAttributionReportSuccessful) - XCTAssertFalse(result) - } - - private func createSUT() -> AdAttributionPixelReporter { - AdAttributionPixelReporter(fetcherStorage: fetcherStorage, - attributionFetcher: attributionFetcher, - pixelFiring: PixelFiringMock.self) - } -} - -class AdAttributionReporterStorageMock: AdAttributionReporterStorage { - func markAttributionReportSuccessful() async { - wasAttributionReportSuccessful = true - } - - private(set) var wasAttributionReportSuccessful: Bool = false -} - -class AdAttributionFetcherMock: AdAttributionFetcher { - var fetchResponse: AdServicesAttributionResponse? - func fetch() async -> AdServicesAttributionResponse? { - fetchResponse - } -} - -final actor PixelFiringMock: PixelFiring { - static var expectedFireError: Error? - - static var lastParams: [String: String]? - static var lastPixel: Pixel.Event? - static func fire(pixel: Pixel.Event, withAdditionalParameters params: [String: String]) async throws { - lastParams = params - lastPixel = pixel - - if let expectedFireError { - throw expectedFireError - } - } - - static func tearDown() { - lastParams = nil - lastPixel = nil - expectedFireError = nil - } - - private init() {} -} - -extension AdServicesAttributionResponse { - init(attribution: Bool) { - self.init( - attribution: attribution, - orgId: 1, - campaignId: 2, - conversionType: "conversionType", - adGroupId: 3, - countryOrRegion: "countryOrRegion", - keywordId: 4, - adId: 5 - ) - } -} From c5f08434c634e96245512eb0000b27da35e3bfa1 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Mariusz=20=C5=9Apiewak?= Date: Fri, 8 Mar 2024 16:41:49 +0100 Subject: [PATCH 2/9] Release 7.111.0-2 (#2563) --- DuckDuckGo.xcodeproj/project.pbxproj | 56 ++++++++++++++-------------- 1 file changed, 28 insertions(+), 28 deletions(-) diff --git a/DuckDuckGo.xcodeproj/project.pbxproj b/DuckDuckGo.xcodeproj/project.pbxproj index ed21691428..e1228c49d7 100644 --- a/DuckDuckGo.xcodeproj/project.pbxproj +++ b/DuckDuckGo.xcodeproj/project.pbxproj @@ -8174,7 +8174,7 @@ CODE_SIGN_ENTITLEMENTS = PacketTunnelProvider/PacketTunnelProvider.entitlements; CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEVELOPMENT_TEAM = HKE973VLUW; GCC_C_LANGUAGE_STANDARD = gnu11; GENERATE_INFOPLIST_FILE = YES; @@ -8211,7 +8211,7 @@ CODE_SIGN_IDENTITY = "iPhone Distribution"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; CODE_SIGN_STYLE = Manual; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEVELOPMENT_TEAM = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; GCC_C_LANGUAGE_STANDARD = gnu11; @@ -8303,7 +8303,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; GCC_C_LANGUAGE_STANDARD = gnu11; INFOPLIST_FILE = ShareExtension/Info.plist; IPHONEOS_DEPLOYMENT_TARGET = 14.0; @@ -8331,7 +8331,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; CODE_SIGN_STYLE = Manual; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEVELOPMENT_TEAM = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; GCC_C_LANGUAGE_STANDARD = gnu11; @@ -8481,7 +8481,7 @@ CODE_SIGN_ENTITLEMENTS = DuckDuckGo/DuckDuckGo.entitlements; CODE_SIGN_IDENTITY = "iPhone Distribution"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; DEVELOPMENT_ASSET_PATHS = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; @@ -8507,7 +8507,7 @@ CODE_SIGN_ENTITLEMENTS = DuckDuckGo/DuckDuckGo.entitlements; CODE_SIGN_IDENTITY = "iPhone Distribution"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; INFOPLIST_FILE = DuckDuckGo/Info.plist; IPHONEOS_DEPLOYMENT_TARGET = 14.0; @@ -8572,7 +8572,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEAD_CODE_STRIPPING = NO; GCC_C_LANGUAGE_STANDARD = gnu11; INFOPLIST_FILE = Widgets/Info.plist; @@ -8607,7 +8607,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; CODE_SIGN_STYLE = Manual; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEAD_CODE_STRIPPING = NO; DEVELOPMENT_TEAM = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; @@ -8641,7 +8641,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; GCC_C_LANGUAGE_STANDARD = gnu11; INFOPLIST_FILE = OpenAction/Info.plist; IPHONEOS_DEPLOYMENT_TARGET = 14.0; @@ -8672,7 +8672,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; CODE_SIGN_STYLE = Manual; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEVELOPMENT_TEAM = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; GCC_C_LANGUAGE_STANDARD = gnu11; @@ -8959,7 +8959,7 @@ CODE_SIGN_ENTITLEMENTS = DuckDuckGo/DuckDuckGoAlpha.entitlements; CODE_SIGN_IDENTITY = "iPhone Distribution"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; DEVELOPMENT_ASSET_PATHS = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; @@ -8990,7 +8990,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; GCC_C_LANGUAGE_STANDARD = gnu11; INFOPLIST_FILE = ShareExtension/Info.plist; IPHONEOS_DEPLOYMENT_TARGET = 14.0; @@ -9019,7 +9019,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; GCC_C_LANGUAGE_STANDARD = gnu11; INFOPLIST_FILE = OpenAction/Info.plist; IPHONEOS_DEPLOYMENT_TARGET = 14.0; @@ -9053,7 +9053,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEAD_CODE_STRIPPING = NO; GCC_C_LANGUAGE_STANDARD = gnu11; INFOPLIST_FILE = Widgets/Info.plist; @@ -9084,7 +9084,7 @@ CODE_SIGN_ENTITLEMENTS = PacketTunnelProvider/PacketTunnelProviderAlpha.entitlements; CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEVELOPMENT_TEAM = HKE973VLUW; GCC_C_LANGUAGE_STANDARD = gnu11; GENERATE_INFOPLIST_FILE = YES; @@ -9117,11 +9117,11 @@ CLANG_ENABLE_MODULES = YES; CODE_SIGN_IDENTITY = ""; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; DEFINES_MODULE = YES; DYLIB_COMPATIBILITY_VERSION = 1; - DYLIB_CURRENT_VERSION = 1; + DYLIB_CURRENT_VERSION = 2; DYLIB_INSTALL_NAME_BASE = "@rpath"; INFOPLIST_FILE = Core/Info.plist; INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks"; @@ -9355,7 +9355,7 @@ CODE_SIGN_ENTITLEMENTS = DuckDuckGo/DuckDuckGoAlpha.entitlements; CODE_SIGN_IDENTITY = "iPhone Distribution"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; DEVELOPMENT_ASSET_PATHS = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; @@ -9382,7 +9382,7 @@ CODE_SIGN_IDENTITY = "Apple Development"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; CODE_SIGN_STYLE = Manual; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEVELOPMENT_TEAM = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; GCC_C_LANGUAGE_STANDARD = gnu11; @@ -9415,7 +9415,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; CODE_SIGN_STYLE = Manual; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEVELOPMENT_TEAM = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; GCC_C_LANGUAGE_STANDARD = gnu11; @@ -9453,7 +9453,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; CODE_SIGN_STYLE = Manual; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEAD_CODE_STRIPPING = NO; DEVELOPMENT_TEAM = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; @@ -9489,7 +9489,7 @@ CODE_SIGN_IDENTITY = "Apple Development"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; CODE_SIGN_STYLE = Manual; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEVELOPMENT_TEAM = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; GCC_C_LANGUAGE_STANDARD = gnu11; @@ -9524,11 +9524,11 @@ CLANG_ENABLE_MODULES = YES; CODE_SIGN_IDENTITY = ""; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; DEFINES_MODULE = YES; DYLIB_COMPATIBILITY_VERSION = 1; - DYLIB_CURRENT_VERSION = 1; + DYLIB_CURRENT_VERSION = 2; DYLIB_INSTALL_NAME_BASE = "@rpath"; INFOPLIST_FILE = Core/Info.plist; INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks"; @@ -9702,11 +9702,11 @@ CLANG_ENABLE_MODULES = YES; CODE_SIGN_IDENTITY = ""; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; DEFINES_MODULE = YES; DYLIB_COMPATIBILITY_VERSION = 1; - DYLIB_CURRENT_VERSION = 1; + DYLIB_CURRENT_VERSION = 2; DYLIB_INSTALL_NAME_BASE = "@rpath"; INFOPLIST_FILE = Core/Info.plist; INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks"; @@ -9735,10 +9735,10 @@ CLANG_ENABLE_MODULES = YES; CODE_SIGN_IDENTITY = ""; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 2; DEFINES_MODULE = YES; DYLIB_COMPATIBILITY_VERSION = 1; - DYLIB_CURRENT_VERSION = 1; + DYLIB_CURRENT_VERSION = 2; DYLIB_INSTALL_NAME_BASE = "@rpath"; INFOPLIST_FILE = Core/Info.plist; INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks"; From 9beb215a40cdd7de2760bddb9c3c64438fec3e22 Mon Sep 17 00:00:00 2001 From: Daniel Bernal Date: Fri, 8 Mar 2024 18:26:46 +0100 Subject: [PATCH 3/9] 18. Subscription Entitlements caching (#2556) Task/Issue URL: https://app.asana.com/0/1204099484721401/1206574474409573/f Description: This PR just bumps BSK (No changes required on iOS) MainActor change has no relation to subscriptions - iOS app was broken in main because of a change in BSK, this fixes that --- DuckDuckGo.xcodeproj/project.pbxproj | 2 +- .../project.xcworkspace/xcshareddata/swiftpm/Package.resolved | 4 ++-- DuckDuckGo/SettingsViewModel.swift | 4 +++- DuckDuckGo/SubscriptionDebugViewController.swift | 2 +- 4 files changed, 7 insertions(+), 5 deletions(-) diff --git a/DuckDuckGo.xcodeproj/project.pbxproj b/DuckDuckGo.xcodeproj/project.pbxproj index bdda85746f..87aa6f9bf9 100644 --- a/DuckDuckGo.xcodeproj/project.pbxproj +++ b/DuckDuckGo.xcodeproj/project.pbxproj @@ -10017,7 +10017,7 @@ repositoryURL = "https://github.com/DuckDuckGo/BrowserServicesKit"; requirement = { kind = exactVersion; - version = 119.0.1; + version = 119.1.0; }; }; B6F997C22B8F374300476735 /* XCRemoteSwiftPackageReference "apple-toolbox" */ = { diff --git a/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved b/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved index d6347676ad..0529cdbeb6 100644 --- a/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved +++ b/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved @@ -32,8 +32,8 @@ "kind" : "remoteSourceControl", "location" : "https://github.com/DuckDuckGo/BrowserServicesKit", "state" : { - "revision" : "a32ff4084e910354bdd948812793de6b04804a1b", - "version" : "119.0.1" + "revision" : "38f51b83343d533dab2b9280a9618462149be4c8", + "version" : "119.1.0" } }, { diff --git a/DuckDuckGo/SettingsViewModel.swift b/DuckDuckGo/SettingsViewModel.swift index 1e31387cd3..50ee464566 100644 --- a/DuckDuckGo/SettingsViewModel.swift +++ b/DuckDuckGo/SettingsViewModel.swift @@ -357,7 +357,7 @@ extension SettingsViewModel { cacheSubscriptionState(active: true) // Check entitlements and update UI accordingly - let entitlements: [AccountManager.Entitlement] = [.identityTheftRestoration, .dataBrokerProtection, .networkProtection] + let entitlements: [Entitlement.ProductName] = [.identityTheftRestoration, .dataBrokerProtection, .networkProtection] for entitlement in entitlements { if case .success = await AccountManager().hasEntitlement(for: entitlement) { switch entitlement { @@ -367,6 +367,8 @@ extension SettingsViewModel { self.shouldShowDBP = true case .networkProtection: self.shouldShowNetP = true + case .unknown: + return } } } diff --git a/DuckDuckGo/SubscriptionDebugViewController.swift b/DuckDuckGo/SubscriptionDebugViewController.swift index 27c4b67c96..7fd5c9a032 100644 --- a/DuckDuckGo/SubscriptionDebugViewController.swift +++ b/DuckDuckGo/SubscriptionDebugViewController.swift @@ -238,7 +238,7 @@ final class SubscriptionDebugViewController: UITableViewController { showAlert(title: "Not authenticated", message: "No authenticated user found! - Subscription not available") return } - let entitlements: [AccountManager.Entitlement] = [.networkProtection, .dataBrokerProtection, .identityTheftRestoration] + let entitlements: [Entitlement.ProductName] = [.networkProtection, .dataBrokerProtection, .identityTheftRestoration] for entitlement in entitlements { if case let .success(result) = await AccountManager().hasEntitlement(for: entitlement) { let resultSummary = "Entitlement check for \(entitlement.rawValue): \(result)" From 92ee914eec16e1fc33af6e3082b8ef55fa1bc89f Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Fri, 8 Mar 2024 18:19:53 +0000 Subject: [PATCH 4/9] Bump submodules/privacy-reference-tests from `40ce868` to `a603ff9` (#2500) Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- submodules/privacy-reference-tests | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/submodules/privacy-reference-tests b/submodules/privacy-reference-tests index 6b7ad1e7f1..a603ff9af2 160000 --- a/submodules/privacy-reference-tests +++ b/submodules/privacy-reference-tests @@ -1 +1 @@ -Subproject commit 6b7ad1e7f15270f9dfeb58a272199f4d57c3eb22 +Subproject commit a603ff9af22ca3ff7ce2e7ffbfe18c447d9f23e8 From 072cbaf06c5d6d35d7ef795411f7acacfa75dc7d Mon Sep 17 00:00:00 2001 From: Sam Symons Date: Sun, 10 Mar 2024 15:06:33 -0700 Subject: [PATCH 5/9] Fix VPN view model memory leak (#2570) Task/Issue URL: https://app.asana.com/0/414235014887631/1206803287661273/f Tech Design URL: CC: @graeme Description: This PR fixes a view model memory leak in the VPN code. --- DuckDuckGo/NetworkProtectionStatusViewModel.swift | 10 ++++------ 1 file changed, 4 insertions(+), 6 deletions(-) diff --git a/DuckDuckGo/NetworkProtectionStatusViewModel.swift b/DuckDuckGo/NetworkProtectionStatusViewModel.swift index ab3ca1f57b..eb603f6162 100644 --- a/DuckDuckGo/NetworkProtectionStatusViewModel.swift +++ b/DuckDuckGo/NetworkProtectionStatusViewModel.swift @@ -37,7 +37,6 @@ final class NetworkProtectionStatusViewModel: ObservableObject { private let serverInfoObserver: ConnectionServerInfoObserver private let errorObserver: ConnectionErrorObserver private var cancellables: Set = [] - private var delayedToggleReenableCancellable: Cancellable? // MARK: Error @@ -158,12 +157,11 @@ final class NetworkProtectionStatusViewModel: ObservableObject { // Each event cancels the previous delayed publisher $shouldDisableToggle .filter { $0 } - .map { - Just(!$0) - .delay(for: 2.0, scheduler: DispatchQueue.main) - .assign(to: \.shouldDisableToggle, onWeaklyHeld: self) + .map { _ -> AnyPublisher in + Just(false).delay(for: 2.0, scheduler: DispatchQueue.main).eraseToAnyPublisher() } - .assign(to: \.delayedToggleReenableCancellable, onWeaklyHeld: self) + .switchToLatest() + .assign(to: \.shouldDisableToggle, onWeaklyHeld: self) .store(in: &cancellables) } From 0576c795c450986503afabb6297b3128b01beb8b Mon Sep 17 00:00:00 2001 From: Anh Do <18567+quanganhdo@users.noreply.github.com> Date: Sun, 10 Mar 2024 23:31:21 -0400 Subject: [PATCH 6/9] Move vpnFirstEnabled and networkPathChange out of VPNSettings (#2560) --- DuckDuckGo.xcodeproj/project.pbxproj | 2 +- .../xcshareddata/swiftpm/Package.resolved | 4 ++-- DuckDuckGo/Feedback/VPNMetadataCollector.swift | 7 +++++-- DuckDuckGo/NetworkProtectionTunnelController.swift | 2 +- .../NetworkProtectionPacketTunnelProvider.swift | 6 +++--- 5 files changed, 12 insertions(+), 9 deletions(-) diff --git a/DuckDuckGo.xcodeproj/project.pbxproj b/DuckDuckGo.xcodeproj/project.pbxproj index 87aa6f9bf9..2700fe8674 100644 --- a/DuckDuckGo.xcodeproj/project.pbxproj +++ b/DuckDuckGo.xcodeproj/project.pbxproj @@ -10017,7 +10017,7 @@ repositoryURL = "https://github.com/DuckDuckGo/BrowserServicesKit"; requirement = { kind = exactVersion; - version = 119.1.0; + version = 120.0.0; }; }; B6F997C22B8F374300476735 /* XCRemoteSwiftPackageReference "apple-toolbox" */ = { diff --git a/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved b/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved index 0529cdbeb6..4afb646162 100644 --- a/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved +++ b/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved @@ -32,8 +32,8 @@ "kind" : "remoteSourceControl", "location" : "https://github.com/DuckDuckGo/BrowserServicesKit", "state" : { - "revision" : "38f51b83343d533dab2b9280a9618462149be4c8", - "version" : "119.1.0" + "revision" : "cea7c43e5ab1d7484ab29abeda429350ed8a7dc1", + "version" : "120.0.0" } }, { diff --git a/DuckDuckGo/Feedback/VPNMetadataCollector.swift b/DuckDuckGo/Feedback/VPNMetadataCollector.swift index 06ee3a44c2..9d04ef794f 100644 --- a/DuckDuckGo/Feedback/VPNMetadataCollector.swift +++ b/DuckDuckGo/Feedback/VPNMetadataCollector.swift @@ -99,13 +99,16 @@ final class DefaultVPNMetadataCollector: VPNMetadataCollector { private let statusObserver: ConnectionStatusObserver private let serverInfoObserver: ConnectionServerInfoObserver private let settings: VPNSettings + private let defaults: UserDefaults init(statusObserver: ConnectionStatusObserver = ConnectionStatusObserverThroughSession(), serverInfoObserver: ConnectionServerInfoObserver = ConnectionServerInfoObserverThroughSession(), - settings: VPNSettings = .init(defaults: .networkProtectionGroupDefaults)) { + settings: VPNSettings = .init(defaults: .networkProtectionGroupDefaults), + defaults: UserDefaults = .networkProtectionGroupDefaults) { self.statusObserver = statusObserver self.serverInfoObserver = serverInfoObserver self.settings = settings + self.defaults = defaults } func collectMetadata() async -> VPNMetadata { @@ -149,7 +152,7 @@ final class DefaultVPNMetadataCollector: VPNMetadataCollector { dateFormatter.timeZone = TimeZone(secondsFromGMT: 0) dateFormatter.dateFormat = "yyyy-MM-dd HH:mm:ss" - let networkPathChange = settings.networkPathChange + let networkPathChange = defaults.networkPathChange let lastPathChange = String(describing: networkPathChange) var lastPathChangeDate = "unknown" diff --git a/DuckDuckGo/NetworkProtectionTunnelController.swift b/DuckDuckGo/NetworkProtectionTunnelController.swift index 2ba3add399..24e993cb86 100644 --- a/DuckDuckGo/NetworkProtectionTunnelController.swift +++ b/DuckDuckGo/NetworkProtectionTunnelController.swift @@ -146,7 +146,7 @@ final class NetworkProtectionTunnelController: TunnelController { try tunnelManager.connection.startVPNTunnel(options: options) UniquePixel.fire(pixel: .networkProtectionNewUser) { error in guard error != nil else { return } - VPNSettings(defaults: .networkProtectionGroupDefaults).vpnFirstEnabled = Pixel.Event.networkProtectionNewUser.lastFireDate( + UserDefaults.networkProtectionGroupDefaults.vpnFirstEnabled = Pixel.Event.networkProtectionNewUser.lastFireDate( uniquePixelStorage: UniquePixel.storage ) } diff --git a/PacketTunnelProvider/NetworkProtection/NetworkProtectionPacketTunnelProvider.swift b/PacketTunnelProvider/NetworkProtection/NetworkProtectionPacketTunnelProvider.swift index 4a20ed31eb..f4a05f4059 100644 --- a/PacketTunnelProvider/NetworkProtection/NetworkProtectionPacketTunnelProvider.swift +++ b/PacketTunnelProvider/NetworkProtection/NetworkProtectionPacketTunnelProvider.swift @@ -38,12 +38,12 @@ final class NetworkProtectionPacketTunnelProvider: PacketTunnelProvider { // MARK: - PacketTunnelProvider.Event reporting private static var packetTunnelProviderEvents: EventMapping = .init { event, _, _, _ in - let settings = VPNSettings(defaults: .networkProtectionGroupDefaults) + let defaults = UserDefaults.networkProtectionGroupDefaults switch event { case .userBecameActive: DailyPixel.fire(pixel: .networkProtectionActiveUser, - withAdditionalParameters: ["cohort": UniquePixel.dateString(for: settings.vpnFirstEnabled)]) + withAdditionalParameters: ["cohort": UniquePixel.dateString(for: defaults.vpnFirstEnabled)]) case .reportConnectionAttempt(attempt: let attempt): switch attempt { case .connecting: @@ -60,7 +60,7 @@ final class NetworkProtectionPacketTunnelProvider: PacketTunnelProvider { case .failureRecovered: DailyPixel.fireDailyAndCount(pixel: .networkProtectionTunnelFailureRecovered) case .networkPathChanged(let newPath): - settings.apply(change: .setNetworkPathChange(newPath)) + defaults.updateNetworkPath(with: newPath) } case .reportLatency(result: let result): switch result { From 861dd2c6bc5c9fe44872409dd5de6425d9399d2d Mon Sep 17 00:00:00 2001 From: Elle Sullivan Date: Mon, 11 Mar 2024 09:44:58 +0000 Subject: [PATCH 7/9] Update SwiftSoup and Kingfisher versions (#2566) --- DuckDuckGo.xcodeproj/project.pbxproj | 2 +- .../project.xcworkspace/xcshareddata/swiftpm/Package.resolved | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/DuckDuckGo.xcodeproj/project.pbxproj b/DuckDuckGo.xcodeproj/project.pbxproj index 2700fe8674..a8dc2102a3 100644 --- a/DuckDuckGo.xcodeproj/project.pbxproj +++ b/DuckDuckGo.xcodeproj/project.pbxproj @@ -10033,7 +10033,7 @@ repositoryURL = "https://github.com/scinfu/SwiftSoup"; requirement = { kind = exactVersion; - version = 2.7.0; + version = 2.7.1; }; }; F1D43AF82B99C1D300BAB743 /* XCRemoteSwiftPackageReference "BareBonesBrowser" */ = { diff --git a/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved b/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved index 4afb646162..5a5344ce43 100644 --- a/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved +++ b/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved @@ -167,8 +167,8 @@ "kind" : "remoteSourceControl", "location" : "https://github.com/scinfu/SwiftSoup", "state" : { - "revision" : "f83c097597094a04124eb6e0d1e894d24129af87", - "version" : "2.7.0" + "revision" : "1d39e56d364cba79ce43b341f9661b534cccb18d", + "version" : "2.7.1" } }, { From 3e69aa52386ee609e2fb53bb2dd14bfdd0c8063a Mon Sep 17 00:00:00 2001 From: Federico Cappelli Date: Mon, 11 Mar 2024 10:49:48 +0000 Subject: [PATCH 8/9] Subscription pro pixels (#2531) Task/Issue URL: https://app.asana.com/0/1200019156869587/1205469290776415/f Description: Pixels added to all privacy pro subscription flows. --- Core/AppLastCompiledRulesStore.swift | 20 +- .../AppPrivacyConfigurationDataProvider.swift | 2 +- Core/AppTrackerDataSetProvider.swift | 2 +- .../AppTrackingProtectionAllowlistModel.swift | 30 +- Core/AppTrackingProtectionFeedbackModel.swift | 6 +- Core/AppTrackingProtectionListViewModel.swift | 22 +- Core/AtbAndVariantCleanup.swift | 4 +- Core/BookmarkObjects.swift | 12 +- Core/BookmarksCachingSearch.swift | 62 ++-- Core/BookmarksCoreDataStorage.swift | 90 +++--- Core/BookmarksExporter.swift | 2 +- Core/BookmarksModelsErrorHandling.swift | 20 +- Core/Configuration.swift | 2 +- Core/ContentBlockerRulesLists.swift | 12 +- Core/ContentBlocking.swift | 47 ++- Core/CookieStorage.swift | 34 +-- Core/DDGPersistenceContainer.swift | 4 +- Core/DailyPixel.swift | 22 +- Core/Database.swift | 12 +- Core/Debounce.swift | 8 +- Core/DebugUserScript.swift | 28 +- Core/DefaultVariantManager.swift | 38 +-- Core/EtagStorage.swift | 14 +- Core/FaviconSourcesProvider.swift | 20 +- Core/FaviconUserScript.swift | 18 +- Core/FileStore.swift | 16 +- Core/HTTPSUpgradeParser.swift | 6 +- Core/HistoryManager.swift | 2 +- Core/Instruments.swift | 16 +- Core/InternalUserStore.swift | 2 +- Core/LegacyBookmarksStoreMigration.swift | 62 ++-- Core/Link.swift | 6 +- Core/LocaleExtension.swift | 2 +- Core/LoginFormDetectionUserScript.swift | 14 +- Core/NSAttributedStringExtension.swift | 6 +- Core/NSManagedObjectContextExtension.swift | 8 +- Core/NavigatorSharePatchUserScript.swift | 10 +- Core/NotFoundCachingDownloader.swift | 4 +- Core/Pixel.swift | 38 +-- Core/PixelEvent.swift | 281 +++++++++++------- Core/PreserveLogins.swift | 10 +- Core/PrivacyFeatures.swift | 2 +- Core/ReturnUserMeasurement.swift | 2 +- Core/SchemeHandler.swift | 10 +- Core/StatisticsLoader.swift | 30 +- Core/StatisticsUserDefaults.swift | 4 +- Core/StorageCache.swift | 8 +- Core/StringExtension.swift | 8 +- Core/TabInstrumentation.swift | 36 +-- Core/TextFieldWithInsets.swift | 4 +- Core/TextSizeUserScript.swift | 12 +- Core/TimeIntervalExtension.swift | 2 +- Core/TimedPixel.swift | 8 +- Core/UIColorExtension.swift | 36 +-- Core/UIKeyCommandExtension.swift | 4 +- Core/UIViewControllerExtension.swift | 12 +- Core/UIViewExtension.swift | 4 +- Core/URLFileExtension.swift | 4 +- Core/UserAgentManager.swift | 49 ++- Core/UserDefaultsExtension.swift | 2 +- Core/UserDefaultsPropertyWrapper.swift | 28 +- Core/WebCacheManager.swift | 32 +- Core/global.swift | 4 +- DuckDuckGo.xcodeproj/project.pbxproj | 2 + .../xcschemes/DuckDuckGo-Alpha.xcscheme | 4 +- DuckDuckGo/AppDelegate.swift | 21 ++ DuckDuckGo/SettingsViewModel.swift | 1 + ...scriptionPagesUseSubscriptionFeature.swift | 31 +- .../SubscriptionPagesUserScript.swift | 3 + .../SubscriptionEmailViewModel.swift | 5 +- .../ViewModel/SubscriptionFlowViewModel.swift | 41 ++- .../ViewModel/SubscriptionITPViewModel.swift | 1 + .../ViewModel/SubscriptionPIRViewModel.swift | 3 + .../SubscriptionRestoreViewModel.swift | 10 +- .../SubscriptionSettingsViewModel.swift | 3 +- .../Views/PurchaseInProgressView.swift | 15 +- .../Views/SubscriptionEmailView.swift | 10 + .../Views/SubscriptionFlowView.swift | 44 +-- .../Views/SubscriptionITPView.swift | 9 + .../Views/SubscriptionPIRView.swift | 13 +- .../Views/SubscriptionRestoreView.swift | 26 +- .../Views/SubscriptionSettingsView.swift | 30 +- DuckDuckGo/VPNWaitlistView.swift | 7 + submodules/privacy-reference-tests | 2 +- 84 files changed, 916 insertions(+), 680 deletions(-) diff --git a/Core/AppLastCompiledRulesStore.swift b/Core/AppLastCompiledRulesStore.swift index 05a65d3cb4..100d52239e 100644 --- a/Core/AppLastCompiledRulesStore.swift +++ b/Core/AppLastCompiledRulesStore.swift @@ -22,12 +22,12 @@ import BrowserServicesKit import TrackerRadarKit struct AppLastCompiledRules: LastCompiledRules, Codable { - + var name: String var trackerData: TrackerData var etag: String var identifier: ContentBlockerRulesIdentifier - + } protocol Storage { @@ -36,7 +36,7 @@ protocol Storage { } struct LastCompiledRulesStorage: Storage { - + private enum Const { static let filename = "LastCompiledRules" static let path = FileManager @@ -44,7 +44,7 @@ struct LastCompiledRulesStorage: Storage { .containerURL(forSecurityApplicationGroupIdentifier: ContentBlockerStoreConstants.groupName)! .appendingPathComponent(filename) } - + func persist(_ data: Data) -> Bool { do { try data.write(to: Const.path, options: .atomic) @@ -53,7 +53,7 @@ struct LastCompiledRulesStorage: Storage { return false } } - + var data: Data? { do { return try Data(contentsOf: Const.path) @@ -61,17 +61,17 @@ struct LastCompiledRulesStorage: Storage { return nil } } - + } final class AppLastCompiledRulesStore: LastCompiledRulesStore { - + private var storage: Storage init(with storage: Storage = LastCompiledRulesStorage()) { self.storage = storage } - + var rules: [LastCompiledRules] { guard let data = storage.data, @@ -88,10 +88,10 @@ final class AppLastCompiledRulesStore: LastCompiledRulesStore { etag: rules.etag, identifier: rules.identifier) } - + if !rules.isEmpty, let encodedRules = try? JSONEncoder().encode(rules) { _ = storage.persist(encodedRules) } } - + } diff --git a/Core/AppPrivacyConfigurationDataProvider.swift b/Core/AppPrivacyConfigurationDataProvider.swift index 37ce38ac13..4f17bd9b2e 100644 --- a/Core/AppPrivacyConfigurationDataProvider.swift +++ b/Core/AppPrivacyConfigurationDataProvider.swift @@ -39,7 +39,7 @@ final public class AppPrivacyConfigurationDataProvider: EmbeddedDataProvider { if let url = Bundle.main.url(forResource: "ios-config", withExtension: "json") { return url } - + return Bundle(for: self).url(forResource: "ios-config", withExtension: "json")! } diff --git a/Core/AppTrackerDataSetProvider.swift b/Core/AppTrackerDataSetProvider.swift index e4041d3cad..cf23ed6b99 100644 --- a/Core/AppTrackerDataSetProvider.swift +++ b/Core/AppTrackerDataSetProvider.swift @@ -39,7 +39,7 @@ final public class AppTrackerDataSetProvider: EmbeddedDataProvider { if let url = Bundle.main.url(forResource: "trackerData", withExtension: "json") { return url } - + return Bundle(for: Self.self).url(forResource: "trackerData", withExtension: "json")! } diff --git a/Core/AppTrackingProtectionAllowlistModel.swift b/Core/AppTrackingProtectionAllowlistModel.swift index 0698cfba3f..cdbd6f1917 100644 --- a/Core/AppTrackingProtectionAllowlistModel.swift +++ b/Core/AppTrackingProtectionAllowlistModel.swift @@ -24,28 +24,28 @@ public class AppTrackingProtectionAllowlistModel { public static let groupID = "\(Global.groupIdPrefix).apptp" public static let fileName = "appTPallowlist" } - + private let filename: String - + lazy private var allowlistUrl: URL? = { let groupContainerUrl = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: Constants.groupID) return groupContainerUrl?.appendingPathComponent(self.filename, conformingTo: .text) }() - + var allowedDomains: Set - + public init(filename: String = Constants.fileName) { self.allowedDomains = Set() self.filename = filename - + readFromFile() } - + func writeToFile() { guard let allowlistUrl = allowlistUrl else { fatalError("Unable to get file location") } - + // Write the allowlist as a textfile with one domain per line do { let string = allowedDomains.joined(separator: "\n") @@ -59,7 +59,7 @@ public class AppTrackingProtectionAllowlistModel { print(error.localizedDescription) } } - + public func readFromFile() { guard let allowlistUrl = allowlistUrl else { fatalError("Unable to get file location") @@ -67,7 +67,7 @@ public class AppTrackingProtectionAllowlistModel { guard FileManager.default.fileExists(atPath: allowlistUrl.path) else { return } - + // Read allowlist from file. Break the string into array then cast to a set. do { let strData = try String(contentsOf: allowlistUrl) @@ -77,30 +77,30 @@ public class AppTrackingProtectionAllowlistModel { print(error.localizedDescription) } } - + public func allow(domain: String) { allowedDomains.insert(domain) writeToFile() } - + public func contains(domain: String) -> Bool { var check = domain while check.contains(".") { if allowedDomains.contains(check) { return true } - + check = String(check.components(separatedBy: ".").dropFirst().joined(separator: ".")) } - + return false } - + public func remove(domain: String) { allowedDomains.remove(domain) writeToFile() } - + public func clearList() { allowedDomains.removeAll() writeToFile() diff --git a/Core/AppTrackingProtectionFeedbackModel.swift b/Core/AppTrackingProtectionFeedbackModel.swift index bfa9243050..98a26c8b81 100644 --- a/Core/AppTrackingProtectionFeedbackModel.swift +++ b/Core/AppTrackingProtectionFeedbackModel.swift @@ -41,19 +41,19 @@ public class AppTrackingProtectionFeedbackModel: ObservableObject { return [] } } - + public func sendReport(appName: String, category: String, description: String) { let date = Calendar.current.date(byAdding: .minute, value: -10, to: Date())! let trackers = trackers(moreRecentThan: date) let trackersString = trackers.map { $0.domain }.joined(separator: ",") - + let parameters = [ "appName": appName, "category": category, "description": description, "blockedTrackers": trackersString ] - + Pixel.fire(pixel: .appTPBreakageReport, withAdditionalParameters: parameters) } diff --git a/Core/AppTrackingProtectionListViewModel.swift b/Core/AppTrackingProtectionListViewModel.swift index 7ff61c28e1..c77ed22884 100644 --- a/Core/AppTrackingProtectionListViewModel.swift +++ b/Core/AppTrackingProtectionListViewModel.swift @@ -33,7 +33,7 @@ public class AppTrackingProtectionListViewModel: NSObject, ObservableObject, NSF @Published public var debugModeEnabled = false @Published public var isOnboarding = false - + // We only want to show "Manage Trackers" and "Report an issue" if the user has enabled AppTP at least once @UserDefaultsWrapper(key: .appTPUsed, defaultValue: false) public var appTPUsed { @@ -49,7 +49,7 @@ public class AppTrackingProtectionListViewModel: NSObject, ObservableObject, NSF queue.maxConcurrentOperationCount = 1 return queue }() - + private let relativeFormatter: DateFormatter = { let formatter = DateFormatter() formatter.timeStyle = .none @@ -57,13 +57,13 @@ public class AppTrackingProtectionListViewModel: NSObject, ObservableObject, NSF formatter.doesRelativeDateFormatting = true return formatter }() - + private let relativeTimeFormatter: RelativeDateTimeFormatter = { let formatter = RelativeDateTimeFormatter() formatter.unitsStyle = .short return formatter }() - + private let dateFormatter: DateFormatter = { let formatter = DateFormatter() formatter.dateFormat = "MMMM dd" @@ -76,26 +76,26 @@ public class AppTrackingProtectionListViewModel: NSObject, ObservableObject, NSF formatter.timeStyle = .medium return formatter }() - + private let inputFormatter: DateFormatter = { let formatter = DateFormatter() formatter.dateFormat = "yyyy-MM-dd" return formatter }() - + public func formattedDate(_ sectionName: String) -> String { guard let date = inputFormatter.date(from: sectionName) else { return "Invalid Date" } - + let relativeDate = relativeFormatter.string(from: date) if relativeDate.rangeOfCharacter(from: .decimalDigits) != nil { return dateFormatter.string(from: date) } - + return relativeDate } - + /// Returns a relative datestring for the given timestamp. e.g. "5 min. ago" /// If the timestamp is within 1 second of the current time this function will return `nil` /// A `nil` return value should be considered "just now". @@ -108,7 +108,7 @@ public class AppTrackingProtectionListViewModel: NSObject, ObservableObject, NSF // return nil here and replace it with UserText on the view side. return nil } - + return relativeTimeFormatter.localizedString(for: timestamp, relativeTo: Date()) } @@ -138,7 +138,7 @@ public class AppTrackingProtectionListViewModel: NSObject, ObservableObject, NSF self.context.stalenessInterval = 0 super.init() - + self.isOnboarding = !appTPUsed setupFetchedResultsController() diff --git a/Core/AtbAndVariantCleanup.swift b/Core/AtbAndVariantCleanup.swift index 7d0cb0c63e..05824b853c 100644 --- a/Core/AtbAndVariantCleanup.swift +++ b/Core/AtbAndVariantCleanup.swift @@ -25,14 +25,14 @@ public class AtbAndVariantCleanup { static func cleanup(statisticsStorage: StatisticsStore = StatisticsUserDefaults(), variantManager: VariantManager = DefaultVariantManager()) { - + guard let variant = statisticsStorage.variant else { return } // clean up ATB if let atb = statisticsStorage.atb, atb.hasSuffix(variant) { statisticsStorage.atb = String(atb.dropLast(variant.count)) } - + // remove existing variant if not in an active experiment if variantManager.currentVariant == nil { statisticsStorage.variant = nil diff --git a/Core/BookmarkObjects.swift b/Core/BookmarkObjects.swift index f6e66ae86f..53719f279b 100644 --- a/Core/BookmarkObjects.swift +++ b/Core/BookmarkObjects.swift @@ -33,22 +33,22 @@ private struct Constants { public protocol Bookmark: BookmarkItem { var url: URL? { get set } - + var displayTitle: String? { get } } public extension Bookmark { - + var displayTitle: String? { let host = url?.host?.droppingWwwPrefix() ?? url?.absoluteString - + var displayTitle = (title?.isEmpty ?? true) ? host : title - + if let url = url, url.isDuckDuckGo, let title = displayTitle, title.hasSuffix(Constants.ddgSuffix) { displayTitle = String(title.dropLast(Constants.ddgSuffix.count)) } - + return displayTitle } } @@ -58,7 +58,7 @@ public protocol BookmarkFolder: BookmarkItem { } public extension BookmarkFolder { - + var numberOfChildrenDeep: Int { guard let children = children else { return 0 } return children.reduce(children.count) { $0 + (($1 as? BookmarkFolder)?.numberOfChildrenDeep ?? 0) } diff --git a/Core/BookmarksCachingSearch.swift b/Core/BookmarksCachingSearch.swift index 09247b71c6..abbaaff361 100644 --- a/Core/BookmarksCachingSearch.swift +++ b/Core/BookmarksCachingSearch.swift @@ -37,30 +37,30 @@ public protocol BookmarksStringSearchResult { } public protocol BookmarksSearchStore { - + var dataDidChange: AnyPublisher { get } - + func bookmarksAndFavorites(completion: @escaping ([BookmarksCachingSearch.ScoredBookmark]) -> Void) } public class CoreDataBookmarksSearchStore: BookmarksSearchStore { - + private let bookmarksStore: CoreDataDatabase - + private let subject = PassthroughSubject() public var dataDidChange: AnyPublisher - + public init(bookmarksStore: CoreDataDatabase) { self.bookmarksStore = bookmarksStore self.dataDidChange = self.subject.eraseToAnyPublisher() - + registerForCoreDataStorageNotifications() } - + public func bookmarksAndFavorites(completion: @escaping ([BookmarksCachingSearch.ScoredBookmark]) -> Void) { - + let context = bookmarksStore.makeContext(concurrencyType: .privateQueueConcurrencyType) - + let fetchRequest = NSFetchRequest(entityName: "BookmarkEntity") fetchRequest.predicate = NSPredicate( format: "%K = false AND %K == NO", @@ -72,7 +72,7 @@ public class CoreDataBookmarksSearchStore: BookmarksSearchStore { #keyPath(BookmarkEntity.url), #keyPath(BookmarkEntity.objectID)] fetchRequest.relationshipKeyPathsForPrefetching = [#keyPath(BookmarkEntity.favoriteFolders)] - + context.perform { let result = try? context.fetch(fetchRequest) as? [[String: Any]] @@ -83,14 +83,14 @@ public class CoreDataBookmarksSearchStore: BookmarksSearchStore { } } } - + private func registerForCoreDataStorageNotifications() { NotificationCenter.default.addObserver(self, selector: #selector(coreDataDidSave), name: NSManagedObjectContext.didSaveObjectsNotification, object: nil) } - + @objc func coreDataDidSave(notification: Notification) { guard let externalContext = notification.object as? NSManagedObjectContext, externalContext.persistentStoreCoordinator == bookmarksStore.coordinator else { return } @@ -106,20 +106,20 @@ public class BookmarksCachingSearch: BookmarksStringSearch { public let url: URL public let isFavorite: Bool var score: Int - + init(objectID: NSManagedObjectID, title: String, url: URL, isFavorite: Bool) { self.objectID = objectID self.title = title self.url = url self.isFavorite = isFavorite - + if isFavorite { score = 0 } else { score = -1 } } - + init?(bookmark: [String: Any]) { guard let title = bookmark[#keyPath(BookmarkEntity.title)] as? String, let urlString = bookmark[#keyPath(BookmarkEntity.url)] as? String, @@ -127,7 +127,7 @@ public class BookmarksCachingSearch: BookmarksStringSearch { let objectID = bookmark[#keyPath(BookmarkEntity.objectID)] as? NSManagedObjectID else { return nil } - + self.init(objectID: objectID, title: title, url: url, @@ -138,7 +138,7 @@ public class BookmarksCachingSearch: BookmarksStringSearch { return Self.init(objectID: objectID, title: title, url: url, isFavorite: !isFavorite) } } - + private let bookmarksStore: BookmarksSearchStore private var cancellable: AnyCancellable? @@ -147,17 +147,17 @@ public class BookmarksCachingSearch: BookmarksStringSearch { self.cancellable = bookmarksStore.dataDidChange.sink { [weak self] _ in self?.refreshCache() } - + loadCache() } public var hasData: Bool { return cachedBookmarksAndFavorites.count > 0 } - + private var cachedBookmarksAndFavorites = [ScoredBookmark]() private var cacheLoadedCondition = RunLoop.ResumeCondition() - + private func loadCache() { bookmarksStore.bookmarksAndFavorites { result in self.cachedBookmarksAndFavorites = result @@ -166,7 +166,7 @@ public class BookmarksCachingSearch: BookmarksStringSearch { } } } - + private var bookmarksAndFavorites: [ScoredBookmark] { RunLoop.current.run(until: cacheLoadedCondition) return cachedBookmarksAndFavorites @@ -182,25 +182,25 @@ public class BookmarksCachingSearch: BookmarksStringSearch { private func score(query: String, input: [ScoredBookmark]) -> [ScoredBookmark] { let query = query.lowercased() let tokens = query.split(separator: " ").filter { !$0.isEmpty }.map { String($0).lowercased() } - + var input = input var result = [ScoredBookmark]() - + for index in 0.. 1 { var matchesAllTokens = true for token in tokens { @@ -210,11 +210,11 @@ public class BookmarksCachingSearch: BookmarksStringSearch { break } } - + if matchesAllTokens { // Score tokenized matches input[index].score += 10 - + // Boost score if first token matches: if let firstToken = tokens.first { // domain - high score boost if domain.starts(with: firstToken) { @@ -242,9 +242,9 @@ public class BookmarksCachingSearch: BookmarksStringSearch { guard hasData else { return [] } - + let bookmarks = bookmarksAndFavorites - + let trimmed = query.trimmingWhitespace() var finalResult = self.score(query: trimmed, input: bookmarks) finalResult = finalResult.sorted { $0.score > $1.score } diff --git a/Core/BookmarksCoreDataStorage.swift b/Core/BookmarksCoreDataStorage.swift index a0428fa7a4..8844e0b7b5 100644 --- a/Core/BookmarksCoreDataStorage.swift +++ b/Core/BookmarksCoreDataStorage.swift @@ -22,10 +22,10 @@ import CoreData import Bookmarks public class LegacyBookmarksCoreDataStorage { - + private let storeLoadedCondition = RunLoop.ResumeCondition() internal var persistentContainer: NSPersistentContainer - + public lazy var viewContext: NSManagedObjectContext = { RunLoop.current.run(until: storeLoadedCondition) let context = persistentContainer.viewContext @@ -33,7 +33,7 @@ public class LegacyBookmarksCoreDataStorage { context.name = Constants.viewContextName return context }() - + public func getTemporaryPrivateContext() -> NSManagedObjectContext { RunLoop.current.run(until: storeLoadedCondition) let context = persistentContainer.newBackgroundContext() @@ -41,10 +41,10 @@ public class LegacyBookmarksCoreDataStorage { context.name = Constants.privateContextName return context } - + private var cachedReadOnlyTopLevelBookmarksFolder: BookmarkFolderManagedObject? private var cachedReadOnlyTopLevelFavoritesFolder: BookmarkFolderManagedObject? - + internal static var managedObjectModel: NSManagedObjectModel { let coreBundle = Bundle(identifier: "com.duckduckgo.mobile.ios.Core")! guard let managedObjectModel = NSManagedObjectModel.mergedModel(from: [coreBundle]) else { @@ -52,41 +52,41 @@ public class LegacyBookmarksCoreDataStorage { } return managedObjectModel } - + private var storeDescription: NSPersistentStoreDescription { return NSPersistentStoreDescription(url: storeURL) } - + public static var defaultStoreURL: URL { let containerURL = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: BookmarksDatabase.Constants.bookmarksGroupID)! return containerURL.appendingPathComponent("\(Constants.databaseName).sqlite") } - + private let storeURL: URL - + public init?(storeURL: URL = defaultStoreURL, createIfNeeded: Bool = false) { if !FileManager.default.fileExists(atPath: storeURL.path), createIfNeeded == false { return nil } - + self.storeURL = storeURL - + persistentContainer = NSPersistentContainer(name: Constants.databaseName, managedObjectModel: Self.managedObjectModel) persistentContainer.persistentStoreDescriptions = [storeDescription] } - + public func removeStore() { - + typealias StoreInfo = (url: URL?, type: String) - + do { var storesToDelete = [StoreInfo]() for store in persistentContainer.persistentStoreCoordinator.persistentStores { storesToDelete.append((url: store.url, type: store.type)) try persistentContainer.persistentStoreCoordinator.remove(store) } - + for (url, type) in storesToDelete { if let url = url { try persistentContainer.persistentStoreCoordinator.destroyPersistentStore(at: url, @@ -97,21 +97,21 @@ public class LegacyBookmarksCoreDataStorage { Pixel.fire(pixel: .bookmarksMigrationCouldNotRemoveOldStore, error: error) } - + try? FileManager.default.removeItem(atPath: storeURL.path) try? FileManager.default.removeItem(atPath: storeURL.path.appending("-wal")) try? FileManager.default.removeItem(atPath: storeURL.path.appending("-shm")) } - + public func loadStoreAndCaches(andMigrate handler: @escaping (NSManagedObjectContext) -> Void = { _ in }) { - + loadStore(andMigrate: handler) - + RunLoop.current.run(until: storeLoadedCondition) cacheReadOnlyTopLevelBookmarksFolder() cacheReadOnlyTopLevelFavoritesFolder() } - + internal func loadStore(andMigrate handler: @escaping (NSManagedObjectContext) -> Void = { _ in }) { persistentContainer = NSPersistentContainer(name: Constants.databaseName, managedObjectModel: Self.managedObjectModel) @@ -120,7 +120,7 @@ public class LegacyBookmarksCoreDataStorage { if let error = error { fatalError("Unable to load persistent stores: \(error)") } - + let context = NSManagedObjectContext(concurrencyType: .privateQueueConcurrencyType) context.persistentStoreCoordinator = self.persistentContainer.persistentStoreCoordinator context.name = "Migration" @@ -154,14 +154,14 @@ public class LegacyBookmarksCoreDataStorage { // MARK: public interface extension LegacyBookmarksCoreDataStorage { - + public var topLevelBookmarksFolder: BookmarkFolderManagedObject? { guard let folder = cachedReadOnlyTopLevelBookmarksFolder else { return nil } return folder } - + public var topLevelFavoritesFolder: BookmarkFolderManagedObject? { guard let folder = cachedReadOnlyTopLevelFavoritesFolder else { return nil @@ -185,7 +185,7 @@ extension LegacyBookmarksCoreDataStorage { case favorite case bookmark } - + /* This function will return nil if the database desired structure is not met i.e: If there are more than one root level folder OR @@ -193,16 +193,16 @@ extension LegacyBookmarksCoreDataStorage { */ internal func fetchReadOnlyTopLevelFolder(withFolderType folderType: TopLevelFolderType) -> BookmarkFolderManagedObject? { - + var folder: BookmarkFolderManagedObject? - + viewContext.performAndWait { let fetchRequest = NSFetchRequest(entityName: Constants.folderClassName) fetchRequest.predicate = NSPredicate(format: "%K == nil AND %K == %@", #keyPath(BookmarkManagedObject.parent), #keyPath(BookmarkManagedObject.isFavorite), NSNumber(value: folderType == .favorite)) - + let results = try? viewContext.fetch(fetchRequest) guard (results?.count ?? 0) == 1, let fetchedFolder = results?.first else { @@ -213,11 +213,11 @@ extension LegacyBookmarksCoreDataStorage { } return folder } - + internal func cacheReadOnlyTopLevelBookmarksFolder() { guard let folder = fetchReadOnlyTopLevelFolder(withFolderType: .bookmark) else { fixFolderDataStructure(withFolderType: .bookmark) - + // https://app.asana.com/0/414709148257752/1202779945035904/f guard let fixedFolder = fetchReadOnlyTopLevelFolder(withFolderType: .bookmark) else { Pixel.fire(pixel: .debugCouldNotFixBookmarkFolder) @@ -229,11 +229,11 @@ extension LegacyBookmarksCoreDataStorage { } self.cachedReadOnlyTopLevelBookmarksFolder = folder } - + internal func cacheReadOnlyTopLevelFavoritesFolder() { guard let folder = fetchReadOnlyTopLevelFolder(withFolderType: .favorite) else { fixFolderDataStructure(withFolderType: .favorite) - + // https://app.asana.com/0/414709148257752/1202779945035904/f guard let fixedFolder = fetchReadOnlyTopLevelFolder(withFolderType: .favorite) else { Pixel.fire(pixel: .debugCouldNotFixFavoriteFolder) @@ -266,25 +266,25 @@ extension LegacyBookmarksCoreDataStorage { // This is a temporary workaround, do not use the following functions for anything else extension LegacyBookmarksCoreDataStorage { - + private func deleteExtraOrphanedFolders(_ orphanedFolders: [BookmarkFolderManagedObject], onContext context: NSManagedObjectContext, withFolderType folderType: TopLevelFolderType) { let count = orphanedFolders.count let pixelParam = [PixelParameters.bookmarkErrorOrphanedFolderCount: "\(count)"] - + if folderType == .favorite { Pixel.fire(pixel: .debugFavoriteOrphanFolderNew, withAdditionalParameters: pixelParam) } else { Pixel.fire(pixel: .debugBookmarkOrphanFolderNew, withAdditionalParameters: pixelParam) } - + // Sort all orphaned folders by number of children let sorted = orphanedFolders.sorted { ($0.children?.count ?? 0) > ($1.children?.count ?? 0) } - + // Get the folder with the highest number of children let folderWithMoreChildren = sorted.first - + // Separate the other folders let otherFolders = sorted.suffix(from: 1) @@ -297,7 +297,7 @@ extension LegacyBookmarksCoreDataStorage { context.delete(folder) } } - + /* Top level (orphaned) folders need to match its type i.e: Favorites and Bookmarks each have their own root folder @@ -318,7 +318,7 @@ extension LegacyBookmarksCoreDataStorage { bookmarksFetchRequest.returnsObjectsAsFaults = false let bookmarks = try? context.fetch(bookmarksFetchRequest) - + if bookmarks?.count ?? 0 > 0 { if folderType == .favorite { Pixel.fire(pixel: .debugMissingTopFolderFixHasFavorites) @@ -326,7 +326,7 @@ extension LegacyBookmarksCoreDataStorage { Pixel.fire(pixel: .debugMissingTopFolderFixHasBookmarks) } } - + // Create root folder for the specified folder type let bookmarksFolder: BookmarkFolderManagedObject if folderType == .favorite { @@ -334,31 +334,31 @@ extension LegacyBookmarksCoreDataStorage { } else { bookmarksFolder = Self.rootFolderManagedObject(context) } - + // Assign all bookmarks to the parent folder bookmarks?.forEach { $0.parent = bookmarksFolder } } - + internal func fixFolderDataStructure(withFolderType folderType: TopLevelFolderType) { let privateContext = getTemporaryPrivateContext() - + privateContext.performAndWait { let fetchRequest = NSFetchRequest(entityName: Constants.folderClassName) fetchRequest.predicate = NSPredicate(format: "%K == nil AND %K == %@", #keyPath(BookmarkManagedObject.parent), #keyPath(BookmarkManagedObject.isFavorite), NSNumber(value: folderType == .favorite)) - + let results = try? privateContext.fetch(fetchRequest) - + if let orphanedFolders = results, orphanedFolders.count > 1 { deleteExtraOrphanedFolders(orphanedFolders, onContext: privateContext, withFolderType: folderType) } else { createMissingTopLevelFolder(onContext: privateContext, withFolderType: folderType) } - + do { try privateContext.save() } catch { diff --git a/Core/BookmarksExporter.swift b/Core/BookmarksExporter.swift index 294b59a076..5a46ec1749 100644 --- a/Core/BookmarksExporter.swift +++ b/Core/BookmarksExporter.swift @@ -42,7 +42,7 @@ public struct BookmarksExporter { func exportBookmarksToContent() throws -> String { var content = [Template.header] - + let context = coreDataStorage.makeContext(concurrencyType: .mainQueueConcurrencyType) guard let rootFolder = BookmarkUtils.fetchRootFolder(context) else { throw BookmarksExporterError.brokenDatabaseStructure diff --git a/Core/BookmarksModelsErrorHandling.swift b/Core/BookmarksModelsErrorHandling.swift index b60ecdd4d8..3539714f78 100644 --- a/Core/BookmarksModelsErrorHandling.swift +++ b/Core/BookmarksModelsErrorHandling.swift @@ -32,7 +32,7 @@ public class BookmarksModelsErrorHandling: EventMapping { var domainEvent: Pixel.Event? var params = [String: String]() switch event { - + case .bookmarkFolderExpected: domainEvent = .bookmarkFolderExpected case .bookmarksListIndexNotMatchingBookmark: @@ -49,12 +49,12 @@ public class BookmarksModelsErrorHandling: EventMapping { domainEvent = .indexOutOfRange(model) case .saveFailed(let model): domainEvent = .saveFailed(model) - + if let error = error as? NSError { let processedErrors = CoreDataErrorsParser.parse(error: error) params = processedErrors.errorPixelParameters } - + case .missingParent(let object): domainEvent = .missingParent(object) } @@ -68,14 +68,14 @@ public class BookmarksModelsErrorHandling: EventMapping { } } } - + override init(mapping: @escaping EventMapping.Mapping) { fatalError("Use init()") } } public extension BookmarkEditorViewModel { - + convenience init(editingEntityID: NSManagedObjectID, bookmarksDatabase: CoreDataDatabase, favoritesDisplayMode: FavoritesDisplayMode, @@ -84,9 +84,9 @@ public extension BookmarkEditorViewModel { bookmarksDatabase: bookmarksDatabase, favoritesDisplayMode: favoritesDisplayMode, errorEvents: BookmarksModelsErrorHandling(syncService: syncService)) - + } - + convenience init(creatingFolderWithParentID parentFolderID: NSManagedObjectID?, bookmarksDatabase: CoreDataDatabase, favoritesDisplayMode: FavoritesDisplayMode, @@ -99,7 +99,7 @@ public extension BookmarkEditorViewModel { } public extension BookmarkListViewModel { - + convenience init(bookmarksDatabase: CoreDataDatabase, parentID: NSManagedObjectID?, favoritesDisplayMode: FavoritesDisplayMode, @@ -112,14 +112,14 @@ public extension BookmarkListViewModel { } public extension FavoritesListViewModel { - + convenience init(bookmarksDatabase: CoreDataDatabase, favoritesDisplayMode: FavoritesDisplayMode) { self.init(bookmarksDatabase: bookmarksDatabase, errorEvents: BookmarksModelsErrorHandling(), favoritesDisplayMode: favoritesDisplayMode) } } public extension MenuBookmarksViewModel { - + convenience init(bookmarksDatabase: CoreDataDatabase, syncService: DDGSyncing?) { self.init(bookmarksDatabase: bookmarksDatabase, errorEvents: BookmarksModelsErrorHandling(syncService: syncService)) } diff --git a/Core/Configuration.swift b/Core/Configuration.swift index 2cd73a6266..e7c9dcb631 100644 --- a/Core/Configuration.swift +++ b/Core/Configuration.swift @@ -33,5 +33,5 @@ public extension Configuration { case .FBConfig: return "FBConfig" } } - + } diff --git a/Core/ContentBlockerRulesLists.swift b/Core/ContentBlockerRulesLists.swift index b7de2e2d63..89407284fc 100644 --- a/Core/ContentBlockerRulesLists.swift +++ b/Core/ContentBlockerRulesLists.swift @@ -20,17 +20,17 @@ import BrowserServicesKit public final class ContentBlockerRulesLists: DefaultContentBlockerRulesListsSource { - + private let adClickAttribution: AdClickAttributionFeature - + init(trackerDataManager: TrackerDataManager, adClickAttribution: AdClickAttributionFeature) { self.adClickAttribution = adClickAttribution super.init(trackerDataManager: trackerDataManager) } - + public override var contentBlockerRulesLists: [ContentBlockerRulesList] { var result = super.contentBlockerRulesLists - + if adClickAttribution.isEnabled, let tdsRulesIndex = result.firstIndex(where: { $0.name == Constants.trackerDataSetRulesListName }) { let tdsRules = result[tdsRulesIndex] @@ -42,8 +42,8 @@ public final class ContentBlockerRulesLists: DefaultContentBlockerRulesListsSour result.append(splitRules.1) } } - + return result } - + } diff --git a/Core/ContentBlocking.swift b/Core/ContentBlocking.swift index 2767e48e4d..43b5348b42 100644 --- a/Core/ContentBlocking.swift +++ b/Core/ContentBlocking.swift @@ -22,7 +22,6 @@ import BrowserServicesKit import Combine import Common - public final class ContentBlocking { public static let shared = ContentBlocking() @@ -84,25 +83,25 @@ public final class ContentBlocking { switch event { case .trackerDataParseFailed: domainEvent = .trackerDataParseFailed - + case .trackerDataReloadFailed: domainEvent = .trackerDataReloadFailed - + case .trackerDataCouldNotBeLoaded: domainEvent = .trackerDataCouldNotBeLoaded - + case .privacyConfigurationReloadFailed: domainEvent = .privacyConfigurationReloadFailed - + case .privacyConfigurationParseFailed: domainEvent = .privacyConfigurationParseFailed - + case .privacyConfigurationCouldNotBeLoaded: domainEvent = .privacyConfigurationCouldNotBeLoaded - + case .contentBlockingCompilationFailed(let listName, let component): let defaultTDSListName = DefaultContentBlockerRulesListsSource.Constants.trackerDataSetRulesListName - + let listType: Pixel.Event.CompileRulesListType switch listName { case defaultTDSListName: @@ -116,11 +115,11 @@ public final class ContentBlocking { } domainEvent = .contentBlockingCompilationFailed(listType: listType, component: component) - + case .contentBlockingCompilationTime: domainEvent = .contentBlockingCompilationTime } - + if let error = error { Pixel.fire(pixel: domainEvent, error: error, @@ -132,7 +131,7 @@ public final class ContentBlocking { includedParameters: [], onComplete: onComplete) } - + } public func makeAdClickAttributionDetection(tld: TLD) -> AdClickAttributionDetection { @@ -142,7 +141,7 @@ public final class ContentBlocking { errorReporting: attributionDebugEvents, log: .adAttributionLog) } - + public func makeAdClickAttributionLogic(tld: TLD) -> AdClickAttributionLogic { AdClickAttributionLogic(featureConfig: adClickAttribution, rulesProvider: adClickAttributionRulesProvider, @@ -151,7 +150,7 @@ public final class ContentBlocking { errorReporting: attributionDebugEvents, log: .adAttributionLog) } - + private let attributionEvents = EventMapping { event, _, parameters, _ in var shouldIncludeAppVersion = true let domainEvent: Pixel.Event @@ -164,10 +163,10 @@ public final class ContentBlocking { domainEvent = .adClickAttributionPageLoads shouldIncludeAppVersion = false } - + Pixel.fire(pixel: domainEvent, withAdditionalParameters: parameters ?? [:], includedParameters: shouldIncludeAppVersion ? [.appVersion] : []) } - + private let attributionDebugEvents = EventMapping { event, _, _, _ in let domainEvent: Pixel.Event switch event { @@ -192,29 +191,29 @@ public final class ContentBlocking { case .adAttributionLogicWrongVendorOnFailedCompilation: domainEvent = .adAttributionLogicWrongVendorOnFailedCompilation } - + Pixel.fire(pixel: domainEvent, includedParameters: []) } - + } public class DomainsProtectionUserDefaultsStore: DomainsProtectionStore { - + private struct Keys { static let unprotectedDomains = "com.duckduckgo.contentblocker.whitelist" static let trackerList = "com.duckduckgo.trackerList" } - + private let suiteName: String - + public init(suiteName: String = ContentBlockerStoreConstants.groupName) { self.suiteName = suiteName } - + private var userDefaults: UserDefaults? { return UserDefaults(suiteName: suiteName) } - + public private(set) var unprotectedDomains: Set { get { guard let data = userDefaults?.data(forKey: Keys.unprotectedDomains) else { return Set() } @@ -229,13 +228,13 @@ public class DomainsProtectionUserDefaultsStore: DomainsProtectionStore { userDefaults?.set(data, forKey: Keys.unprotectedDomains) } } - + public func disableProtection(forDomain domain: String) { var domains = unprotectedDomains domains.insert(domain) unprotectedDomains = domains } - + public func enableProtection(forDomain domain: String) { var domains = unprotectedDomains domains.remove(domain) diff --git a/Core/CookieStorage.swift b/Core/CookieStorage.swift index 0349d9841a..0cfc79c970 100644 --- a/Core/CookieStorage.swift +++ b/Core/CookieStorage.swift @@ -34,7 +34,7 @@ public class CookieStorage { } private var userDefaults: UserDefaults - + var isConsumed: Bool { get { return userDefaults.bool(forKey: Keys.consumed, defaultValue: false) @@ -43,7 +43,7 @@ public class CookieStorage { userDefaults.set(newValue, forKey: Keys.consumed) } } - + /// Use the `updateCookies` function rather than the setter which is only visible for testing. var cookies: [HTTPCookie] { get { @@ -54,17 +54,17 @@ public class CookieStorage { cookieData.forEach({ properties[HTTPCookiePropertyKey(rawValue: $0.key)] = $0.value }) - + if let cookie = HTTPCookie(properties: properties) { os_log("read cookie %s %s %s", log: .generalLog, type: .debug, cookie.domain, cookie.name, cookie.value) storedCookies.append(cookie) } } } - + return storedCookies } - + set { var cookies = [[String: Any?]]() newValue.forEach { cookie in @@ -91,16 +91,16 @@ public class CookieStorage { case different case notConsumed } - + /// Update ALL cookies. The absence of cookie domains here indicateds they have been removed by the website, so be sure to call this with all cookies that might need to be persisted even if those websites have not been visited yet. @discardableResult func updateCookies(_ cookies: [HTTPCookie], keepingPreservedLogins preservedLogins: PreserveLogins) -> CookieDomainsOnUpdateDiagnostic { guard isConsumed else { return .notConsumed } - + isConsumed = false - + let persisted = self.cookies - + func cookiesByDomain(_ cookies: [HTTPCookie]) -> [String: [HTTPCookie]] { var byDomain = [String: [HTTPCookie]]() cookies.forEach { cookie in @@ -110,7 +110,7 @@ public class CookieStorage { } return byDomain } - + let updatedCookiesByDomain = cookiesByDomain(cookies) var persistedCookiesByDomain = cookiesByDomain(persisted) @@ -119,16 +119,16 @@ public class CookieStorage { updatedDomains: updatedCookiesByDomain.keys.sorted(), persistedDomains: persistedCookiesByDomain.keys.sorted() ) - + let cookieDomains = Set(updatedCookiesByDomain.keys.map { $0 } + persistedCookiesByDomain.keys.map { $0 }) - + cookieDomains.forEach { persistedCookiesByDomain[$0] = updatedCookiesByDomain[$0] } - + persistedCookiesByDomain.keys.forEach { guard !URL.isDuckDuckGo(domain: $0) else { return } // DDG cookies are for SERP settings only - + if !preservedLogins.isAllowed(cookieDomain: $0) { persistedCookiesByDomain.removeValue(forKey: $0) } @@ -137,10 +137,10 @@ public class CookieStorage { let now = Date() self.cookies = persistedCookiesByDomain.map { $0.value }.joined().compactMap { $0 } .filter { $0.expiresDate == nil || $0.expiresDate! > now } - + return diagnosticResult } - + private func evaluateDomains(updatedDomains: [String], persistedDomains: [String]) -> CookieDomainsOnUpdateDiagnostic { if persistedDomains.isEmpty { return .empty @@ -152,5 +152,5 @@ public class CookieStorage { return .different } } - + } diff --git a/Core/DDGPersistenceContainer.swift b/Core/DDGPersistenceContainer.swift index ef61f85edc..5bec649281 100644 --- a/Core/DDGPersistenceContainer.swift +++ b/Core/DDGPersistenceContainer.swift @@ -41,7 +41,7 @@ public class DDGPersistenceContainer { private static func createPersistenceStoreCoordinator(name: String, model: NSManagedObjectModel) -> NSPersistentStoreCoordinator? { guard let storeURL = storeURL(for: name) else { return nil } - + let persistenceStoreCoordinator = NSPersistentStoreCoordinator(managedObjectModel: model) let options = [ NSMigratePersistentStoresAutomaticallyOption: true, NSInferMappingModelAutomaticallyOption: true ] @@ -53,7 +53,7 @@ public class DDGPersistenceContainer { return persistenceStoreCoordinator } - + public static func storeURL(for name: String) -> URL? { let fileManager = FileManager.default guard let docsDir = fileManager.urls(for: .documentDirectory, in: .userDomainMask).last else { return nil } diff --git a/Core/DailyPixel.swift b/Core/DailyPixel.swift index ef878eae28..e257b9828b 100644 --- a/Core/DailyPixel.swift +++ b/Core/DailyPixel.swift @@ -30,17 +30,17 @@ import Foundation /// In those scenarios a 'DailyPixelError' is returned denoting the reason. /// public final class DailyPixel { - + public enum Error: Swift.Error { - + case alreadyFired - + } - + private enum Constant { - + static let dailyPixelStorageIdentifier = "com.duckduckgo.daily.pixel.storage" - + } private static let storage: UserDefaults = UserDefaults(suiteName: Constant.dailyPixelStorageIdentifier)! @@ -53,7 +53,7 @@ public final class DailyPixel { withAdditionalParameters params: [String: String] = [:], includedParameters: [Pixel.QueryParameters] = [.atb, .appVersion], onComplete: @escaping (Swift.Error?) -> Void = { _ in }) { - + if !pixel.hasBeenFiredToday(dailyPixelStorage: storage) { Pixel.fire(pixel: pixel, withAdditionalParameters: params, @@ -96,20 +96,20 @@ public final class DailyPixel { onComplete: onCountComplete ) } - + private static func updatePixelLastFireDate(pixel: Pixel.Event) { storage.set(Date(), forKey: pixel.name) } - + } private extension Pixel.Event { - + func hasBeenFiredToday(dailyPixelStorage: UserDefaults) -> Bool { if let lastFireDate = dailyPixelStorage.object(forKey: name) as? Date { return Date().isSameDay(lastFireDate) } return false } - + } diff --git a/Core/Database.swift b/Core/Database.swift index 0c682643c3..a9fc351606 100644 --- a/Core/Database.swift +++ b/Core/Database.swift @@ -23,17 +23,17 @@ import CoreData import Persistence public class Database { - + fileprivate struct Constants { static let databaseGroupID = "\(Global.groupIdPrefix).database" - + static let databaseName = "Database" } - + public static let shared = makeCoreDataDatabase() - + static func makeCoreDataDatabase() -> CoreDataDatabase { - + guard let appRatingModel = CoreDataDatabase.loadModel(from: .main, named: "AppRatingPrompt"), let remoteMessagingModel = CoreDataDatabase.loadModel(from: .main, named: "RemoteMessaging"), let managedObjectModel = NSManagedObjectModel(byMerging: [appRatingModel, @@ -41,7 +41,7 @@ public class Database { HTTPSUpgrade.managedObjectModel]) else { fatalError("No DB scheme found") } - + let url = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: Database.Constants.databaseGroupID)! return CoreDataDatabase(name: Constants.databaseName, containerLocation: url, diff --git a/Core/Debounce.swift b/Core/Debounce.swift index b96204614b..6994c3dda3 100644 --- a/Core/Debounce.swift +++ b/Core/Debounce.swift @@ -20,17 +20,17 @@ import Foundation public class Debounce { - + private let queue: DispatchQueue private let interval: TimeInterval - + private var currentWorkItem = DispatchWorkItem(block: {}) - + public init(queue: DispatchQueue, seconds: TimeInterval) { self.queue = queue self.interval = seconds } - + public func schedule(_ block: @escaping (() -> Void)) { currentWorkItem.cancel() currentWorkItem = DispatchWorkItem(block: { block() }) diff --git a/Core/DebugUserScript.swift b/Core/DebugUserScript.swift index d3c83f93f7..1c7ee0ea47 100644 --- a/Core/DebugUserScript.swift +++ b/Core/DebugUserScript.swift @@ -22,47 +22,47 @@ import WebKit import UserScript public class DebugUserScript: NSObject, UserScript { - + struct MessageNames { - + static let signpost = "signpost" static let log = "log" - + } - + public lazy var source: String = { return "" }() - + public var injectionTime: WKUserScriptInjectionTime = .atDocumentStart - + public var forMainFrameOnly: Bool = false - + public var messageNames: [String] = [ MessageNames.signpost, MessageNames.log ] - + public weak var instrumentation: TabInstrumentation? public func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) { switch message.name { - + case MessageNames.signpost: handleSignpost(message: message) - + case MessageNames.log: handleLog(message: message) - + default: break } } - + private func handleLog(message: WKScriptMessage) { os_log("%s", log: .generalLog, type: .debug, String(describing: message.body)) } - + private func handleSignpost(message: WKScriptMessage) { guard let dict = message.body as? [String: Any], let event = dict["event"] as? String else { return } - + if event == "Request Allowed" { if let elapsedTimeInMs = dict["time"] as? Double, let url = dict["url"] as? String { diff --git a/Core/DefaultVariantManager.swift b/Core/DefaultVariantManager.swift index 3072c39ab7..b17b8de45d 100644 --- a/Core/DefaultVariantManager.swift +++ b/Core/DefaultVariantManager.swift @@ -31,7 +31,7 @@ extension FeatureName { } public struct VariantIOS: Variant { - + struct When { static let always = { return true } static let padDevice = { return UIDevice.current.userInterfaceIdiom == .pad } @@ -39,7 +39,7 @@ public struct VariantIOS: Variant { static let inRequiredCountry = { return ["AU", "AT", "DK", "FI", "FR", "DE", "IT", "IE", "NZ", "NO", "ES", "SE", "GB"] .contains(where: { Locale.current.regionCode == $0 }) } - + static let inEnglish = { return Locale.current.languageCode == "en" } static let iOS15 = { () -> Bool in @@ -80,23 +80,23 @@ public struct VariantIOS: Variant { } public protocol VariantRNG { - + func nextInt(upperBound: Int) -> Int - + } public class DefaultVariantManager: VariantManager { - + public var currentVariant: Variant? { let variantName = ProcessInfo.processInfo.environment["VARIANT", default: storage.variant ?? "" ] return variants.first(where: { $0.name == variantName }) } - + private let variants: [Variant] private let storage: StatisticsStore private let rng: VariantRNG private let returningUserMeasurement: ReturnUserMeasurement - + init(variants: [Variant], storage: StatisticsStore, rng: VariantRNG, @@ -120,30 +120,30 @@ public class DefaultVariantManager: VariantManager { public func isSupported(feature: FeatureName) -> Bool { return currentVariant?.features.contains(feature) ?? false } - + public func assignVariantIfNeeded(_ newInstallCompletion: (VariantManager) -> Void) { guard !storage.hasInstallStatistics else { os_log("no new variant needed for existing user", log: .generalLog, type: .debug) return } - + if let variant = currentVariant { os_log("already assigned variant: %s", log: .generalLog, type: .debug, String(describing: variant)) return } - + guard let variant = selectVariant() else { os_log("Failed to assign variant", log: .generalLog, type: .debug) - + // it's possible this failed because there are none to assign, we should still let new install logic execute _ = newInstallCompletion(self) return } - + storage.variant = variant.name newInstallCompletion(self) } - + private func selectVariant() -> Variant? { if returningUserMeasurement.isReturningUser { return VariantIOS.returningUser @@ -151,7 +151,7 @@ public class DefaultVariantManager: VariantManager { let totalWeight = variants.reduce(0, { $0 + $1.weight }) let randomPercent = rng.nextInt(upperBound: totalWeight) - + var runningTotal = 0 for variant in variants { runningTotal += variant.weight @@ -159,19 +159,19 @@ public class DefaultVariantManager: VariantManager { return variant.isIncluded() ? variant : nil } } - + return nil } - + } public class Arc4RandomUniformVariantRNG: VariantRNG { - + public init() { } - + public func nextInt(upperBound: Int) -> Int { // swiftlint:disable:next legacy_random return Int(arc4random_uniform(UInt32(upperBound))) } - + } diff --git a/Core/EtagStorage.swift b/Core/EtagStorage.swift index b3644ee70d..3b02fe96e7 100644 --- a/Core/EtagStorage.swift +++ b/Core/EtagStorage.swift @@ -22,26 +22,26 @@ import Foundation import Configuration public protocol BlockerListETagStorage { - + func saveEtag(_ etag: String, for configuration: Configuration) func loadEtag(for configuration: Configuration) -> String? - + } public struct UserDefaultsETagStorage: BlockerListETagStorage { - + private let defaults = UserDefaults(suiteName: "com.duckduckgo.blocker-list.etags") - + public init() { } - + public func loadEtag(for configuration: Configuration) -> String? { let etag = defaults?.string(forKey: configuration.storeKey) os_log("stored etag for %s %s", log: .generalLog, type: .debug, configuration.storeKey, etag ?? "nil") return etag } - + public func saveEtag(_ etag: String, for configuration: Configuration) { defaults?.set(etag, forKey: configuration.storeKey) } - + } diff --git a/Core/FaviconSourcesProvider.swift b/Core/FaviconSourcesProvider.swift index 17e3f38ade..4690e805bb 100644 --- a/Core/FaviconSourcesProvider.swift +++ b/Core/FaviconSourcesProvider.swift @@ -20,35 +20,35 @@ import Foundation protocol FaviconSourcesProvider { - + func mainSource(forDomain: String) -> URL? - + func additionalSources(forDomain: String) -> [URL] - + } class DefaultFaviconSourcesProvider: FaviconSourcesProvider { - + enum ImageNames: String { - + case appleTouch = "apple-touch-icon.png" case favicon = "favicon.ico" - + } - + func mainSource(forDomain domain: String) -> URL? { return imageSource(forDomain: domain, imageName: ImageNames.appleTouch, secure: true) } - + func additionalSources(forDomain domain: String) -> [URL] { return [ imageSource(forDomain: domain, imageName: .favicon, secure: true), imageSource(forDomain: domain, imageName: .favicon, secure: false) ].compactMap { $0 } } - + private func imageSource(forDomain domain: String, imageName: ImageNames, secure: Bool) -> URL? { return URL(string: (secure ? "https" : "http") + "://" + domain + "/" + imageName.rawValue) } - + } diff --git a/Core/FaviconUserScript.swift b/Core/FaviconUserScript.swift index 6b7d48ae8d..b3563bfa4e 100644 --- a/Core/FaviconUserScript.swift +++ b/Core/FaviconUserScript.swift @@ -27,7 +27,7 @@ public protocol FaviconUserScriptDelegate: NSObjectProtocol { } public class FaviconUserScript: NSObject, UserScript { - + public var source: String = """ (function() { @@ -35,7 +35,7 @@ public class FaviconUserScript: NSObject, UserScript { function getFavicon() { return findFavicons()[0]; }; - + function findFavicons() { var selectors = [ @@ -50,7 +50,7 @@ public class FaviconUserScript: NSObject, UserScript { var icons = document.head.querySelectorAll(selector); for (var i = 0; i < icons.length; i++) { var href = icons[i].href; - + // Exclude SVGs since we can't handle them if (href.indexOf("svg") >= 0 || (icons[i].type && icons[i].type.indexOf("svg") >= 0)) { continue; @@ -72,15 +72,15 @@ public class FaviconUserScript: NSObject, UserScript { }) (); """ - + public var injectionTime: WKUserScriptInjectionTime = .atDocumentEnd - + public var forMainFrameOnly: Bool = true - + public var messageNames: [String] = ["faviconFound"] - + public weak var delegate: FaviconUserScriptDelegate? - + public func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) { let url: URL? @@ -93,5 +93,5 @@ public class FaviconUserScript: NSObject, UserScript { let host = message.messageHost delegate?.faviconUserScript(self, didRequestUpdateFaviconForHost: host, withUrl: url) } - + } diff --git a/Core/FileStore.swift b/Core/FileStore.swift index 04fe3f41fb..5696711dbf 100644 --- a/Core/FileStore.swift +++ b/Core/FileStore.swift @@ -21,11 +21,11 @@ import Foundation import Configuration public class FileStore { - + private let groupIdentifier: String = ContentBlockerStoreConstants.groupName public init() { } - + public func persist(_ data: Data, for configuration: Configuration) throws { do { try data.write(to: persistenceLocation(for: configuration), options: .atomic) @@ -34,26 +34,26 @@ public class FileStore { throw error } } - + func removeData(forFile file: String) -> Bool { var fileUrl = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: groupIdentifier) fileUrl = fileUrl!.appendingPathComponent(file) guard let fileUrl = fileUrl else { return false } guard FileManager.default.fileExists(atPath: fileUrl.path) else { return true } - + do { try FileManager.default.removeItem(at: fileUrl) } catch { return false } - + return true } - + public func loadAsString(for configuration: Configuration) -> String? { try? String(contentsOf: persistenceLocation(for: configuration)) } - + public func loadAsData(for configuration: Configuration) -> Data? { do { return try Data(contentsOf: persistenceLocation(for: configuration)) @@ -65,7 +65,7 @@ public class FileStore { return nil } } - + func hasData(for configuration: Configuration) -> Bool { FileManager.default.fileExists(atPath: persistenceLocation(for: configuration).path) } diff --git a/Core/HTTPSUpgradeParser.swift b/Core/HTTPSUpgradeParser.swift index 85dbd9cee7..1d144d390e 100644 --- a/Core/HTTPSUpgradeParser.swift +++ b/Core/HTTPSUpgradeParser.swift @@ -21,7 +21,7 @@ import Foundation import BrowserServicesKit public class HTTPSUpgradeParser { - + static func convertExcludedDomainsData(_ data: Data) throws -> [String] { do { let decoder = JSONDecoder() @@ -32,7 +32,7 @@ public class HTTPSUpgradeParser { throw JsonError.typeMismatch } } - + static func convertBloomFilterSpecification(fromJSONData data: Data) throws -> HTTPSBloomFilterSpecification { do { let decoder = JSONDecoder() @@ -43,5 +43,5 @@ public class HTTPSUpgradeParser { throw JsonError.typeMismatch } } - + } diff --git a/Core/HistoryManager.swift b/Core/HistoryManager.swift index 957a8d327e..49b2049f78 100644 --- a/Core/HistoryManager.swift +++ b/Core/HistoryManager.swift @@ -33,5 +33,5 @@ class HistoryManager { func isHistoryFeatureEnabled() -> Bool { return privacyConfig.isEnabled(featureKey: .history) && variantManager.isSupported(feature: .history) } - + } diff --git a/Core/Instruments.swift b/Core/Instruments.swift index 54a61ee677..5a059931dd 100644 --- a/Core/Instruments.swift +++ b/Core/Instruments.swift @@ -21,28 +21,28 @@ import Foundation import os.signpost public class Instruments { - + public enum TimedEvent: String { case fetchingContentBlockerData - + case loadingDisconnectMeStore case loadingEasylistStore - + case tabInitialisation - + case clearingData case injectScripts } - + static public let shared = Instruments() - + static var eventsLog = OSLog(subsystem: "com.duckduckgo.instrumentation", category: "Events") public func startTimedEvent(_ event: TimedEvent, info: String? = nil) -> Any? { if #available(iOSApplicationExtension 12.0, *) { let id = OSSignpostID(log: Instruments.eventsLog) - + os_signpost(.begin, log: Instruments.eventsLog, name: "Timed Event", @@ -52,7 +52,7 @@ public class Instruments { } return nil } - + public func endTimedEvent(for spid: Any?, result: String? = nil) { if #available(iOSApplicationExtension 12.0, *), let id = spid as? OSSignpostID { diff --git a/Core/InternalUserStore.swift b/Core/InternalUserStore.swift index 5d58edb0a7..ba0b3210a2 100644 --- a/Core/InternalUserStore.swift +++ b/Core/InternalUserStore.swift @@ -23,7 +23,7 @@ import BrowserServicesKit public class InternalUserStore: InternalUserStoring { public init() { } - + @UserDefaultsWrapper(key: .featureFlaggingDidVerifyInternalUser, defaultValue: false) public var isInternalUser: Bool } diff --git a/Core/LegacyBookmarksStoreMigration.swift b/Core/LegacyBookmarksStoreMigration.swift index dfd6a4c371..0cb3d417ac 100644 --- a/Core/LegacyBookmarksStoreMigration.swift +++ b/Core/LegacyBookmarksStoreMigration.swift @@ -22,12 +22,12 @@ import CoreData import Bookmarks public class LegacyBookmarksStoreMigration { - + internal enum LegacyTopLevelFolderType { case favorite case bookmark } - + public static func migrate(from legacyStorage: LegacyBookmarksCoreDataStorage?, to context: NSManagedObjectContext) { if let legacyStorage = legacyStorage { @@ -50,37 +50,37 @@ public class LegacyBookmarksStoreMigration { } } } - + private static func fetchTopLevelFolder(_ folderType: LegacyTopLevelFolderType, in context: NSManagedObjectContext) -> [BookmarkFolderManagedObject] { - + let fetchRequest = NSFetchRequest(entityName: LegacyBookmarksCoreDataStorage.Constants.folderClassName) fetchRequest.predicate = NSPredicate(format: "%K == nil AND %K == %@", #keyPath(BookmarkManagedObject.parent), #keyPath(BookmarkManagedObject.isFavorite), NSNumber(value: folderType == .favorite)) - + guard let results = try? context.fetch(fetchRequest) else { return [] } - + // In case of corruption, we can cat more than one 'root' return results } // swiftlint:disable cyclomatic_complexity // swiftlint:disable function_body_length - + private static func migrate(source: NSManagedObjectContext, destination: NSManagedObjectContext) { - + // Do not migrate more than once guard BookmarkUtils.fetchRootFolder(destination) == nil else { Pixel.fire(pixel: .bookmarksMigrationAlreadyPerformed) return } - + BookmarkUtils.prepareFoldersStructure(in: destination) - + guard let newRoot = BookmarkUtils.fetchRootFolder(destination), let newFavoritesRoot = BookmarkUtils.fetchFavoritesFolder(withUUID: FavoritesFolderID.unified.rawValue, in: destination), let newMobileFavoritesRoot = BookmarkUtils.fetchFavoritesFolder(withUUID: FavoritesFolderID.mobile.rawValue, in: destination) else { @@ -88,37 +88,37 @@ public class LegacyBookmarksStoreMigration { Thread.sleep(forTimeInterval: 2) fatalError("Could not write to Bookmarks DB") } - + // Fetch all 'roots' in case we had some kind of inconsistency and duplicated objects let bookmarkRoots = fetchTopLevelFolder(.bookmark, in: source) let favoriteRoots = fetchTopLevelFolder(.favorite, in: source) - + var index = 0 var folderMap = [NSManagedObjectID: BookmarkEntity]() - + var favoritesToMigrate = [BookmarkItemManagedObject]() var bookmarksToMigrate = [BookmarkItemManagedObject]() - + // Map old roots to new one, prepare list of top level bookmarks to migrate for folder in favoriteRoots { folderMap[folder.objectID] = newRoot - + favoritesToMigrate.append(contentsOf: folder.children?.array as? [BookmarkItemManagedObject] ?? []) } - + for folder in bookmarkRoots { folderMap[folder.objectID] = newRoot - + bookmarksToMigrate.append(contentsOf: folder.children?.array as? [BookmarkItemManagedObject] ?? []) } - + var urlToBookmarkMap = [URL: BookmarkEntity]() - + // Iterate over bookmarks to migrate while index < bookmarksToMigrate.count { - + let objectToMigrate = bookmarksToMigrate[index] - + guard let parent = objectToMigrate.parent, let newParent = folderMap[parent.objectID], let title = objectToMigrate.title else { @@ -126,37 +126,37 @@ public class LegacyBookmarksStoreMigration { index += 1 continue } - + if let folder = objectToMigrate as? BookmarkFolderManagedObject { let newFolder = BookmarkEntity.makeFolder(title: title, parent: newParent, context: destination) folderMap[folder.objectID] = newFolder - + if let children = folder.children?.array as? [BookmarkItemManagedObject] { bookmarksToMigrate.append(contentsOf: children) } - + } else if let bookmark = objectToMigrate as? BookmarkManagedObject, let url = bookmark.url { - + let newBookmark = BookmarkEntity.makeBookmark(title: title, url: url.absoluteString, parent: newParent, context: destination) - + urlToBookmarkMap[url] = newBookmark } - + index += 1 } - + // Process favorites starting from the last one, so we preserve the order while adding at begining for favorite in favoritesToMigrate.reversed() { guard let favorite = favorite as? BookmarkManagedObject, let title = favorite.title, let url = favorite.url else { continue } - + let bookmark = { if let existingBookmark = urlToBookmarkMap[url] { return existingBookmark @@ -173,12 +173,12 @@ public class LegacyBookmarksStoreMigration { bookmark.addToFavorites(insertAt: 0, favoritesRoot: newMobileFavoritesRoot) } - + do { try destination.save(onErrorFire: .bookmarksMigrationFailed) } catch { destination.reset() - + BookmarkUtils.prepareLegacyFoldersStructure(in: destination) do { try destination.save(onErrorFire: .bookmarksMigrationCouldNotPrepareDatabaseOnFailedMigration) diff --git a/Core/Link.swift b/Core/Link.swift index 0dcbf45e17..cb868d81f9 100644 --- a/Core/Link.swift +++ b/Core/Link.swift @@ -20,7 +20,7 @@ import Foundation public class Link: NSObject, NSCoding { - + struct Constants { static let ddgSuffix = " at DuckDuckGo" } @@ -30,11 +30,11 @@ public class Link: NSObject, NSCoding { static let url = "url" static let localPath = "localPath" } - + public let title: String? public let url: URL public let localFileURL: URL? - + public var displayTitle: String { let host = url.host?.droppingWwwPrefix() ?? url.absoluteString diff --git a/Core/LocaleExtension.swift b/Core/LocaleExtension.swift index cfbb0126bb..3cb01d997c 100644 --- a/Core/LocaleExtension.swift +++ b/Core/LocaleExtension.swift @@ -20,7 +20,7 @@ import Foundation extension Locale { - + public var isRegionInEurope: Bool { ["AD", "AL", "AT", "AZ", "BA", "BE", "BG", "BY", "CH", "CY", "CZ", "DE", "DK", "EE", "ES", "FI", "FR", "GE", "GI", "GR", "HR", "HU", "IE", "IS", "IT", "KZ", "LI", "LT", "LU", "LV", diff --git a/Core/LoginFormDetectionUserScript.swift b/Core/LoginFormDetectionUserScript.swift index 67c43f2d43..81a0fbbec0 100644 --- a/Core/LoginFormDetectionUserScript.swift +++ b/Core/LoginFormDetectionUserScript.swift @@ -21,9 +21,9 @@ import WebKit import UserScript public protocol LoginFormDetectionDelegate: NSObjectProtocol { - + func loginFormDetectionUserScriptDetectedLoginForm(_ script: LoginFormDetectionUserScript) - + } public class LoginFormDetectionUserScript: NSObject, UserScript { @@ -33,15 +33,15 @@ public class LoginFormDetectionUserScript: NSObject, UserScript { "$IS_DEBUG$": isDebugBuild ? "true" : "false" ]) }() - + public var injectionTime: WKUserScriptInjectionTime = .atDocumentStart - + public var forMainFrameOnly: Bool = false - + public var messageNames: [String] = [ "loginFormDetected" ] - + public weak var delegate: LoginFormDetectionDelegate? - + public func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) { delegate?.loginFormDetectionUserScriptDetectedLoginForm(self) } diff --git a/Core/NSAttributedStringExtension.swift b/Core/NSAttributedStringExtension.swift index 5454e14ac0..f0a3a61be7 100644 --- a/Core/NSAttributedStringExtension.swift +++ b/Core/NSAttributedStringExtension.swift @@ -27,15 +27,15 @@ extension NSAttributedString { mutableText.mutableString.setString(text) return mutableText } - + public var font: UIFont? { return attributes(at: 0, effectiveRange: nil)[.font] as? UIFont } - + public func stringWithFontSize(_ size: CGFloat) -> NSAttributedString? { guard let font = font else { return nil } let newFont = font.withSize(size) - + let newString = NSMutableAttributedString(attributedString: self) newString.setAttributes([.font: newFont], range: string.fullRange) return newString diff --git a/Core/NSManagedObjectContextExtension.swift b/Core/NSManagedObjectContextExtension.swift index d23551da22..27b6236d77 100644 --- a/Core/NSManagedObjectContextExtension.swift +++ b/Core/NSManagedObjectContextExtension.swift @@ -21,7 +21,7 @@ import CoreData import Persistence extension Array where Element == CoreDataErrorsParser.ErrorInfo { - + var errorPixelParameters: [String: String] { let params: [String: String] if let first = first { @@ -38,18 +38,18 @@ extension Array where Element == CoreDataErrorsParser.ErrorInfo { } extension NSManagedObjectContext { - + public func save(onErrorFire event: Pixel.Event) throws { do { try save() } catch { let nsError = error as NSError let processedErrors = CoreDataErrorsParser.parse(error: nsError) - + Pixel.fire(pixel: event, error: error, withAdditionalParameters: processedErrors.errorPixelParameters) - + throw error } } diff --git a/Core/NavigatorSharePatchUserScript.swift b/Core/NavigatorSharePatchUserScript.swift index 6f600cfafb..80b875ce14 100644 --- a/Core/NavigatorSharePatchUserScript.swift +++ b/Core/NavigatorSharePatchUserScript.swift @@ -25,15 +25,15 @@ public class NavigatorSharePatchUserScript: NSObject, UserScript { public var source: String { return Self.loadJS("navigatorsharepatch", from: Bundle.core) } - + public var injectionTime: WKUserScriptInjectionTime = .atDocumentStart - + public var forMainFrameOnly: Bool = false - + public var messageNames: [String] = [] - + public func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) { - + } } diff --git a/Core/NotFoundCachingDownloader.swift b/Core/NotFoundCachingDownloader.swift index edbfd59402..c9ba6daa67 100644 --- a/Core/NotFoundCachingDownloader.swift +++ b/Core/NotFoundCachingDownloader.swift @@ -40,7 +40,7 @@ class NotFoundCachingDownloader: ImageDownloader { if shouldDownload(url) { return super.downloadImage(with: url, options: options, completionHandler: completionHandler) } - + completionHandler?(.failure(.requestError(reason: .emptyRequest))) return nil } @@ -49,7 +49,7 @@ class NotFoundCachingDownloader: ImageDownloader { guard let hashedKey = FaviconsHelper.defaultResource(forDomain: domain, sourcesProvider: sourcesProvider)?.cacheKey else { return } notFoundCache[hashedKey] = Date().timeIntervalSince1970 } - + func shouldDownload(_ url: URL, referenceDate: Date = Date()) -> Bool { guard let domain = url.host else { return false } return shouldDownload(forDomain: domain, referenceDate: referenceDate) diff --git a/Core/Pixel.swift b/Core/Pixel.swift index 33b3ca2214..0e30542c6b 100644 --- a/Core/Pixel.swift +++ b/Core/Pixel.swift @@ -27,22 +27,22 @@ public struct PixelParameters { public static let duration = "dur" static let test = "test" public static let appVersion = "appVersion" - + public static let autocompleteBookmarkCapable = "bc" public static let autocompleteIncludedLocalResults = "sb" - + public static let originatedFromMenu = "om" - + public static let applicationState = "as" public static let dataAvailability = "dp" - + static let errorCode = "e" static let errorDomain = "d" static let errorDescription = "de" static let errorCount = "c" static let underlyingErrorCode = "ue" static let underlyingErrorDomain = "ud" - + static let coreDataErrorCode = "coreDataCode" static let coreDataErrorDomain = "coreDataDomain" static let coreDataErrorEntity = "coreDataEntity" @@ -62,12 +62,12 @@ public struct PixelParameters { static let clearWebDataTimedOut = "cd" public static let tabPreviewCountDelta = "cd" - + public static let etag = "et" public static let emailCohort = "cohort" public static let emailLastUsed = "duck_address_last_used" - + // Cookie clearing public static let storeInitialCount = "store_initial_count" public static let storeProtectedCount = "store_protected_count" @@ -78,27 +78,27 @@ public struct PixelParameters { public static let storageAfterDeletionCount = "storage_after_deletion_count" public static let storeAfterDeletionDiffCount = "store_after_deletion_diff_count" public static let storageAfterDeletionDiffCount = "storage_after_deletion_diff_count" - + public static let tabsModelCount = "tabs_model_count" public static let tabControllerCacheCount = "tab_controller_cache_count" - + public static let count = "count" public static let textSizeInitial = "text_size_initial" public static let textSizeUpdated = "text_size_updated" - + public static let canAutoPreviewMIMEType = "can_auto_preview_mime_type" public static let mimeType = "mime_type" public static let fileSizeGreaterThan10MB = "file_size_greater_than_10mb" public static let downloadListCount = "download_list_count" - + public static let bookmarkCount = "bco" - + public static let isBackgrounded = "is_backgrounded" public static let isDataProtected = "is_data_protected" - + public static let isInternalUser = "is_internal_user" - + // Email manager public static let emailKeychainAccessType = "access_type" public static let emailKeychainError = "error" @@ -122,7 +122,7 @@ public struct PixelParameters { public static let function = "function" public static let line = "line" public static let reason = "reason" - + // Return user public static let returnUserErrorCode = "error_code" public static let returnUserOldATB = "old_atb" @@ -150,10 +150,10 @@ public class Pixel { case atb case appVersion } - + private init() { } - + public static func fire(pixel: Pixel.Event, forDeviceType deviceType: UIUserInterfaceIdiom? = UIDevice.current.userInterfaceIdiom, withAdditionalParameters params: [String: String] = [:], @@ -222,11 +222,11 @@ public class Pixel { onComplete(error) } } - + } extension Pixel { - + public static func fire(pixel: Pixel.Event, error: Error?, includedParameters: [QueryParameters] = [.appVersion], diff --git a/Core/PixelEvent.swift b/Core/PixelEvent.swift index 0814b1a731..796409c66b 100644 --- a/Core/PixelEvent.swift +++ b/Core/PixelEvent.swift @@ -24,9 +24,8 @@ import Configuration import DDGSync import NetworkProtection -// swiftlint:disable file_length extension Pixel { - + // swiftlint:disable:next type_body_length public enum Event { @@ -79,7 +78,7 @@ extension Pixel { case addressBarShare case addressBarSettings - + case shareSheetResultSuccess case shareSheetResultFail case shareSheetActivityCopy @@ -89,7 +88,7 @@ extension Pixel { case shareSheetActivityPrint case shareSheetActivityAddToReadingList case shareSheetActivityOther - + case tabBarBackPressed case tabBarForwardPressed case bookmarksButtonPressed @@ -106,7 +105,7 @@ extension Pixel { case feedbackPositive case feedbackNegativePrefix(category: String) - + case brokenSiteReport case daxDialogsSerp @@ -119,7 +118,7 @@ extension Pixel { case daxDialogsFireEducationShown case daxDialogsFireEducationConfirmed case daxDialogsFireEducationCancelled - + case defaultBrowserButtonPressedSettings case widgetsOnboardingCTAPressed @@ -166,14 +165,14 @@ extension Pixel { case downloadAttemptToOpenBLOBviaJS case jsAlertShown - + case featureFlaggingInternalUserAuthenticated case autofillLoginsSaveLoginModalDisplayed case autofillLoginsSaveLoginModalConfirmed case autofillLoginsSaveLoginModalDismissed case autofillLoginsSaveLoginModalExcludeSiteConfirmed - + case autofillLoginsSavePasswordModalDisplayed case autofillLoginsSavePasswordModalConfirmed case autofillLoginsSavePasswordModalDismissed @@ -185,7 +184,7 @@ extension Pixel { case autofillLoginsUpdateUsernameModalDisplayed case autofillLoginsUpdateUsernameModalConfirmed case autofillLoginsUpdateUsernameModalDismissed - + case autofillLoginsFillLoginInlineManualDisplayed case autofillLoginsFillLoginInlineManualConfirmed case autofillLoginsFillLoginInlineManualDismissed @@ -203,18 +202,18 @@ extension Pixel { case autofillLoginsFillLoginInlineDisablePromptShown case autofillLoginsFillLoginInlineDisablePromptAutofillKept case autofillLoginsFillLoginInlineDisablePromptAutofillDisabled - + case autofillSettingsOpened case autofillLoginsSettingsEnabled case autofillLoginsSettingsDisabled case autofillLoginsSettingsResetExcludedDisplayed case autofillLoginsSettingsResetExcludedConfirmed case autofillLoginsSettingsResetExcludedDismissed - + case autofillLoginsPasswordGenerationPromptDisplayed case autofillLoginsPasswordGenerationPromptConfirmed case autofillLoginsPasswordGenerationPromptDismissed - + case autofillJSPixelFired(_ pixel: AutofillUserScript.JSPixel) case secureVaultError @@ -238,13 +237,13 @@ extension Pixel { // MARK: AppTP case appTPBreakageReport - + case appTPFailedToCreateProxyServer case appTPFailedToSetTunnelNetworkSettings case appTPFailedToAccessPreferences case appTPFailedToAccessPreferencesDuringSetup case appTPFailedToStartTunnel - + case appTPVPNDisconnect case appTPVPNMemoryWarning case appTPVPNMemoryCritical @@ -260,9 +259,9 @@ extension Pixel { case appTPDBFeedbackTrackerFetchFailed case appTPDBTrackerStoreFailure case appTPCouldNotLoadDatabase - + // MARK: Network Protection - + case networkProtectionActiveUser case networkProtectionNewUser @@ -281,27 +280,27 @@ extension Pixel { case networkProtectionEnableAttemptConnecting case networkProtectionEnableAttemptSuccess case networkProtectionEnableAttemptFailure - + case networkProtectionTunnelFailureDetected case networkProtectionTunnelFailureRecovered - + case networkProtectionLatency(quality: NetworkProtectionLatencyMonitor.ConnectionQuality) case networkProtectionLatencyError - + case networkProtectionEnabledOnSearch - + case networkProtectionBreakageReport case networkProtectionRekeyAttempt case networkProtectionRekeyFailure case networkProtectionRekeyCompleted - + case networkProtectionTunnelConfigurationNoServerRegistrationInfo case networkProtectionTunnelConfigurationCouldNotSelectClosestServer case networkProtectionTunnelConfigurationCouldNotGetPeerPublicKey case networkProtectionTunnelConfigurationCouldNotGetPeerHostName case networkProtectionTunnelConfigurationCouldNotGetInterfaceAddressRange - + case networkProtectionClientFailedToFetchServerList case networkProtectionClientFailedToParseServerListResponse case networkProtectionClientFailedToEncodeRegisterKeyRequest @@ -314,38 +313,38 @@ extension Pixel { case networkProtectionClientFailedToRedeemInviteCode case networkProtectionClientFailedToParseRedeemResponse case networkProtectionClientInvalidAuthToken - + case networkProtectionServerListStoreFailedToEncodeServerList case networkProtectionServerListStoreFailedToDecodeServerList case networkProtectionServerListStoreFailedToWriteServerList case networkProtectionServerListStoreFailedToReadServerList - + case networkProtectionKeychainErrorFailedToCastKeychainValueToData case networkProtectionKeychainReadError case networkProtectionKeychainWriteError case networkProtectionKeychainUpdateError case networkProtectionKeychainDeleteError - + case networkProtectionWireguardErrorCannotLocateTunnelFileDescriptor case networkProtectionWireguardErrorInvalidState case networkProtectionWireguardErrorFailedDNSResolution case networkProtectionWireguardErrorCannotSetNetworkSettings case networkProtectionWireguardErrorCannotStartWireguardBackend - + case networkProtectionFailedToLoadFromPreferences case networkProtectionFailedToSaveToPreferences case networkProtectionActivationRequestFailed case networkProtectionFailedToStartTunnel - + case networkProtectionDisconnected - + case networkProtectionNoAuthTokenFoundError - + case networkProtectionMemoryWarning case networkProtectionMemoryCritical - + case networkProtectionUnhandledError - + case networkProtectionWaitlistUserActive case networkProtectionSettingsRowDisplayed case networkProtectionWaitlistIntroScreenDisplayed @@ -354,14 +353,14 @@ extension Pixel { case networkProtectionWaitlistNotificationShown case networkProtectionWaitlistNotificationLaunched case networkProtectionWaitlistRetriedInviteCodeRedemption - + case networkProtectionGeoswitchingOpened case networkProtectionGeoswitchingSetNearest case networkProtectionGeoswitchingSetCustom case networkProtectionGeoswitchingNoLocations - + // MARK: remote messaging pixels - + case remoteMessageShown case remoteMessageShownUnique case remoteMessageDismissed @@ -369,10 +368,10 @@ extension Pixel { case remoteMessagePrimaryActionClicked case remoteMessageSecondaryActionClicked case remoteMessageSheet - + // MARK: debug pixels case dbCrashDetected - + case dbMigrationError case dbRemovalError case dbDestroyError @@ -388,7 +387,7 @@ extension Pixel { case dbRemoteMessagingUpdateMessageStatusError case dbRemoteMessagingDeleteScheduledMessageError case dbLocalAuthenticationError - + case configurationFetchInfo case trackerDataParseFailed @@ -405,11 +404,11 @@ extension Pixel { case contentBlockingCompilationTime case ampBlockingRulesCompilationFailed - + case webKitDidTerminate case webKitTerminationDidReloadCurrentTab case webKitDidTerminateDuringWarmup - + case backgroundTaskSubmissionFailed case blankOverlayNotDismissed @@ -426,10 +425,10 @@ extension Pixel { case compilationResult(result: CompileRulesResult, waitTime: CompileRulesWaitTime, appState: AppState) case emailAutofillKeychainError - + case adAttributionGlobalAttributedRulesDoNotExist case adAttributionCompilationFailedForAttributedRulesList - + case adAttributionLogicUnexpectedStateOnInheritedAttribution case adAttributionLogicUnexpectedStateOnRulesCompiled case adAttributionLogicUnexpectedStateOnRulesCompilationFailed @@ -452,18 +451,18 @@ extension Pixel { case debugMissingTopFolderFixHasBookmarks case debugCantSaveBookmarkFix - + case debugCannotClearObservationsDatabase case debugWebsiteDataStoresNotClearedMultiple case debugWebsiteDataStoresNotClearedOne case debugCookieCleanupError - + case debugBookmarksMigratedMoreThanOnce - + // Return user measurement case debugReturnUserAddATB case debugReturnUserUpdateATB - + // Errors from Bookmarks Module case bookmarkFolderExpected case bookmarksListIndexNotMatchingBookmark @@ -483,7 +482,7 @@ extension Pixel { case bookmarksMigrationCouldNotPrepareDatabaseOnFailedMigration case bookmarksMigrationCouldNotRemoveOldStore case bookmarksMigrationCouldNotPrepareMultipleFavoriteFolders - + case syncSignupDirect case syncSignupConnect case syncLogin @@ -498,7 +497,7 @@ extension Pixel { case syncCredentialsCountLimitExceededDaily case syncBookmarksRequestSizeLimitExceededDaily case syncCredentialsRequestSizeLimitExceededDaily - + case syncSentUnauthenticatedRequest case syncMetadataCouldNotLoadDatabase case syncBookmarksFailed @@ -513,22 +512,23 @@ extension Pixel { case syncRemoveDeviceError case syncDeleteAccountError case syncLoginExistingAccountError + case syncWrongEnvironment case swipeTabsUsed case swipeTabsUsedDaily - + case bookmarksCleanupFailed case bookmarksCleanupAttemptedWhileSyncWasEnabled case favoritesCleanupFailed case bookmarksFaviconsFetcherStateStoreInitializationFailed case bookmarksFaviconsFetcherFailed - + case credentialsDatabaseCleanupFailed case credentialsCleanupAttemptedWhileSyncWasEnabled - + case invalidPayload(Configuration) - + case emailIncontextPromptDisplayed case emailIncontextPromptConfirmed case emailIncontextPromptDismissed @@ -537,9 +537,9 @@ extension Pixel { case emailIncontextModalDismissed case emailIncontextModalExitEarly case emailIncontextModalExitEarlyContinue - + case compilationFailed - + case appRatingPromptFetchError case userBehaviorReloadTwice @@ -549,6 +549,38 @@ extension Pixel { case userBehaviorReloadAndTogglePrivacyControls case userBehaviorFireButtonAndRestart case userBehaviorFireButtonAndTogglePrivacyControls + + // MARK: Privacy pro + case privacyProSubscriptionActive + case privacyProOfferScreenImpression + case privacyProPurchaseAttempt + case privacyProPurchaseFailure + case privacyProPurchaseFailureStoreError + case privacyProPurchaseFailureBackendError + case privacyProPurchaseFailureAccountNotCreated + case privacyProPurchaseSuccess + case privacyProRestorePurchaseOfferPageEntry + case privacyProRestorePurchaseEmailStart + case privacyProRestorePurchaseStoreStart + case privacyProRestorePurchaseEmailSuccess + case privacyProRestorePurchaseStoreSuccess + case privacyProRestorePurchaseStoreFailureNotFound + case privacyProRestorePurchaseStoreFailureOther + case privacyProRestoreAfterPurchaseAttempt + case privacyProSubscriptionActivated + case privacyProWelcomeAddDevice + case privacyProSettingsAddDevice + case privacyProAddDeviceEnterEmail + case privacyProWelcomeVPN + case privacyProWelcomePersonalInformationRemoval + case privacyProWelcomeIdentityRestoration + case privacyProSubscriptionSettings + case privacyProVPNSettings + case privacyProPersonalInformationRemovalSettings + case privacyProIdentityRestorationSettings + case privacyProSubscriptionManagementEmail + case privacyProSubscriptionManagementPlanBilling + case privacyProSubscriptionManagementRemoval } } @@ -562,14 +594,14 @@ extension Pixel.Event { case .appLaunch: return "ml" case .refreshPressed: return "m_r" case .pullToRefresh: return "m_pull-to-reload" - + case .forgetAllPressedBrowsing: return "mf_bp" case .forgetAllPressedTabSwitching: return "mf_tp" case .forgetAllExecuted: return "mf" case .forgetAllDataCleared: return "mf_dc" case .privacyDashboardOpened: return "mp" - + case .dashboardProtectionAllowlistAdd: return "mp_wla" case .dashboardProtectionAllowlistRemove: return "mp_wlr" @@ -582,7 +614,7 @@ extension Pixel.Event { case .settingsDoNotSellShown: return "ms_dns" case .settingsDoNotSellOn: return "ms_dns_on" case .settingsDoNotSellOff: return "ms_dns_off" - + case .settingsAutoconsentShown: return "m_settings_autoconsent_shown" case .settingsAutoconsentOn: return "m_settings_autoconsent_on" case .settingsAutoconsentOff: return "m_settings_autoconsent_off" @@ -604,9 +636,9 @@ extension Pixel.Event { case .browsingMenuReportBrokenSite: return "mb_rb" case .browsingMenuFireproof: return "mb_f" case .browsingMenuAutofill: return "m_nav_autofill_menu_item_pressed" - + case .browsingMenuShare: return "m_browsingmenu_share" - + case .addressBarShare: return "m_addressbar_share" case .addressBarSettings: return "m_addressbar_settings" case .shareSheetResultSuccess: return "m_sharesheet_result_success" @@ -618,7 +650,7 @@ extension Pixel.Event { case .shareSheetActivityPrint: return "m_sharesheet_activity_print" case .shareSheetActivityAddToReadingList: return "m_sharesheet_activity_addtoreadinglist" case .shareSheetActivityOther: return "m_sharesheet_activity_other" - + case .tabBarBackPressed: return "mt_bk" case .tabBarForwardPressed: return "mt_fw" case .bookmarksButtonPressed: return "mt_bm" @@ -632,10 +664,10 @@ extension Pixel.Event { case .autocompleteSelectedLocal: return "m_au_l" case .autocompleteSelectedRemote: return "m_au_r" - + case .feedbackPositive: return "mfbs_positive_submit" case .feedbackNegativePrefix(category: let category): return "mfbs_negative_\(category)" - + case .brokenSiteReport: return "epbf" case .daxDialogsSerp: return "m_dx_s" @@ -693,28 +725,28 @@ extension Pixel.Event { case .downloadsSharingPredownloadedLocalFile: return "m_downloads_sharing_predownloaded_local_file" case .downloadAttemptToOpenBLOBviaJS: return "m_download_attempt_to_open_blob_js" - + case .jsAlertShown: return "m_js_alert_shown" - + case .featureFlaggingInternalUserAuthenticated: return "m_internal-user_authenticated" - + case .autofillLoginsSaveLoginModalDisplayed: return "m_autofill_logins_save_login_inline_displayed" case .autofillLoginsSaveLoginModalConfirmed: return "m_autofill_logins_save_login_inline_confirmed" case .autofillLoginsSaveLoginModalDismissed: return "m_autofill_logins_save_login_inline_dismissed" case .autofillLoginsSaveLoginModalExcludeSiteConfirmed: return "m_autofill_logins_save_login_exclude_site_confirmed" - + case .autofillLoginsSavePasswordModalDisplayed: return "m_autofill_logins_save_password_inline_displayed" case .autofillLoginsSavePasswordModalConfirmed: return "m_autofill_logins_save_password_inline_confirmed" case .autofillLoginsSavePasswordModalDismissed: return "m_autofill_logins_save_password_inline_dismissed" - + case .autofillLoginsUpdatePasswordModalDisplayed: return "m_autofill_logins_update_password_inline_displayed" case .autofillLoginsUpdatePasswordModalConfirmed: return "m_autofill_logins_update_password_inline_confirmed" case .autofillLoginsUpdatePasswordModalDismissed: return "m_autofill_logins_update_password_inline_dismissed" - + case .autofillLoginsUpdateUsernameModalDisplayed: return "m_autofill_logins_update_username_inline_displayed" case .autofillLoginsUpdateUsernameModalConfirmed: return "m_autofill_logins_update_username_inline_confirmed" case .autofillLoginsUpdateUsernameModalDismissed: return "m_autofill_logins_update_username_inline_dismissed" - + case .autofillLoginsFillLoginInlineManualDisplayed: return "m_autofill_logins_fill_login_inline_manual_displayed" case .autofillLoginsFillLoginInlineManualConfirmed: return "m_autofill_logins_fill_login_inline_manual_confirmed" case .autofillLoginsFillLoginInlineManualDismissed: return "m_autofill_logins_fill_login_inline_manual_dismissed" @@ -745,11 +777,11 @@ extension Pixel.Event { case .autofillLoginsSettingsResetExcludedDisplayed: return "m_autofill_settings_reset_excluded_displayed" case .autofillLoginsSettingsResetExcludedConfirmed: return "m_autofill_settings_reset_excluded_confirmed" case .autofillLoginsSettingsResetExcludedDismissed: return "m_autofill_settings_reset_excluded_dismissed" - + case .autofillLoginsPasswordGenerationPromptDisplayed: return "m_autofill_logins_password_generation_prompt_displayed" case .autofillLoginsPasswordGenerationPromptConfirmed: return "m_autofill_logins_password_generation_prompt_confirmed" case .autofillLoginsPasswordGenerationPromptDismissed: return "m_autofill_logins_password_generation_prompt_dismissed" - + case .autofillJSPixelFired(let pixel): return "m_ios_\(pixel.pixelName)" @@ -760,19 +792,19 @@ extension Pixel.Event { case .secureVaultIsEnabledCheckedWhenEnabledAndDataProtected: return "m_secure-vault_is-enabled-checked_when-enabled-and-data-protected" - // MARK: Ad Click Attribution pixels + // MARK: Ad Click Attribution pixels case .adClickAttributionDetected: return "m_ad_click_detected" case .adClickAttributionActive: return "m_ad_click_active" case .adClickAttributionPageLoads: return "m_pageloads_with_ad_attribution" - // MARK: SERP pixels + // MARK: SERP pixels case .serpRequerySame: return "rq_0" case .serpRequeryNew: return "rq_1" - // MARK: AppTP pixels - + // MARK: AppTP pixels + case .appTPBreakageReport: return "m_apptp_breakage_report" case .appTPFailedToCreateProxyServer: return "m_apptp_failed_to_create_proxy_server" case .appTPFailedToSetTunnelNetworkSettings: return "m_apptp_failed_to_set_tunnel_network_settings" @@ -782,7 +814,7 @@ extension Pixel.Event { case .appTPVPNDisconnect: return "m_apptp_vpn_disconnect" case .appTPVPNMemoryWarning: return "m_apptp_vpn_memory_warning" case .appTPVPNMemoryCritical: return "m_apptp_vpn_memory_critical" - + case .appTPBlocklistParseFailed: return "m_apptp_blocklist_parse_failed" case .appTPActiveUser: return "m_apptp_active_user" case .appTPDBLocationFailed: return "m_apptp_db_location_not_found" @@ -793,9 +825,9 @@ extension Pixel.Event { case .appTPDBFeedbackTrackerFetchFailed: return "m_apptp_db_feedback_tracker_fetch_failed" case .appTPDBTrackerStoreFailure: return "m_apptp_db_tracker_store_failure" case .appTPCouldNotLoadDatabase: return "m_apptp_could_not_load_database" - - // MARK: Network Protection pixels - + + // MARK: Network Protection pixels + case .networkProtectionActiveUser: return "m_netp_daily_active_d" case .networkProtectionNewUser: return "m_netp_daily_active_u" case .networkProtectionControllerStartAttempt: return "m_netp_controller_start_attempt" @@ -862,7 +894,7 @@ extension Pixel.Event { case .networkProtectionMemoryWarning: return "m_netp_vpn_memory_warning" case .networkProtectionMemoryCritical: return "m_netp_vpn_memory_critical" case .networkProtectionUnhandledError: return "m_netp_unhandled_error" - + case .networkProtectionWaitlistUserActive: return "m_netp_waitlist_user_active" case .networkProtectionSettingsRowDisplayed: return "m_netp_waitlist_settings_entry_viewed" case .networkProtectionWaitlistIntroScreenDisplayed: return "m_netp_waitlist_intro_screen_viewed" @@ -871,14 +903,14 @@ extension Pixel.Event { case .networkProtectionWaitlistNotificationShown: return "m_netp_waitlist_notification_shown" case .networkProtectionWaitlistNotificationLaunched: return "m_netp_waitlist_notification_launched" case .networkProtectionWaitlistRetriedInviteCodeRedemption: return "m_netp_waitlist_retried_invite_code_redemption" - + case .networkProtectionGeoswitchingOpened: return "m_netp_imp_geoswitching" case .networkProtectionGeoswitchingSetNearest: return "m_netp_ev_geoswitching_set_nearest" case .networkProtectionGeoswitchingSetCustom: return "m_netp_ev_geoswitching_set_custom" case .networkProtectionGeoswitchingNoLocations: return "m_netp_ev_geoswitching_no_locations" - - // MARK: remote messaging pixels - + + // MARK: remote messaging pixels + case .remoteMessageShown: return "m_remote_message_shown" case .remoteMessageShownUnique: return "m_remote_message_shown_unique" case .remoteMessageDismissed: return "m_remote_message_dismissed" @@ -886,9 +918,9 @@ extension Pixel.Event { case .remoteMessagePrimaryActionClicked: return "m_remote_message_primary_action_clicked" case .remoteMessageSecondaryActionClicked: return "m_remote_message_secondary_action_clicked" case .remoteMessageSheet: return "m_remote_message_sheet" - - // MARK: debug pixels - + + // MARK: debug pixels + case .dbCrashDetected: return "m_d_crash" case .dbMigrationError: return "m_d_dbme" case .dbRemovalError: return "m_d_dbre" @@ -905,9 +937,9 @@ extension Pixel.Event { case .dbRemoteMessagingUpdateMessageStatusError: return "m_d_db_rm_update_message_status" case .dbRemoteMessagingDeleteScheduledMessageError: return "m_d_db_rm_delete_scheduled_message" case .dbLocalAuthenticationError: return "m_d_local_auth_error" - + case .debugBookmarksMigratedMoreThanOnce: return "m_debug_bookmarks_migrated-more-than-once" - + case .configurationFetchInfo: return "m_d_cfgfetch" case .trackerDataParseFailed: return "m_d_tds_p" @@ -924,7 +956,7 @@ extension Pixel.Event { case .contentBlockingCompilationTime: return "m_content_blocking_compilation_time" case .ampBlockingRulesCompilationFailed: return "m_debug_amp_rules_compilation_failed" - + case .webKitDidTerminate: return "m_d_wkt" case .webKitDidTerminateDuringWarmup: return "m_d_webkit-terminated-during-warmup" case .webKitTerminationDidReloadCurrentTab: return "m_d_wktct" @@ -946,25 +978,25 @@ extension Pixel.Event { return "m_compilation_result_\(result)_time_\(waitTime)_state_\(appState)" case .emailAutofillKeychainError: return "m_email_autofill_keychain_error" - + case .debugBookmarkOrphanFolderNew: return "m_d_bookmark_orphan_folder_new" case .debugBookmarkTopLevelMissingNew: return "m_d_bookmark_top_level_missing_new" case .debugCouldNotFixBookmarkFolder: return "m_d_cannot_fix_bookmark_folder" case .debugMissingTopFolderFixHasBookmarks: return "m_d_missing_top_folder_has_bookmarks" - + case .debugFavoriteOrphanFolderNew: return "m_d_favorite_orphan_folder_new" case .debugFavoriteTopLevelMissingNew: return "m_d_favorite_top_level_missing_new" case .debugCouldNotFixFavoriteFolder: return "m_d_cannot_fix_favorite_folder" case .debugMissingTopFolderFixHasFavorites: return "m_d_missing_top_folder_has_favorites" case .debugCantSaveBookmarkFix: return "m_d_cant_save_bookmark_fix" - + case .debugCannotClearObservationsDatabase: return "m_d_cannot_clear_observations_database" case .debugWebsiteDataStoresNotClearedMultiple: return "m_d_wkwebsitedatastoresnotcleared_multiple" case .debugWebsiteDataStoresNotClearedOne: return "m_d_wkwebsitedatastoresnotcleared_one" case .debugCookieCleanupError: return "m_d_cookie-cleanup-error" - - // MARK: Ad Attribution + + // MARK: Ad Attribution case .adAttributionGlobalAttributedRulesDoNotExist: return "m_attribution_global_attributed_rules_do_not_exist" case .adAttributionCompilationFailedForAttributedRulesList: return "m_attribution_compilation_failed_for_attributed_rules_list" @@ -997,7 +1029,7 @@ extension Pixel.Event { return "m_d_bookmarks_migration_could_not_prepare_database_on_failed_migration" case .bookmarksMigrationCouldNotRemoveOldStore: return "m_d_bookmarks_migration_could_not_remove_old_store" case .bookmarksMigrationCouldNotPrepareMultipleFavoriteFolders: return "m_d_bookmarks_migration_could_not_prepare_multiple_favorite_folders" - + case .syncSignupDirect: return "m_sync_signup_direct" case .syncSignupConnect: return "m_sync_signup_connect" case .syncLogin: return "m_sync_login" @@ -1012,7 +1044,7 @@ extension Pixel.Event { case .syncCredentialsCountLimitExceededDaily: return "m_d_sync_credentials_count_limit_exceeded_daily" case .syncBookmarksRequestSizeLimitExceededDaily: return "m_d_sync_bookmarks_request_size_limit_exceeded_daily" case .syncCredentialsRequestSizeLimitExceededDaily: return "m_d_sync_credentials_request_size_limit_exceeded_daily" - + case .syncSentUnauthenticatedRequest: return "m_d_sync_sent_unauthenticated_request" case .syncMetadataCouldNotLoadDatabase: return "m_d_sync_metadata_could_not_load_database" case .syncBookmarksFailed: return "m_d_sync_bookmarks_failed" @@ -1032,19 +1064,19 @@ extension Pixel.Event { case .swipeTabsUsed: return "m_swipe-tabs-used" case .swipeTabsUsedDaily: return "m_swipe-tabs-used-daily" - + case .bookmarksCleanupFailed: return "m_d_bookmarks_cleanup_failed" case .bookmarksCleanupAttemptedWhileSyncWasEnabled: return "m_d_bookmarks_cleanup_attempted_while_sync_was_enabled" case .favoritesCleanupFailed: return "m_d_favorites_cleanup_failed" case .bookmarksFaviconsFetcherStateStoreInitializationFailed: return "m_d_bookmarks_favicons_fetcher_state_store_initialization_failed" case .bookmarksFaviconsFetcherFailed: return "m_d_bookmarks_favicons_fetcher_failed" - + case .credentialsDatabaseCleanupFailed: return "m_d_credentials_database_cleanup_failed_2" case .credentialsCleanupAttemptedWhileSyncWasEnabled: return "m_d_credentials_cleanup_attempted_while_sync_was_enabled" - + case .invalidPayload(let configuration): return "m_d_\(configuration.rawValue)_invalid_payload".lowercased() - - // MARK: - InContext Email Protection + + // MARK: - InContext Email Protection case .emailIncontextPromptDisplayed: return "m_email_incontext_prompt_displayed" case .emailIncontextPromptConfirmed: return "m_email_incontext_prompt_confirmed" case .emailIncontextPromptDismissed: return "m_email_incontext_prompt_dismissed" @@ -1053,15 +1085,15 @@ extension Pixel.Event { case .emailIncontextModalDismissed: return "m_email_incontext_modal_dismissed" case .emailIncontextModalExitEarly: return "m_email_incontext_modal_exit_early" case .emailIncontextModalExitEarlyContinue: return "m_email_incontext_modal_exit_early_continue" - + case .compilationFailed: return "m_d_compilation_failed" - // MARK: - Return user measurement + // MARK: - Return user measurement case .debugReturnUserAddATB: return "m_debug_return_user_add_atb" case .debugReturnUserUpdateATB: return "m_debug_return_user_update_atb" - + case .appRatingPromptFetchError: return "m_d_app_rating_prompt_fetch_error" - // MARK: - User behavior + // MARK: - User behavior case .userBehaviorReloadTwice: return "m_reload-twice" case .userBehaviorReloadAndRestart: return "m_reload-and-restart" case .userBehaviorReloadAndFireButton: return "m_reload-and-fire-button" @@ -1069,12 +1101,43 @@ extension Pixel.Event { case .userBehaviorReloadAndTogglePrivacyControls: return "m_reload-and-toggle-privacy-controls" case .userBehaviorFireButtonAndRestart: return "m_fire-button-and-restart" case .userBehaviorFireButtonAndTogglePrivacyControls: return "m_fire-button-and-toggle-privacy-controls" + + // MARK: Privacy pro + case .privacyProSubscriptionActive: return "m_privacy-pro_app_subscription_active" + case .privacyProOfferScreenImpression: return "m_privacy-pro_offer_screen_impression" + case .privacyProPurchaseAttempt: return "m_privacy-pro_terms-conditions_subscribe_click" + case .privacyProPurchaseFailure: return "m_privacy-pro_app_subscription-purchase_failure_other" + case .privacyProPurchaseFailureStoreError: return "m_privacy-pro_app_subscription-purchase_failure_store" + case .privacyProPurchaseFailureAccountNotCreated: return "m_privacy-pro_app_subscription-purchase_failure_backend" + case .privacyProPurchaseFailureBackendError: return "m_privacy-pro_app_subscription-purchase_failure_account-creation" + case .privacyProPurchaseSuccess: return "m_privacy-pro_app_subscription-purchase_success" + case .privacyProRestorePurchaseOfferPageEntry: return "m_privacy-pro_offer_restore-purchase_click" + case .privacyProRestorePurchaseEmailStart: return "m_privacy-pro_activate-subscription_enter-email_click" + case .privacyProRestorePurchaseStoreStart: return "m_privacy-pro_activate-subscription_restore-purchase_click" + case .privacyProRestorePurchaseEmailSuccess: return "m_privacy-pro_app_subscription-restore-using-email_success" + case .privacyProRestorePurchaseStoreSuccess: return "m_privacy-pro_app_subscription-restore-using-store_success" + case .privacyProRestorePurchaseStoreFailureNotFound: return "m_privacy-pro_app_subscription-restore-using-store_failure_not-found" + case .privacyProRestorePurchaseStoreFailureOther: return "m_privacy-pro_app_subscription-restore-using-store_failure_other" + case .privacyProRestoreAfterPurchaseAttempt: return "m_privacy-pro_app_subscription-restore-after-purchase-attempt_success" + case .privacyProSubscriptionActivated: return "m_privacy-pro_app_subscription_activated_u" + case .privacyProWelcomeAddDevice: return "m_privacy-pro_welcome_add-device_click_u" + case .privacyProSettingsAddDevice: return "m_privacy-pro_settings_add-device_click" + case .privacyProAddDeviceEnterEmail: return "m_privacy-pro_add-device_enter-email_click" + case .privacyProWelcomeVPN: return "m_privacy-pro_welcome_vpn_click_u" + case .privacyProWelcomePersonalInformationRemoval: return "m_privacy-pro_welcome_personal-information-removal_click_u" + case .privacyProWelcomeIdentityRestoration: return "m_privacy-pro_welcome_identity-theft-restoration_click_u" + case .privacyProSubscriptionSettings: return "m_privacy-pro_settings_screen_impression" + case .privacyProVPNSettings: return "m_privacy-pro_app-settings_vpn_click" + case .privacyProPersonalInformationRemovalSettings: return "m_privacy-pro_app-settings_personal-information-removal_click" + case .privacyProIdentityRestorationSettings: return "m_privacy-pro_app-settings_identity-theft-restoration_click" + case .privacyProSubscriptionManagementEmail: return "m_privacy-pro_manage-email_edit_click" + case .privacyProSubscriptionManagementPlanBilling: return "m_privacy-pro_settings_change-plan-or-billing_click" + case .privacyProSubscriptionManagementRemoval: return "m_privacy-pro_settings_remove-from-device_click" } - } - } +// swiftlint:disable file_length extension Pixel.Event { public enum CompileRulesWaitTime: String, CustomStringConvertible { @@ -1128,9 +1191,9 @@ extension Pixel.Event { case regular } - - public enum CompileRulesListType: String, CustomStringConvertible { + public enum CompileRulesListType: String, CustomStringConvertible { + public var description: String { rawValue } case tds diff --git a/Core/PreserveLogins.swift b/Core/PreserveLogins.swift index 2c747a4abd..6ce320e0ef 100644 --- a/Core/PreserveLogins.swift +++ b/Core/PreserveLogins.swift @@ -20,13 +20,13 @@ import Foundation public class PreserveLogins { - + public struct Notifications { public static let loginDetectionStateChanged = Foundation.Notification.Name("com.duckduckgo.ios.PreserveLogins.loginDetectionStateChanged") } - + public static let shared = PreserveLogins() - + @UserDefaultsWrapper(key: .preserveLoginsAllowedDomains, defaultValue: []) private(set) public var allowedDomains: [String] @@ -36,7 +36,7 @@ public class PreserveLogins { NotificationCenter.default.post(name: Notifications.loginDetectionStateChanged, object: nil) } } - + public func addToAllowed(domain: String) { allowedDomains += [domain] } @@ -55,7 +55,7 @@ public class PreserveLogins { public func clearAll() { allowedDomains = [] } - + public func isAllowed(fireproofDomain domain: String) -> Bool { return allowedDomains.contains(domain) } diff --git a/Core/PrivacyFeatures.swift b/Core/PrivacyFeatures.swift index 977a5d94f9..422484ce5b 100644 --- a/Core/PrivacyFeatures.swift +++ b/Core/PrivacyFeatures.swift @@ -58,5 +58,5 @@ public final class PrivacyFeatures { } public static let httpsUpgrade = HTTPSUpgrade(store: httpsUpgradeStore, privacyManager: ContentBlocking.shared.privacyConfigurationManager) - + } diff --git a/Core/ReturnUserMeasurement.swift b/Core/ReturnUserMeasurement.swift index f4a9507a08..ba33d836ab 100644 --- a/Core/ReturnUserMeasurement.swift +++ b/Core/ReturnUserMeasurement.swift @@ -105,7 +105,7 @@ class KeychainReturnUserMeasurement: ReturnUserMeasurement { kSecClass as String: secClassCFString, kSecMatchLimit as String: kSecMatchLimitOne, kSecReturnAttributes as String: true, // Needs to be true or returns nothing. - kSecReturnRef as String: true, + kSecReturnRef as String: true ] var returnArrayRef: CFTypeRef? let status = SecItemCopyMatching(query as CFDictionary, &returnArrayRef) diff --git a/Core/SchemeHandler.swift b/Core/SchemeHandler.swift index 75acbae352..605e5c58df 100644 --- a/Core/SchemeHandler.swift +++ b/Core/SchemeHandler.swift @@ -21,13 +21,13 @@ import Foundation import BrowserServicesKit public class SchemeHandler { - + public enum Action: Equatable { case open case askForConfirmation case cancel } - + public enum SchemeType: Equatable { case navigational case external(Action) @@ -51,14 +51,14 @@ public class SchemeHandler { case shortcutsProduction = "shortcuts-production" case workflow } - + private enum BlockedScheme: String { case appleDataDetectors = "x-apple-data-detectors" } - + public static func schemeType(for url: URL) -> SchemeType { guard let schemeString = url.scheme else { return .unknown } - + guard BlockedScheme(rawValue: schemeString) == nil else { return .external(.cancel) } diff --git a/Core/StatisticsLoader.swift b/Core/StatisticsLoader.swift index 8b7a6e481c..3572d40dc0 100644 --- a/Core/StatisticsLoader.swift +++ b/Core/StatisticsLoader.swift @@ -23,21 +23,21 @@ import BrowserServicesKit import Networking public class StatisticsLoader { - + public typealias Completion = (() -> Void) - + public static let shared = StatisticsLoader() - + private let statisticsStore: StatisticsStore private let returnUserMeasurement: ReturnUserMeasurement private let parser = AtbParser() - + init(statisticsStore: StatisticsStore = StatisticsUserDefaults(), returnUserMeasurement: ReturnUserMeasurement = KeychainReturnUserMeasurement()) { self.statisticsStore = statisticsStore self.returnUserMeasurement = returnUserMeasurement } - + public func load(completion: @escaping Completion = {}) { if statisticsStore.hasInstallStatistics { completion() @@ -45,18 +45,18 @@ public class StatisticsLoader { } requestInstallStatistics(completion: completion) } - + private func requestInstallStatistics(completion: @escaping Completion = {}) { let configuration = APIRequest.Configuration(url: .atb) let request = APIRequest(configuration: configuration, urlSession: .session()) - + request.fetch { response, error in if let error = error { os_log("Initial atb request failed with error %s", log: .generalLog, type: .debug, error.localizedDescription) completion() return } - + if let data = response?.data, let atb = try? self.parser.convert(fromJsonData: data) { self.requestExti(atb: atb, completion: completion) } else { @@ -64,14 +64,14 @@ public class StatisticsLoader { } } } - + private func requestExti(atb: Atb, completion: @escaping Completion = {}) { let installAtb = atb.version + (statisticsStore.variant ?? "") let url = URL.makeExtiURL(atb: installAtb) - + let configuration = APIRequest.Configuration(url: url) let request = APIRequest(configuration: configuration, urlSession: .session()) - + request.fetch { _, error in if let error = error { os_log("Exti request failed with error %s", log: .generalLog, type: .debug, error.localizedDescription) @@ -84,7 +84,7 @@ public class StatisticsLoader { completion() } } - + public func refreshSearchRetentionAtb(completion: @escaping Completion = {}) { guard let url = StatisticsDependentURLFactory(statisticsStore: statisticsStore).makeSearchAtbURL() else { requestInstallStatistics(completion: completion) @@ -93,7 +93,7 @@ public class StatisticsLoader { let configuration = APIRequest.Configuration(url: url) let request = APIRequest(configuration: configuration, urlSession: .session()) - + request.fetch { response, error in if let error = error { os_log("Search atb request failed with error %s", log: .generalLog, type: .debug, error.localizedDescription) @@ -107,7 +107,7 @@ public class StatisticsLoader { completion() } } - + public func refreshAppRetentionAtb(completion: @escaping Completion = {}) { guard let url = StatisticsDependentURLFactory(statisticsStore: statisticsStore).makeAppAtbURL() else { requestInstallStatistics(completion: completion) @@ -116,7 +116,7 @@ public class StatisticsLoader { let configuration = APIRequest.Configuration(url: url) let request = APIRequest(configuration: configuration, urlSession: .session()) - + request.fetch { response, error in if let error = error { os_log("App atb request failed with error %s", log: .generalLog, type: .debug, error.localizedDescription) diff --git a/Core/StatisticsUserDefaults.swift b/Core/StatisticsUserDefaults.swift index 290db449d0..0ff51cc809 100644 --- a/Core/StatisticsUserDefaults.swift +++ b/Core/StatisticsUserDefaults.swift @@ -56,7 +56,7 @@ public class StatisticsUserDefaults: StatisticsStore { userDefaults?.setValue(newValue, forKey: Keys.atb) } } - + public var installDate: Date? { get { guard let interval = userDefaults?.double(forKey: Keys.installDate), interval > 0 else { @@ -77,7 +77,7 @@ public class StatisticsUserDefaults: StatisticsStore { userDefaults?.setValue(newValue, forKey: Keys.searchRetentionAtb) } } - + public var appRetentionAtb: String? { get { return userDefaults?.string(forKey: Keys.appRetentionAtb) ?? atb diff --git a/Core/StorageCache.swift b/Core/StorageCache.swift index 27ea656435..1336fdfef0 100644 --- a/Core/StorageCache.swift +++ b/Core/StorageCache.swift @@ -22,15 +22,15 @@ import BrowserServicesKit import Common public class StorageCache { - + public let tld: TLD - + public init() { tld = TLD() } - + public init(tld: TLD) { self.tld = tld } - + } diff --git a/Core/StringExtension.swift b/Core/StringExtension.swift index f0ad33ba90..b79146a7ea 100644 --- a/Core/StringExtension.swift +++ b/Core/StringExtension.swift @@ -26,7 +26,7 @@ extension String { public func format(arguments: CVarArg...) -> String { return String(format: self, arguments: arguments) } - + public func sha256() -> String { if let stringData = self.data(using: String.Encoding.utf8) { return stringData.sha256 @@ -40,17 +40,17 @@ extension String { verticalOffset: CGFloat = 0.0) -> NSAttributedString? { let components = self.components(separatedBy: placeholder) guard components.count > 1 else { return nil } - + let attachment = NSTextAttachment() attachment.image = image attachment.bounds = CGRect(x: 0, y: verticalOffset, width: image.size.width, height: image.size.height) let attachmentString = NSAttributedString(attachment: attachment) - + let paddingAttachment = NSTextAttachment() paddingAttachment.bounds = CGRect(x: 0, y: 0, width: horizontalPadding, height: 0) let startPadding = NSAttributedString(attachment: paddingAttachment) let endPadding = NSAttributedString(attachment: paddingAttachment) - + let firstString = NSMutableAttributedString(string: components[0]) for component in components.dropFirst() { let endString = NSMutableAttributedString(string: component) diff --git a/Core/TabInstrumentation.swift b/Core/TabInstrumentation.swift index 86c40b5c3e..039623cb46 100644 --- a/Core/TabInstrumentation.swift +++ b/Core/TabInstrumentation.swift @@ -21,31 +21,31 @@ import Foundation import os.signpost public class TabInstrumentation { - + static let tabsLog = OSLog(subsystem: "com.duckduckgo.instrumentation", category: "TabInstrumentation") - + static var tabMaxIdentifier: UInt64 = 0 - + private var siteLoadingSPID: Any? private var currentURL: String? private var currentTabIdentifier: UInt64 - + public init() { type(of: self).tabMaxIdentifier += 1 currentTabIdentifier = type(of: self).tabMaxIdentifier } - + private var tabInitSPID: Any? - + public func willPrepareWebView() { tabInitSPID = Instruments.shared.startTimedEvent(.tabInitialisation, info: "Tab-\(currentTabIdentifier)") } - + public func didPrepareWebView() { Instruments.shared.endTimedEvent(for: tabInitSPID) } - + public func willLoad(url: URL) { currentURL = url.absoluteString if #available(iOSApplicationExtension 12.0, *) { @@ -58,7 +58,7 @@ public class TabInstrumentation { "Loading URL: %@ in %llu", url.absoluteString, currentTabIdentifier) } } - + public func didLoadURL() { if #available(iOSApplicationExtension 12.0, *), let id = siteLoadingSPID as? OSSignpostID { @@ -69,42 +69,42 @@ public class TabInstrumentation { "Loading Finished: %{private}@", "T") } } - + // MARK: - JS events - + public func request(url: String, allowedIn timeInMs: Double) { request(url: url, isTracker: false, blocked: false, in: timeInMs) } - + public func tracker(url: String, allowedIn timeInMs: Double, reason: String?) { request(url: url, isTracker: true, blocked: false, reason: reason ?? "?", in: timeInMs) } - + public func tracker(url: String, blockedIn timeInMs: Double) { request(url: url, isTracker: true, blocked: true, in: timeInMs) } - + private func request(url: String, isTracker: Bool, blocked: Bool, reason: String = "", in timeInMs: Double) { if #available(iOSApplicationExtension 12.0, *) { let currentURL = self.currentURL ?? "unknown" let requestType = isTracker ? "Tracker" : "Regular" let status = blocked ? "Blocked" : "Allowed" - + // 0 is treated as 1ms let timeInNS: UInt64 = timeInMs.asNanos - + os_log(.debug, log: type(of: self).tabsLog, "[%@] Request: %@ - %@ - %@ (%@) in %llu", currentURL, url, requestType, status, reason, timeInNS) } } - + public func jsEvent(name: String, executedIn timeInMs: Double) { if #available(iOSApplicationExtension 12.0, *) { let currentURL = self.currentURL ?? "unknown" // 0 is treated as 1ms let timeInNS: UInt64 = timeInMs.asNanos - + os_log(.debug, log: type(of: self).tabsLog, "[%@] JSEvent: %@ executedIn: %llu", currentURL, name, timeInNS) diff --git a/Core/TextFieldWithInsets.swift b/Core/TextFieldWithInsets.swift index 11a378026e..c367c82f9e 100644 --- a/Core/TextFieldWithInsets.swift +++ b/Core/TextFieldWithInsets.swift @@ -21,7 +21,7 @@ import UIKit @IBDesignable class TextFieldWithInsets: UITextField { - + var onCopyAction: ((UITextField) -> Void)? @IBInspectable var leftInset: CGFloat = 0 @@ -52,7 +52,7 @@ class TextFieldWithInsets: UITextField { let height = bounds.size.height - topInset - bottomInset return CGRect(x: x, y: y, width: width, height: height) } - + override func copy(_ sender: Any?) { if let action = onCopyAction { action(self) diff --git a/Core/TextSizeUserScript.swift b/Core/TextSizeUserScript.swift index a24d9aa6d1..a246d48e63 100644 --- a/Core/TextSizeUserScript.swift +++ b/Core/TextSizeUserScript.swift @@ -22,25 +22,25 @@ import WebKit import UserScript public class TextSizeUserScript: NSObject, UserScript { - + public static let knownDynamicTypeExceptions: [String] = ["wikipedia.org"] public var textSizeAdjustmentInPercents: Int = 100 - + public var source: String { TextSizeUserScript.makeSource(for: textSizeAdjustmentInPercents) } public var injectionTime: WKUserScriptInjectionTime = .atDocumentStart public var forMainFrameOnly: Bool = false public var messageNames: [String] = [] - + public init(textSizeAdjustmentInPercents: Int) { self.textSizeAdjustmentInPercents = textSizeAdjustmentInPercents } public func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) { } - + fileprivate static func makeSource(for textSizeAdjustmentInPercents: Int) -> String { let dynamicTypeScalePercentage = UIFontMetrics.default.scaledValue(for: 100) - + return loadJS("textsize", from: Bundle.core, withReplacements: [ "$KNOWN_DYNAMIC_TYPE_EXCEPTIONS$": knownDynamicTypeExceptions.joined(separator: "\n"), "$TEXT_SIZE_ADJUSTMENT_IN_PERCENTS$": "\(textSizeAdjustmentInPercents)", @@ -50,7 +50,7 @@ public class TextSizeUserScript: NSObject, UserScript { } public extension WKWebView { - + func adjustTextSize(_ percentage: Int) { let jsString = TextSizeUserScript.makeSource(for: percentage) evaluateJavaScript(jsString, completionHandler: nil) diff --git a/Core/TimeIntervalExtension.swift b/Core/TimeIntervalExtension.swift index 825bd80577..00f8492ee4 100644 --- a/Core/TimeIntervalExtension.swift +++ b/Core/TimeIntervalExtension.swift @@ -20,7 +20,7 @@ import Foundation extension TimeInterval { - + // MARK: - Computed Type Properties internal static var secondsPerDay: Double { return 24 * 60 * 60 } internal static var secondsPerHour: Double { return 60 * 60 } diff --git a/Core/TimedPixel.swift b/Core/TimedPixel.swift index e51591f80b..8f9afb9d45 100644 --- a/Core/TimedPixel.swift +++ b/Core/TimedPixel.swift @@ -20,20 +20,20 @@ import Foundation public class TimedPixel { - + let pixel: Pixel.Event let date: Date - + public init(_ pixel: Pixel.Event, date: Date = Date()) { self.pixel = pixel self.date = date } - + public func fire(_ fireDate: Date = Date(), withAdditionalParameters params: [String: String] = [:]) { let duration = String(fireDate.timeIntervalSince(date)) var newParams = params newParams[PixelParameters.duration] = duration Pixel.fire(pixel: pixel, withAdditionalParameters: newParams) } - + } diff --git a/Core/UIColorExtension.swift b/Core/UIColorExtension.swift index 20a49393d1..66cfb360b6 100644 --- a/Core/UIColorExtension.swift +++ b/Core/UIColorExtension.swift @@ -28,7 +28,7 @@ extension UIColor { public static var nearlyBlackLight: UIColor { return UIColor(red: 51.0 / 255.0, green: 51.0 / 255.0, blue: 51.0 / 255.0, alpha: 1.0) } - + public static var nearlyBlack: UIColor { return UIColor(red: 34.0 / 255.0, green: 34.0 / 255.0, blue: 34.0 / 255.0, alpha: 1.0) } @@ -36,11 +36,11 @@ extension UIColor { public static var charcoalGrey: UIColor { return UIColor(red: 68.0 / 255.0, green: 68.0 / 255.0, blue: 68.0 / 255.0, alpha: 1.0) } - + public static var greyishBrown: UIColor { return UIColor(red: 85.0 / 255.0, green: 85.0 / 255.0, blue: 85.0 / 255.0, alpha: 1.0) } - + public static var greyishBrown2: UIColor { return UIColor(red: 102.0 / 255.0, green: 102.0 / 255.0, blue: 102.0 / 255.0, alpha: 1.0) } @@ -48,19 +48,19 @@ extension UIColor { public static var greyish: UIColor { return UIColor(red: 170.0 / 255.0, green: 170.0 / 255.0, blue: 170.0 / 255.0, alpha: 1.0) } - + public static var greyish2: UIColor { return UIColor(red: 153.0 / 255.0, green: 153.0 / 255.0, blue: 153.0 / 255.0, alpha: 1.0) } - + public static var greyish3: UIColor { return UIColor(red: 136.0 / 255.0, green: 136.0 / 255.0, blue: 136.0 / 255.0, alpha: 1.0) } - + public static var lightGreyish: UIColor { return UIColor(red: 234.0 / 255.0, green: 234.0 / 255.0, blue: 234.0 / 255.0, alpha: 1.0) } - + public static var gray20: UIColor { return UIColor(red: 238.0 / 255.0, green: 238.0 / 255.0, blue: 238.0 / 255.0, alpha: 1.0) } @@ -72,7 +72,7 @@ extension UIColor { public static var darkGreyish: UIColor { return UIColor(red: 73.0 / 255.0, green: 73.0 / 255.0, blue: 73.0 / 255.0, alpha: 1.0) } - + public static var lightMercury: UIColor { return UIColor(red: 204.0 / 255.0, green: 204.0 / 255.0, blue: 204.0 / 255.0, alpha: 1.0) } @@ -84,39 +84,39 @@ extension UIColor { public static var cornflowerBlue: UIColor { return UIColor(red: 103.0 / 255.0, green: 143.0 / 255.0, blue: 255.0 / 255.0, alpha: 1.0) } - + public static var cornflowerDark: UIColor { return UIColor(red: 80.0 / 255.0, green: 120.0 / 255.0, blue: 233.0 / 255.0, alpha: 1.0) } - + public static var skyBlue: UIColor { return UIColor(red: 66.0 / 255.0, green: 191.0 / 255.0, blue: 254.0 / 255.0, alpha: 1.0) } - + public static var skyBlueLight: UIColor { return UIColor(red: 120.0 / 255.0, green: 210.0 / 255.0, blue: 255.0 / 255.0, alpha: 1.0) } - + public static var midGreen: UIColor { return UIColor(red: 63.0 / 255.0, green: 161.0 / 255.0, blue: 64.0 / 255.0, alpha: 1.0) } - + public static var orange: UIColor { return UIColor(red: 222.0 / 255.0, green: 88.0 / 255.0, blue: 51.0 / 255.0, alpha: 1.0) } - + public static var orangeLight: UIColor { return UIColor(red: 255.0 / 255.0, green: 135.0 / 255.0, blue: 75.0 / 255.0, alpha: 1.0) } - + public static var nearlyWhiteLight: UIColor { return UIColor(red: 250.0 / 255.0, green: 250.0 / 255.0, blue: 250.0 / 255.0, alpha: 1.0) } - + public static var nearlyWhite: UIColor { return UIColor(red: 245.0 / 255.0, green: 245.0 / 255.0, blue: 245.0 / 255.0, alpha: 1.0) } - + public static var destructive: UIColor { return UIColor.systemRed } @@ -124,7 +124,7 @@ extension UIColor { public static var yellow60: UIColor { return UIColor(hex: "F9BE1A") } - + } extension UIColor { diff --git a/Core/UIKeyCommandExtension.swift b/Core/UIKeyCommandExtension.swift index 1615f0f876..45767ea865 100644 --- a/Core/UIKeyCommandExtension.swift +++ b/Core/UIKeyCommandExtension.swift @@ -20,9 +20,9 @@ import Foundation public extension UIKeyCommand { - + static let inputBackspace = String(UnicodeScalar(8)) static let inputTab = String(UnicodeScalar(9)) static let inputEnter = String(UnicodeScalar(13)) - + } diff --git a/Core/UIViewControllerExtension.swift b/Core/UIViewControllerExtension.swift index 4e91074719..4e787b286b 100644 --- a/Core/UIViewControllerExtension.swift +++ b/Core/UIViewControllerExtension.swift @@ -21,11 +21,11 @@ import UIKit import Core extension UIViewController { - + var isSmall: Bool { return view.frame.height <= 568 } - + var isPad: Bool { return UIDevice.current.userInterfaceIdiom == .pad } @@ -77,7 +77,7 @@ extension UIViewController { } present(controller, animated: true, completion: nil) } - + public func installChildViewController(_ childController: UIViewController) { addChild(childController) childController.view.autoresizingMask = [.flexibleWidth, .flexibleHeight] @@ -114,7 +114,7 @@ extension Core.Bookmark { // Unfortuntely required to make methods available to objc extension Core.BookmarkManagedObject: UIActivityItemSource { - + @objc public func activityViewControllerPlaceholderItem(_ activityViewController: UIActivityViewController) -> Any { (self as Bookmark).activityViewControllerPlaceholderItem(activityViewController) } @@ -141,12 +141,12 @@ extension Core.Link: UIActivityItemSource { public func activityViewController(_ activityViewController: UIActivityViewController, itemForActivityType activityType: UIActivity.ActivityType?) -> Any? { - + // We don't want to save localPath to favorites or bookmarks if let localFileURL = localFileURL, activityType != .saveBookmarkInDuckDuckGo, activityType != .saveFavoriteInDuckDuckGo { - + return localFileURL } return url.removingInternalSearchParameters() diff --git a/Core/UIViewExtension.swift b/Core/UIViewExtension.swift index d1cea20ad8..6b7084ce58 100644 --- a/Core/UIViewExtension.swift +++ b/Core/UIViewExtension.swift @@ -73,7 +73,7 @@ extension UIView { view.removeFromSuperview() } } - + @MainActor public func createImageSnapshot(inBounds bounds: CGRect? = nil) -> UIImage? { let bounds = bounds ?? self.frame @@ -85,5 +85,5 @@ extension UIView { UIGraphicsEndImageContext() return image } - + } diff --git a/Core/URLFileExtension.swift b/Core/URLFileExtension.swift index 286a728c7b..ccc4a4913e 100644 --- a/Core/URLFileExtension.swift +++ b/Core/URLFileExtension.swift @@ -26,14 +26,14 @@ extension URL { public var creation: Date? { (try? resourceValues(forKeys: [.creationDateKey]))?.creationDate } - + /// The time at which the resource was most recently accessed. /// This key corresponds to an Date value, or nil if the volume doesn't support access dates. /// When you set the contentAccessDateKey for a resource, also set contentModificationDateKey in the same call to the setResourceValues(_:) method. Otherwise, the file system may set the contentAccessDateKey value to the current contentModificationDateKey value. public var contentAccess: Date? { (try? resourceValues(forKeys: [.contentAccessDateKey]))?.contentAccessDate } - + public var fileSize: Int? { return (try? resourceValues(forKeys: [.fileSizeKey]))?.fileSize } diff --git a/Core/UserAgentManager.swift b/Core/UserAgentManager.swift index 32f90094d2..6ecc0c7013 100644 --- a/Core/UserAgentManager.swift +++ b/Core/UserAgentManager.swift @@ -32,32 +32,31 @@ public protocol UserAgentManager { func update(webView: WKWebView, isDesktop: Bool, url: URL?) func userAgent(isDesktop: Bool) -> String - + } public class DefaultUserAgentManager: UserAgentManager { - + public static let shared: UserAgentManager = DefaultUserAgentManager() private var userAgent = UserAgent() - + init() { prepareUserAgent() } - + private func prepareUserAgent() { let webview = WKWebView() webview.load(URLRequest.developerInitiated(#URL("about:blank"))) - getDefaultAgent(webView: webview) { [weak self] agent in // Reference webview instance to keep it in scope and allow UA to be returned _ = webview - + guard let defaultAgent = agent else { return } self?.userAgent = UserAgent(defaultAgent: defaultAgent) } } - + public func userAgent(isDesktop: Bool) -> String { return userAgent.agent(forUrl: nil, isDesktop: isDesktop) } @@ -70,21 +69,21 @@ public class DefaultUserAgentManager: UserAgentManager { let agent = userAgent.agent(forUrl: url, isDesktop: isDesktop) webView.customUserAgent = agent } - + private func getDefaultAgent(webView: WKWebView, completion: @escaping (String?) -> Void) { webView.evaluateJavaScript("navigator.userAgent") { (result, _) in let agent = result as? String completion(agent) } } - + public static var duckDuckGoUserAgent: String { duckduckGoUserAgent(for: AppVersion.shared) } - + public static func duckduckGoUserAgent(for appVersion: AppVersion) -> String { let osVersion = UIDevice.current.systemVersion return "ddg_ios/\(appVersion.versionAndBuildNumber) (\(appVersion.identifier); iOS \(osVersion))" } - + } struct UserAgent { @@ -97,7 +96,7 @@ struct UserAgent { case brand } - + private enum Constants { // swiftlint:disable line_length static let fallbackWekKitVersion = "605.1.15" @@ -105,7 +104,7 @@ struct UserAgent { static let fallbackDefaultAgent = "Mozilla/5.0 (iPhone; CPU iPhone OS 13_5 like Mac OS X) AppleWebKit/\(fallbackWekKitVersion) (KHTML, like Gecko) Mobile/15E148" static let desktopPrefixComponent = "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15)" static let fallbackVersionComponent = "Version/13.1.1" - + static let uaOmitSitesConfigKey = "omitApplicationSites" static let uaOmitDomainConfigKey = "domain" @@ -120,13 +119,13 @@ struct UserAgent { static let uaStateKey = "state" // swiftlint:enable line_length } - + private struct Regex { static let suffix = "(AppleWebKit/.*) Mobile" static let webKitVersion = "AppleWebKit/([^ ]+) " static let osVersion = " OS ([0-9_]+)" } - + private let baseAgent: String private let baseDesktopAgent: String private let version: String @@ -146,11 +145,11 @@ struct UserAgent { brandComponent = UserAgent.createBrandComponent(withVersion: version) self.statistics = statistics } - + private func omitApplicationSites(forConfig config: PrivacyConfiguration) -> [String] { let uaSettings = config.settings(for: .customUserAgent) let omitApplicationObjs = uaSettings[Constants.uaOmitSitesConfigKey] as? [[String: String]] ?? [] - + return omitApplicationObjs.map { $0[Constants.uaOmitDomainConfigKey] ?? "" } } @@ -346,11 +345,11 @@ struct UserAgent { private static func createSafariComponent(fromAgent agent: String) -> String { let regex = try? NSRegularExpression(pattern: Regex.webKitVersion) let match = regex?.firstMatch(in: agent, options: [], range: NSRange(location: 0, length: agent.count)) - + guard let range = match?.range(at: 1) else { return Constants.fallbackSafariComponent } - + let version = (agent as NSString).substring(with: range) return "Safari/\(version)" } @@ -360,29 +359,29 @@ struct UserAgent { private static func createBaseAgent(fromAgent agent: String, versionComponent: String) -> String { var agentComponents = agent.split(separator: " ") - + guard !agentComponents.isEmpty else { return agent } - + agentComponents.insert(.init(versionComponent), at: agentComponents.endIndex - 1) return agentComponents.joined(separator: " ") } - + private static func createBaseDesktopAgent(fromAgent agent: String, versionComponent: String) -> String { let regex = try? NSRegularExpression(pattern: Regex.suffix) let match = regex?.firstMatch(in: agent, options: [], range: NSRange(location: 0, length: agent.count)) - + guard let range = match?.range(at: 1) else { return createBaseDesktopAgent(fromAgent: Constants.fallbackDefaultAgent, versionComponent: versionComponent) } - + let suffix = (agent as NSString).substring(with: range) return "\(Constants.desktopPrefixComponent) \(suffix) \(versionComponent)" } - + } private extension StatisticsStore { diff --git a/Core/UserDefaultsExtension.swift b/Core/UserDefaultsExtension.swift index eeb1bea03b..f8ff4afeaf 100644 --- a/Core/UserDefaultsExtension.swift +++ b/Core/UserDefaultsExtension.swift @@ -21,7 +21,7 @@ import Foundation extension UserDefaults { public static var app = UserDefaults.standard - + public func bool(forKey key: String, defaultValue: Bool) -> Bool { return object(forKey: key) as? Bool ?? defaultValue } diff --git a/Core/UserDefaultsPropertyWrapper.swift b/Core/UserDefaultsPropertyWrapper.swift index 9ab2131af7..bb95c04c97 100644 --- a/Core/UserDefaultsPropertyWrapper.swift +++ b/Core/UserDefaultsPropertyWrapper.swift @@ -32,10 +32,10 @@ public struct UserDefaultsWrapper { case favorites = "com.duckduckgo.ios.home.favorites" case keyboardOnNewTab = "com.duckduckgo.ios.keyboard.newtab" case keyboardOnAppLaunch = "com.duckduckgo.ios.keyboard.applaunch" - + case gridViewEnabled = "com.duckduckgo.ios.tabs.grid" case gridViewSeen = "com.duckduckgo.ios.tabs.seen" - + case preserveLoginsAllowedDomains = "com.duckduckgo.ios.PreserveLogins.userDecision.allowedDomains2" case preserveLoginsDetectionEnabled = "com.duckduckgo.ios.PreserveLogins.detectionEnabled" case preserveLoginsLegacyAllowedDomains = "com.duckduckgo.ios.PreserveLogins.userDecision.allowedDomains" @@ -55,7 +55,7 @@ public struct UserDefaultsWrapper { case faviconTabsCacheNeedsCleanup = "com.duckduckgo.ios.favicons.tabsCacheNeedsCleanup" case legacyCovidInfo = "com.duckduckgo.ios.home.covidInfo" - + case lastConfigurationRefreshDate = "com.duckduckgo.ios.lastConfigurationRefreshDate" case lastConfigurationUpdateDate = "com.duckduckgo.ios.lastConfigurationUpdateDate" case lastRemoteMessagingRefreshDate = "com.duckduckgo.ios.lastRemoteMessagingRefreshDate" @@ -73,7 +73,7 @@ public struct UserDefaultsWrapper { case emailWaitlistShouldReceiveNotifications = "com.duckduckgo.ios.showWaitlistNotification" case unseenDownloadsAvailable = "com.duckduckgo.app.unseenDownloadsAvailable" - + case lastCompiledRules = "com.duckduckgo.app.lastCompiledRules" case autofillSaveModalRejectionCount = "com.duckduckgo.ios.autofillSaveModalRejectionCount" @@ -85,9 +85,9 @@ public struct UserDefaultsWrapper { // .v2 suffix added to fix https://app.asana.com/0/547792610048271/1206524375402369/f case featureFlaggingDidVerifyInternalUser = "com.duckduckgo.app.featureFlaggingDidVerifyInternalUser.v2" - + case voiceSearchEnabled = "com.duckduckgo.app.voiceSearchEnabled" - + case autoconsentEnabled = "com.duckduckgo.ios.autoconsentEnabled" case shouldScheduleRulesCompilationOnAppLaunch = "com.duckduckgo.ios.shouldScheduleRulesCompilationOnAppLaunch" @@ -119,9 +119,9 @@ public struct UserDefaultsWrapper { case bookmarksLastGoodVersion = "com.duckduckgo.ios.bookmarksLastGoodVersion" case bookmarksMigrationVersion = "com.duckduckgo.ios.bookmarksMigrationVersion" - + case privacyConfigCustomURL = "com.duckduckgo.ios.privacyConfigCustomURL" - + case subscriptionIsActive = "com.duckduckgo.ios.subscruption.isActive" case didRefreshTimestamp = "com.duckduckgo.ios.userBehavior.didRefreshTimestamp" @@ -146,11 +146,11 @@ public struct UserDefaultsWrapper { if let storedValue = container.object(forKey: key.rawValue) as? T { return storedValue } - + if setIfEmpty { container.set(defaultValue, forKey: key.rawValue) } - + return defaultValue } set { @@ -164,13 +164,13 @@ public struct UserDefaultsWrapper { } private protocol AnyOptional { - + var isNil: Bool { get } - + } extension Optional: AnyOptional { - + var isNil: Bool { self == nil } - + } diff --git a/Core/WebCacheManager.swift b/Core/WebCacheManager.swift index bfaa47d0dd..7974cca03f 100644 --- a/Core/WebCacheManager.swift +++ b/Core/WebCacheManager.swift @@ -34,27 +34,27 @@ extension WKWebsiteDataStore { } extension HTTPCookie { - + func matchesDomain(_ domain: String) -> Bool { return self.domain == domain || (self.domain.hasPrefix(".") && domain.hasSuffix(self.domain)) } - + } @MainActor public class WebCacheManager { - + public static var shared = WebCacheManager() - + private init() { } - + /// We save cookies from the current container rather than copying them to a new container because /// the container only persists cookies to disk when the web view is used. If the user presses the fire button /// twice then the fire proofed cookies will be lost and the user will be logged out any sites they're logged in to. public func consumeCookies(cookieStorage: CookieStorage = CookieStorage(), httpCookieStore: WKHTTPCookieStore) async { guard !cookieStorage.isConsumed else { return } - + let cookies = cookieStorage.cookies var consumedCookiesCount = 0 for cookie in cookies { @@ -63,10 +63,10 @@ public class WebCacheManager { } cookieStorage.isConsumed = true } - + public func removeCookies(forDomains domains: [String], dataStore: WKWebsiteDataStore) async { - + let timeoutTask = Task.detached { try? await Task.sleep(interval: 5.0) if !Task.isCancelled { @@ -75,7 +75,7 @@ public class WebCacheManager { ]) } } - + let cookieStore = dataStore.httpCookieStore let cookies = await cookieStore.allCookies() for cookie in cookies where domains.contains(where: { cookie.matchesDomain($0) }) { @@ -83,7 +83,7 @@ public class WebCacheManager { } timeoutTask.cancel() } - + public func clear(cookieStorage: CookieStorage = CookieStorage(), logins: PreserveLogins = PreserveLogins.shared, dataStoreIdManager: DataStoreIdManager = .shared) async { @@ -92,17 +92,17 @@ public class WebCacheManager { if #available(iOS 17, *), dataStoreIdManager.hasId { cookiesToUpdate += await containerBasedClearing(storeIdManager: dataStoreIdManager) ?? [] } - + // Perform legacy clearing to migrate to new container cookiesToUpdate += await legacyDataClearing() ?? [] cookieStorage.updateCookies(cookiesToUpdate, keepingPreservedLogins: logins) } - + } extension WebCacheManager { - + @available(iOS 17, *) private func checkForLeftBehindDataStores() async { let ids = await WKWebsiteDataStore.allDataStoreIdentifiers @@ -119,17 +119,17 @@ extension WebCacheManager { var dataStore: WKWebsiteDataStore? = WKWebsiteDataStore(forIdentifier: containerId) let cookies = await dataStore?.httpCookieStore.allCookies() dataStore = nil - + let uuids = await WKWebsiteDataStore.allDataStoreIdentifiers for uuid in uuids { try? await WKWebsiteDataStore.remove(forIdentifier: uuid) } await checkForLeftBehindDataStores() - + storeIdManager.allocateNewContainerId() return cookies } - + private func legacyDataClearing() async -> [HTTPCookie]? { let timeoutTask = Task.detached { try? await Task.sleep(interval: 5.0) diff --git a/Core/global.swift b/Core/global.swift index 1a0a11d364..07c90fd66d 100644 --- a/Core/global.swift +++ b/Core/global.swift @@ -39,9 +39,9 @@ public struct Global { public class CoreModule { } extension Bundle { - + public static var core: Bundle { return Bundle(for: CoreModule.self) } - + } diff --git a/DuckDuckGo.xcodeproj/project.pbxproj b/DuckDuckGo.xcodeproj/project.pbxproj index a8dc2102a3..ac4bdd1f2e 100644 --- a/DuckDuckGo.xcodeproj/project.pbxproj +++ b/DuckDuckGo.xcodeproj/project.pbxproj @@ -5121,9 +5121,11 @@ 85CA53A724BB342B00A6288C /* Favicons */, EE50052C29C3692700AE0773 /* FeatureFlags */, 8512BCBF2061B6110085E862 /* global.swift */, + 858479C72B8792C900D156C1 /* History */, F143C2E71E4A4CD400CFDE3A /* Info.plist */, 98B001AE251EABB40090EC07 /* InfoPlist.strings */, F18608DE1E5E648100361C30 /* Javascript */, + EE7A92852AC6DE2500832A36 /* NetworkProtection */, CBAA195B27C3982A00A4BD49 /* PrivacyFeatures.swift */, CBAA195627BFDD9800A4BD49 /* SmarterEncryption */, F1134EA71F3E2B3500B73467 /* Statistics */, diff --git a/DuckDuckGo.xcodeproj/xcshareddata/xcschemes/DuckDuckGo-Alpha.xcscheme b/DuckDuckGo.xcodeproj/xcshareddata/xcschemes/DuckDuckGo-Alpha.xcscheme index 7138aee59b..3d469bdc41 100644 --- a/DuckDuckGo.xcodeproj/xcshareddata/xcschemes/DuckDuckGo-Alpha.xcscheme +++ b/DuckDuckGo.xcodeproj/xcshareddata/xcschemes/DuckDuckGo-Alpha.xcscheme @@ -38,7 +38,9 @@ ignoresPersistentStateOnLaunch = "NO" debugDocumentVersioning = "YES" debugServiceExtension = "internal" - allowLocationSimulation = "YES"> + allowLocationSimulation = "YES" + consoleMode = "0" + structuredConsoleMode = "1"> Subfeature.Handler? { + + os_log("WebView handler: %s", log: .subscription, type: .debug, methodName) + switch methodName { case Handlers.getSubscription: return getSubscription case Handlers.setSubscription: return setSubscription case Handlers.backToSettings: return backToSettings case Handlers.getSubscriptionOptions: return getSubscriptionOptions case Handlers.subscriptionSelected: return subscriptionSelected - case Handlers.activateSubscription: return activateSubscription + case Handlers.activateSubscription: + Pixel.fire(pixel: .privacyProRestorePurchaseOfferPageEntry) + return activateSubscription case Handlers.featureSelected: return featureSelected default: return nil } } - /// Values that the Frontend can use to determine the current state. // swiftlint:disable nesting @@ -173,7 +178,7 @@ final class SubscriptionPagesUseSubscriptionFeature: Subfeature, ObservableObjec case .success(let subscriptionOptions): return subscriptionOptions case .failure: - os_log(.error, log: .subscription, "Failed to obtain subscription options") + os_log("Failed to obtain subscription options", log: .subscription, type: .error) setTransactionError(.failedToGetSubscriptionOptions) return nil } @@ -181,9 +186,11 @@ final class SubscriptionPagesUseSubscriptionFeature: Subfeature, ObservableObjec } } + // swiftlint:disable:next function_body_length func subscriptionSelected(params: Any, original: WKScriptMessage) async -> Encodable? { await withTransactionInProgress { + DailyPixel.fireDailyAndCount(pixel: .privacyProPurchaseAttempt) setTransactionError(nil) setTransactionStatus(.purchasing) resetSubscriptionFlow() @@ -218,6 +225,11 @@ final class SubscriptionPagesUseSubscriptionFeature: Subfeature, ObservableObjec switch error { case .cancelledByUser: setTransactionError(.cancelledByUser) + case .accountCreationFailed: + setTransactionError(.accountCreationFailed) + case .activeSubscriptionAlreadyPresent: + setTransactionError(.hasActiveSubscription) + Pixel.fire(pixel: .privacyProRestoreAfterPurchaseAttempt) default: setTransactionError(.purchaseFailed) } @@ -230,6 +242,8 @@ final class SubscriptionPagesUseSubscriptionFeature: Subfeature, ObservableObjec switch await AppStorePurchaseFlow.completeSubscriptionPurchase(with: purchaseTransactionJWS, subscriptionAppGroup: Bundle.main.appGroup(bundle: .subs)) { case .success(let purchaseUpdate): + DailyPixel.fireDailyAndCount(pixel: .privacyProPurchaseSuccess) + UniquePixel.fire(pixel: .privacyProSubscriptionActivated) await pushPurchaseUpdate(originalMessage: message, purchaseUpdate: purchaseUpdate) case .failure: setTransactionError(.missingEntitlements) @@ -253,7 +267,7 @@ final class SubscriptionPagesUseSubscriptionFeature: Subfeature, ObservableObjec accountManager.storeAuthToken(token: authToken) accountManager.storeAccount(token: accessToken, email: accountDetails.email, externalID: accountDetails.externalID) } else { - os_log(.error, log: .subscription, "Failed to obtain subscription options") + os_log("Failed to obtain subscription options", log: .subscription, type: .error) setTransactionError(.failedToSetSubscription) } @@ -281,11 +295,11 @@ final class SubscriptionPagesUseSubscriptionFeature: Subfeature, ObservableObjec emailActivationComplete = true case .failure: - os_log(.error, log: .subscription, "Failed to restore subscription from Email") + os_log("Failed to restore subscription from Email", log: .subscription, type: .error) setTransactionError(.failedToRestoreFromEmail) } } else { - os_log(.error, log: .subscription, "General error. Could not get account Details") + os_log("General error. Could not get account Details", log: .subscription, type: .error) setTransactionError(.generalError) } return nil @@ -360,6 +374,5 @@ final class SubscriptionPagesUseSubscriptionFeature: Subfeature, ObservableObjec } } -// swiftlint:enable type_body_length #endif diff --git a/DuckDuckGo/Subscription/UserScripts/SubscriptionPagesUserScript.swift b/DuckDuckGo/Subscription/UserScripts/SubscriptionPagesUserScript.swift index 81633fa54c..c3f52e4c40 100644 --- a/DuckDuckGo/Subscription/UserScripts/SubscriptionPagesUserScript.swift +++ b/DuckDuckGo/Subscription/UserScripts/SubscriptionPagesUserScript.swift @@ -25,6 +25,7 @@ import Combine import Foundation import WebKit import UserScript +import Core /// /// The user script that will be the broker for all subscription features @@ -62,8 +63,10 @@ extension SubscriptionPagesUserScript: WKScriptMessageHandlerWithReply { } extension SubscriptionPagesUserScript: WKScriptMessageHandler { + public func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) { // unsupported + os_log("SubscriptionPagesUserScript sent an unsupported message: %s", log: .generalLog, type: .fault, message.messageName) } } diff --git a/DuckDuckGo/Subscription/ViewModel/SubscriptionEmailViewModel.swift b/DuckDuckGo/Subscription/ViewModel/SubscriptionEmailViewModel.swift index 52d746ec87..6cb2ea102f 100644 --- a/DuckDuckGo/Subscription/ViewModel/SubscriptionEmailViewModel.swift +++ b/DuckDuckGo/Subscription/ViewModel/SubscriptionEmailViewModel.swift @@ -52,9 +52,9 @@ final class SubscriptionEmailViewModel: ObservableObject { subscriptionExpired, generalError } - + private var cancellables = Set() - + init(userScript: SubscriptionPagesUserScript = SubscriptionPagesUserScript(), subFeature: SubscriptionPagesUseSubscriptionFeature = SubscriptionPagesUseSubscriptionFeature(), accountManager: AccountManager = AccountManager()) { @@ -115,6 +115,7 @@ final class SubscriptionEmailViewModel: ObservableObject { } private func completeActivation() { + DailyPixel.fireDailyAndCount(pixel: .privacyProRestorePurchaseEmailSuccess) subFeature.emailActivationComplete = false activateSubscription = true } diff --git a/DuckDuckGo/Subscription/ViewModel/SubscriptionFlowViewModel.swift b/DuckDuckGo/Subscription/ViewModel/SubscriptionFlowViewModel.swift index 623d191afb..6d0cb332d7 100644 --- a/DuckDuckGo/Subscription/ViewModel/SubscriptionFlowViewModel.swift +++ b/DuckDuckGo/Subscription/ViewModel/SubscriptionFlowViewModel.swift @@ -124,10 +124,13 @@ final class SubscriptionFlowViewModel: ObservableObject { if value != nil { switch value?.feature { case FeatureName.netP: + UniquePixel.fire(pixel: .privacyProWelcomeVPN) self?.selectedFeature = .netP case FeatureName.itr: + UniquePixel.fire(pixel: .privacyProWelcomePersonalInformationRemoval) self?.selectedFeature = .itr case FeatureName.dbp: + UniquePixel.fire(pixel: .privacyProWelcomeIdentityRestoration) self?.selectedFeature = .dbp default: break @@ -151,28 +154,63 @@ final class SubscriptionFlowViewModel: ObservableObject { } + // swiftlint:disable:next cyclomatic_complexity private func handleTransactionError(error: SubscriptionPagesUseSubscriptionFeature.UseSubscriptionError) { + + var isStoreError = false + var isBackendError = false + switch error { - case .purchaseFailed: + isStoreError = true transactionError = .purchaseFailed case .missingEntitlements: + isBackendError = true transactionError = .missingEntitlements case .failedToGetSubscriptionOptions: + isStoreError = true transactionError = .failedToGetSubscriptionOptions case .failedToSetSubscription: + isBackendError = true transactionError = .failedToSetSubscription + case .failedToRestoreFromEmail, .failedToRestoreFromEmailSubscriptionInactive: + isBackendError = true + transactionError = .generalError case .failedToRestorePastPurchase: + isStoreError = true transactionError = .failedToRestorePastPurchase + case .subscriptionNotFound: + isStoreError = true + transactionError = .generalError case .subscriptionExpired: + isStoreError = true transactionError = .subscriptionExpired case .hasActiveSubscription: + isStoreError = true + isBackendError = true transactionError = .hasActiveSubscription case .cancelledByUser: transactionError = nil + case .accountCreationFailed: + DailyPixel.fireDailyAndCount(pixel: .privacyProPurchaseFailureAccountNotCreated) + transactionError = .generalError default: transactionError = .generalError } + + if isStoreError { + DailyPixel.fireDailyAndCount(pixel: .privacyProPurchaseFailureStoreError) + } + + if isBackendError { + DailyPixel.fireDailyAndCount(pixel: .privacyProPurchaseFailureBackendError) + } + + if let transactionError, + transactionError != .hasActiveSubscription && transactionError != .cancelledByUser { + // The observer of `transactionError` does the same calculation, if the error is anything else than .hasActiveSubscription then shows a "Something went wrong" alert + DailyPixel.fireDailyAndCount(pixel: .privacyProPurchaseFailure) + } } private func setupWebViewObservers() async { @@ -255,6 +293,5 @@ final class SubscriptionFlowViewModel: ObservableObject { func navigateBack() async { await webViewModel.navigationCoordinator.goBack() } - } #endif diff --git a/DuckDuckGo/Subscription/ViewModel/SubscriptionITPViewModel.swift b/DuckDuckGo/Subscription/ViewModel/SubscriptionITPViewModel.swift index 04432065ec..6e613491a8 100644 --- a/DuckDuckGo/Subscription/ViewModel/SubscriptionITPViewModel.swift +++ b/DuckDuckGo/Subscription/ViewModel/SubscriptionITPViewModel.swift @@ -138,6 +138,7 @@ final class SubscriptionITPViewModel: ObservableObject { func initializeView() { webViewModel.navigationCoordinator.navigateTo(url: manageITPURL ) Task { await setupSubscribers() } + Pixel.fire(pixel: .privacyProIdentityRestorationSettings) } private func downloadAttachment(from url: URL) async { diff --git a/DuckDuckGo/Subscription/ViewModel/SubscriptionPIRViewModel.swift b/DuckDuckGo/Subscription/ViewModel/SubscriptionPIRViewModel.swift index 2ce1c2674b..75f6cf72ea 100644 --- a/DuckDuckGo/Subscription/ViewModel/SubscriptionPIRViewModel.swift +++ b/DuckDuckGo/Subscription/ViewModel/SubscriptionPIRViewModel.swift @@ -26,5 +26,8 @@ final class SubscriptionPIRViewModel: ObservableObject { var viewTitle = UserText.subscriptionTitle + func onAppear() { + Pixel.fire(pixel: .privacyProPersonalInformationRemovalSettings) + } } #endif diff --git a/DuckDuckGo/Subscription/ViewModel/SubscriptionRestoreViewModel.swift b/DuckDuckGo/Subscription/ViewModel/SubscriptionRestoreViewModel.swift index 5ad61c5b80..3da604761f 100644 --- a/DuckDuckGo/Subscription/ViewModel/SubscriptionRestoreViewModel.swift +++ b/DuckDuckGo/Subscription/ViewModel/SubscriptionRestoreViewModel.swift @@ -52,10 +52,10 @@ final class SubscriptionRestoreViewModel: ObservableObject { self.purchaseManager = purchaseManager self.accountManager = accountManager self.isAddingDevice = isAddingDevice - initializeView() } func initializeView() { + Pixel.fire(pixel: .privacyProSettingsAddDevice) subscriptionEmail = accountManager.email if accountManager.isUserAuthenticated { isAddingDevice = true @@ -89,6 +89,12 @@ final class SubscriptionRestoreViewModel: ObservableObject { default: activationResult = .error } + + if activationResult == .notFound { + DailyPixel.fireDailyAndCount(pixel: .privacyProRestorePurchaseStoreFailureNotFound) + } else { + DailyPixel.fireDailyAndCount(pixel: .privacyProRestorePurchaseStoreFailureOther) + } } @MainActor @@ -98,10 +104,12 @@ final class SubscriptionRestoreViewModel: ObservableObject { @MainActor func restoreAppstoreTransaction() { + DailyPixel.fireDailyAndCount(pixel: .privacyProRestorePurchaseStoreStart) Task { activationResult = .unknown do { try await subFeature.restoreAccountFromAppStorePurchase() + DailyPixel.fireDailyAndCount(pixel: .privacyProRestorePurchaseStoreSuccess) activationResult = .activated } catch let error { if let specificError = error as? SubscriptionPagesUseSubscriptionFeature.UseSubscriptionError { diff --git a/DuckDuckGo/Subscription/ViewModel/SubscriptionSettingsViewModel.swift b/DuckDuckGo/Subscription/ViewModel/SubscriptionSettingsViewModel.swift index f52e6f75c4..5c65f88e98 100644 --- a/DuckDuckGo/Subscription/ViewModel/SubscriptionSettingsViewModel.swift +++ b/DuckDuckGo/Subscription/ViewModel/SubscriptionSettingsViewModel.swift @@ -23,6 +23,7 @@ import StoreKit #if SUBSCRIPTION import Subscription +import Core @available(iOS 15.0, *) final class SubscriptionSettingsViewModel: ObservableObject { @@ -133,7 +134,5 @@ final class SubscriptionSettingsViewModel: ObservableObject { subscriptionUpdateTimer?.invalidate() signOutObserver = nil } - - } #endif diff --git a/DuckDuckGo/Subscription/Views/PurchaseInProgressView.swift b/DuckDuckGo/Subscription/Views/PurchaseInProgressView.swift index 2793d9c043..fcb83893f1 100644 --- a/DuckDuckGo/Subscription/Views/PurchaseInProgressView.swift +++ b/DuckDuckGo/Subscription/Views/PurchaseInProgressView.swift @@ -83,10 +83,11 @@ struct ViewHeightKey: PreferenceKey { } } -struct PurchaseInProgressView_Previews: PreviewProvider { - static var previews: some View { - PurchaseInProgressView(status: "Completing Purchase... ") - .previewLayout(.sizeThatFits) - .padding() - } -} +// Commented out because CI fails if a SwiftUI preview is enabled https://app.asana.com/0/414709148257752/1206774081310425/f +// struct PurchaseInProgressView_Previews: PreviewProvider { +// static var previews: some View { +// PurchaseInProgressView(status: "Completing Purchase... ") +// .previewLayout(.sizeThatFits) +// .padding() +// } +// } diff --git a/DuckDuckGo/Subscription/Views/SubscriptionEmailView.swift b/DuckDuckGo/Subscription/Views/SubscriptionEmailView.swift index d720aa434f..8712560f36 100644 --- a/DuckDuckGo/Subscription/Views/SubscriptionEmailView.swift +++ b/DuckDuckGo/Subscription/Views/SubscriptionEmailView.swift @@ -20,6 +20,7 @@ #if SUBSCRIPTION import SwiftUI import Foundation +import Core @available(iOS 15.0, *) struct SubscriptionEmailView: View { @@ -74,4 +75,13 @@ struct SubscriptionEmailView: View { } + +// Commented out because CI fails if a SwiftUI preview is enabled https://app.asana.com/0/414709148257752/1206774081310425/f +// @available(iOS 15.0, *) +// struct SubscriptionEmailView_Previews: PreviewProvider { +// static var previews: some View { +// SubscriptionEmailView() +// } +// } + #endif diff --git a/DuckDuckGo/Subscription/Views/SubscriptionFlowView.swift b/DuckDuckGo/Subscription/Views/SubscriptionFlowView.swift index 0637e2c18f..85e820fd3a 100644 --- a/DuckDuckGo/Subscription/Views/SubscriptionFlowView.swift +++ b/DuckDuckGo/Subscription/Views/SubscriptionFlowView.swift @@ -21,6 +21,7 @@ import SwiftUI import Foundation import DesignResourcesKit +import Core @available(iOS 15.0, *) struct SubscriptionFlowView: View { @@ -28,10 +29,11 @@ struct SubscriptionFlowView: View { @Environment(\.dismiss) var dismiss @StateObject var viewModel = SubscriptionFlowViewModel() @State private var shouldShowNavigationBar = false - @State private var isActive: Bool = false + @State private var isActive = false @State private var transactionError: SubscriptionFlowViewModel.SubscriptionPurchaseError? - @State private var shouldPresentError: Bool = false - + @State private var shouldPresentError = false + @State private var isFirstOnAppear = true + enum Constants { static let daxLogo = "Home" static let daxLogoSize: CGFloat = 24.0 @@ -65,7 +67,7 @@ struct SubscriptionFlowView: View { .tint(Color(designSystemColor: .textPrimary)) .environment(\.rootPresentationMode, self.$isActive) } - + @ViewBuilder private var dismissButton: some View { Button(action: { viewModel.finalizeSubscriptionFlow() }, label: { Text(UserText.subscriptionCloseButton) }) @@ -125,10 +127,10 @@ struct SubscriptionFlowView: View { viewModel.shouldDismissView = false } } - + .onChange(of: viewModel.userTappedRestoreButton) { _ in - isActive = true - viewModel.userTappedRestoreButton = false + isActive = true + viewModel.userTappedRestoreButton = false } .onChange(of: viewModel.transactionError) { value in @@ -139,6 +141,12 @@ struct SubscriptionFlowView: View { } .onAppear(perform: { + + if isFirstOnAppear && !viewModel.activateSubscriptionOnLoad { + isFirstOnAppear = false + Pixel.fire(pixel: .privacyProOfferScreenImpression) + } + setUpAppearances() Task { await viewModel.initializeViewData() } @@ -169,7 +177,6 @@ struct SubscriptionFlowView: View { private func getAlert() -> Alert { switch transactionError { - case .hasActiveSubscription: Alert( title: Text(UserText.subscriptionFoundTitle), @@ -190,32 +197,26 @@ struct SubscriptionFlowView: View { } ) } - } - - + @ViewBuilder private var webView: some View { ZStack(alignment: .top) { - // Restore View Hidden Link NavigationLink(destination: SubscriptionRestoreView(), isActive: $isActive) { EmptyView() }.isDetailLink(false) - - + AsyncHeadlessWebView(viewModel: viewModel.webViewModel) .background() if viewModel.transactionStatus != .idle { PurchaseInProgressView(status: getTransactionStatus()) } - } } - - + private func setUpAppearances() { let navAppearance = UINavigationBar.appearance() navAppearance.backgroundColor = UIColor(designSystemColor: .background) @@ -225,4 +226,13 @@ struct SubscriptionFlowView: View { } } + +// Commented out because CI fails if a SwiftUI preview is enabled https://app.asana.com/0/414709148257752/1206774081310425/f +// @available(iOS 15.0, *) +// struct SubscriptionFlowView_Previews: PreviewProvider { +// static var previews: some View { +// SubscriptionFlowView() +// } +// } + #endif diff --git a/DuckDuckGo/Subscription/Views/SubscriptionITPView.swift b/DuckDuckGo/Subscription/Views/SubscriptionITPView.swift index 6111e75246..32f0f60d0e 100644 --- a/DuckDuckGo/Subscription/Views/SubscriptionITPView.swift +++ b/DuckDuckGo/Subscription/Views/SubscriptionITPView.swift @@ -163,4 +163,13 @@ struct SubscriptionITPView: View { navAppearance.tintColor = UIColor(designSystemColor: .textPrimary) } } + +// Commented out because CI fails if a SwiftUI preview is enabled https://app.asana.com/0/414709148257752/1206774081310425/f +// @available(iOS 15.0, *) +// struct SubscriptionITPView_Previews: PreviewProvider { +// static var previews: some View { +// SubscriptionITPView() +// } +// } + #endif diff --git a/DuckDuckGo/Subscription/Views/SubscriptionPIRView.swift b/DuckDuckGo/Subscription/Views/SubscriptionPIRView.swift index 10072d9f8a..52f04501f9 100644 --- a/DuckDuckGo/Subscription/Views/SubscriptionPIRView.swift +++ b/DuckDuckGo/Subscription/Views/SubscriptionPIRView.swift @@ -64,7 +64,9 @@ struct SubscriptionPIRView: View { } .edgesIgnoringSafeArea(.all) - } + }.onAppear(perform: { + viewModel.onAppear() + }) } private var header: some View { @@ -195,7 +197,14 @@ struct SubscriptionPIRView: View { .daxBodyRegular() .tint(Color(designSystemColor: .textPrimary)) } - } +// Commented out because CI fails if a SwiftUI preview is enabled https://app.asana.com/0/414709148257752/1206774081310425/f +// @available(iOS 15.0, *) +// struct SubscriptionPIRView_Previews: PreviewProvider { +// static var previews: some View { +// SubscriptionPIRView() +// } +// } + #endif diff --git a/DuckDuckGo/Subscription/Views/SubscriptionRestoreView.swift b/DuckDuckGo/Subscription/Views/SubscriptionRestoreView.swift index ac67117332..7185163b3f 100644 --- a/DuckDuckGo/Subscription/Views/SubscriptionRestoreView.swift +++ b/DuckDuckGo/Subscription/Views/SubscriptionRestoreView.swift @@ -20,6 +20,7 @@ import Foundation import SwiftUI import DesignResourcesKit +import Core #if SUBSCRIPTION @available(iOS 15.0, *) @@ -120,13 +121,20 @@ struct SubscriptionRestoreView: View { .daxSubheadRegular() .foregroundColor(Color(designSystemColor: .textSecondary)) getCellButton(buttonText: UserText.subscriptionActivateEmailButton, - action: buttonAction) + action: { + DailyPixel.fireDailyAndCount(pixel: .privacyProRestorePurchaseEmailStart) + DailyPixel.fire(pixel: .privacyProWelcomeAddDevice) + buttonAction() + }) } else if viewModel.subscriptionEmail == nil { Text(UserText.subscriptionAddDeviceEmailDescription) .daxSubheadRegular() .foregroundColor(Color(designSystemColor: .textSecondary)) getCellButton(buttonText: UserText.subscriptionRestoreAddEmailButton, - action: buttonAction) + action: { + Pixel.fire(pixel: .privacyProAddDeviceEnterEmail) + buttonAction() + }) } else { Text(viewModel.subscriptionEmail ?? "").daxSubheadSemibold() Text(UserText.subscriptionManageEmailDescription) @@ -134,7 +142,10 @@ struct SubscriptionRestoreView: View { .foregroundColor(Color(designSystemColor: .textSecondary)) HStack { getCellButton(buttonText: UserText.subscriptionManageEmailButton, - action: buttonAction) + action: { + Pixel.fire(pixel: .privacyProSubscriptionManagementEmail) + buttonAction() + }) } } }) @@ -292,4 +303,13 @@ struct SubscriptionRestoreView_Previews: PreviewProvider { .previewDevice("iPhone 12") } } + +// Commented out because CI fails if a SwiftUI preview is enabled https://app.asana.com/0/414709148257752/1206774081310425/f +// @available(iOS 15.0, *) +// struct SubscriptionRestoreView_Previews: PreviewProvider { +// static var previews: some View { +// SubscriptionRestoreView() +// } +// } + #endif diff --git a/DuckDuckGo/Subscription/Views/SubscriptionSettingsView.swift b/DuckDuckGo/Subscription/Views/SubscriptionSettingsView.swift index 2f6a93f513..76abb5875a 100644 --- a/DuckDuckGo/Subscription/Views/SubscriptionSettingsView.swift +++ b/DuckDuckGo/Subscription/Views/SubscriptionSettingsView.swift @@ -20,6 +20,7 @@ import Foundation import SwiftUI import DesignResourcesKit +import Core class SceneEnvironment: ObservableObject { weak var windowScene: UIWindowScene? @@ -33,7 +34,8 @@ struct SubscriptionSettingsView: View { @Environment(\.dismiss) var dismiss @StateObject var viewModel = SubscriptionSettingsViewModel() @StateObject var sceneEnvironment = SceneEnvironment() - + @State var isFirstOnAppear = true + @ViewBuilder private var optionsView: some View { List { @@ -56,7 +58,10 @@ struct SubscriptionSettingsView: View { .daxBodyRegular() .foregroundColor(Color.init(designSystemColor: .accent)) }, - action: { Task { viewModel.manageSubscription() } }, + action: { + Pixel.fire(pixel: .privacyProSubscriptionManagementPlanBilling) + Task { viewModel.manageSubscription() } + }, isButton: true) } @@ -68,7 +73,7 @@ struct SubscriptionSettingsView: View { .daxBodyRegular() }) } - + SettingsCustomCell(content: { Text(UserText.subscriptionRemoveFromDevice) .daxBodyRegular() @@ -105,6 +110,7 @@ struct SubscriptionSettingsView: View { primaryButton: .cancel(Text(UserText.subscriptionRemoveCancel)) { }, secondaryButton: .destructive(Text(UserText.subscriptionRemove)) { + Pixel.fire(pixel: .privacyProSubscriptionManagementRemoval) viewModel.removeSubscription() presentationMode.wrappedValue.dismiss() } @@ -124,10 +130,13 @@ struct SubscriptionSettingsView: View { } else { optionsView } - } + }.onAppear(perform: { + if isFirstOnAppear { + isFirstOnAppear = false + Pixel.fire(pixel: .privacyProSubscriptionSettings) + } + }) .navigationBarTitleDisplayMode(.inline) - - } } @@ -148,4 +157,13 @@ struct SubscriptionSettingsView_Previews: PreviewProvider { // .preferredColorScheme(.dark) } } + +// Commented out because CI fails if a SwiftUI preview is enabled https://app.asana.com/0/414709148257752/1206774081310425/f +// @available(iOS 15.0, *) +// struct SubscriptionSettingsView_Previews: PreviewProvider { +// static var previews: some View { +// SubscriptionSettingsView() +// } +// } + #endif diff --git a/DuckDuckGo/VPNWaitlistView.swift b/DuckDuckGo/VPNWaitlistView.swift index 9f912e48b6..22bd2c6b45 100644 --- a/DuckDuckGo/VPNWaitlistView.swift +++ b/DuckDuckGo/VPNWaitlistView.swift @@ -113,6 +113,13 @@ struct VPNWaitlistSignUpView: View { } +@available(iOS 15.0, *) +struct VPNWaitlistSignUpView_Previews: PreviewProvider { + static var previews: some View { + VPNWaitlistSignUpView(requestInFlight: false) { _ in } + } +} + // MARK: - Joined Waitlist Views @available(iOS 15.0, *) diff --git a/submodules/privacy-reference-tests b/submodules/privacy-reference-tests index a603ff9af2..40ce86837d 160000 --- a/submodules/privacy-reference-tests +++ b/submodules/privacy-reference-tests @@ -1 +1 @@ -Subproject commit a603ff9af22ca3ff7ce2e7ffbfe18c447d9f23e8 +Subproject commit 40ce86837def0adbf558f00ed0531ab4df5839a8 From b22a76879faa2ffca012c3a96080a358f13f9da4 Mon Sep 17 00:00:00 2001 From: Christopher Brind Date: Mon, 11 Mar 2024 11:57:38 +0000 Subject: [PATCH 9/9] update metadata (#2571) Task/Issue URL: https://app.asana.com/0/0/1206806894867215/f Tech Design URL: CC: Description: Updates Metadata --- fastlane/metadata/cs/description.txt | 2 -- fastlane/metadata/de-DE/description.txt | 3 --- fastlane/metadata/en-CA/description.txt | 6 ++--- fastlane/metadata/en-GB/description.txt | 6 ++--- fastlane/metadata/en-US/description.txt | 8 +++--- fastlane/metadata/es-ES/description.txt | 3 --- fastlane/metadata/fi/description.txt | 3 --- fastlane/metadata/fr-FR/description.txt | 2 -- fastlane/metadata/it/description.txt | 2 -- fastlane/metadata/nl-NL/description.txt | 2 -- fastlane/metadata/nl-NL/it/description.txt | 27 ------------------- fastlane/metadata/nl-NL/it/marketing_url.txt | 1 - fastlane/metadata/nl-NL/it/name.txt | 1 - fastlane/metadata/nl-NL/it/privacy_url.txt | 1 - .../metadata/nl-NL/it/promotional_text.txt | 1 - fastlane/metadata/nl-NL/it/subtitle.txt | 1 - fastlane/metadata/nl-NL/it/support_url.txt | 1 - fastlane/metadata/no/description.txt | 5 +--- fastlane/metadata/pl/description.txt | 2 -- fastlane/metadata/pt-PT/description.txt | 4 +-- fastlane/metadata/ro/description.txt | 2 -- fastlane/metadata/ru/description.txt | 4 --- fastlane/metadata/sv/description.txt | 5 +--- fastlane/metadata/tr/description.txt | 3 --- 24 files changed, 10 insertions(+), 85 deletions(-) delete mode 100644 fastlane/metadata/nl-NL/it/description.txt delete mode 100644 fastlane/metadata/nl-NL/it/marketing_url.txt delete mode 100644 fastlane/metadata/nl-NL/it/name.txt delete mode 100644 fastlane/metadata/nl-NL/it/privacy_url.txt delete mode 100644 fastlane/metadata/nl-NL/it/promotional_text.txt delete mode 100644 fastlane/metadata/nl-NL/it/subtitle.txt delete mode 100644 fastlane/metadata/nl-NL/it/support_url.txt diff --git a/fastlane/metadata/cs/description.txt b/fastlane/metadata/cs/description.txt index 9cb9498c21..9d1c85dd9c 100644 --- a/fastlane/metadata/cs/description.txt +++ b/fastlane/metadata/cs/description.txt @@ -20,7 +20,5 @@ KAŽDODENNÍ OCHRANA SOUKROMÍ • Ukaž, jak to vidíš se svým soukromím, pomocí Globální kontroly ochrany osobních údajů (GPC). - Další informace najdeš na https://help.duckduckgo.com/duckduckgo-help-pages/privacy/gpc/. -Poznámka k naší ochraně před načítáním trackerů třetích stran po kliknutí na reklamu na DuckDuckGo: Tato naše ochrana jde vysoko nad rámec toho, co běžně nabízí většina oblíbených prohlížečů, ale neustále pracujeme na tom, aby byla ještě komplexnější. Aktuálně je to tak, že když chce inzerent na svých webových stránkách zjišťovat konverze z našich soukromých reklam ve vyhledávání DuckDuckGo, naše ochrana před načítáním trackerů třetích stran nezastaví načítání požadavků bat.bing.com na webových stránkách inzerenta po kliknutí na reklamu na DuckDuckGo, ale na všech ostatních místech tyto požadavky zablokujeme. Je to proto, že soukromé reklamy ve vyhledávání na DuckDuckGo provozujeme ve spolupráci se společností Microsoft. V rámci tohoto partnerství je zobrazování reklam na DuckDuckGo anonymní a společnost Microsoft se zavázala, že naše uživatele nebude po kliknutí na reklamy profilovat. - Další informace o tomto tématu a bezplatné ochraně před trackery si můžeš přečíst na adrese https://help.duckduckgo.com/privacy/web-tracking-protections. diff --git a/fastlane/metadata/de-DE/description.txt b/fastlane/metadata/de-DE/description.txt index b9dfbca593..aa3e6d5c92 100644 --- a/fastlane/metadata/de-DE/description.txt +++ b/fastlane/metadata/de-DE/description.txt @@ -1,4 +1,3 @@ - Die DuckDuckGo-App bietet den umfassendsten Online-Datenschutz mit nur einem Knopfdruck. Mit einem einzigen Download erhältst du einen neuen Browser für die alltägliche Nutzung, der nahtlosen Schutz beim Suchen und Surfen bietet sowie Zugriff auf Tracking-Schutz beim Empfang von E-Mails. Viele dieser Schutzmaßnahmen werden in den meisten gängigen Browsern standardmäßig nicht angeboten. HIGHLIGHTS DER FUNKTION @@ -17,8 +16,6 @@ DATENSCHUTZKONTROLLE FÜR JEDEN TAG • Signalisiere deine Privatsphäre-Präferenz mit Global Privacy Control (GPC) – GPC ist in unsere App integriert und hilft dir, deine Widerspruchsrechte automatisch auszuüben, indem du Websites darüber informierst, dass sie deine persönlichen Daten nicht verkaufen oder weitergeben dürfen. Ob deine Rechtsansprüche (z. B. aktuelle oder zukünftige CCPA- oder DSGVO-Anforderungen) auf diese Weise durchgesetzt werden können, hängt von den Gesetzen in deiner Gerichtsbarkeit ab. -3rd-Party Tracker Loading Protection nach Klicken auf DuckDuckGo-Werbung geht weit über das hinaus, was die meisten gängigen Browser standardmäßig bieten. Wir arbeiten ständig daran, die Funktion noch umfassender zu gestalten. Wenn ein Werbetreibender die Konvertierung auf seiner eigenen Website für DuckDuckGo Private Search-Werbung ermitteln möchte, verhindert 3rd-Party Tracker Loading Protection nicht, dass bat.bing.com-Anfragen auf der Website des Werbetreibenden geladen werden, nachdem die DuckDuckGo-Werbung angeklickt wurde. Diese Anfragen werden jedoch in allen anderen Zusammenhängen gestoppt. Das liegt daran, dass DuckDuckGo Private Search-Werbung in Partnerschaft mit Microsoft erfolgt. Im Rahmen dieser Zusammenarbeit ist das Ansehen von Werbung auf DuckDuckGo anonym. Zudem hat sich Microsoft verpflichtet, keine Profile der Anzeigenklicks unserer Nutzer zu erstellen. - Weitere Informationen hierzu sowie zu Tracking Protection unter https://help.duckduckgo.com/privacy/web-tracking-protections/ Du musst nicht warten, um dir deine Privatsphäre zurückzuholen. Schließ dich den Millionen von Menschen an, die DuckDuckGo nutzen, und schütze viele deiner täglichen Online-Aktivitäten mit einer einzigen App. Das ist Datenschutz, vereinfacht. diff --git a/fastlane/metadata/en-CA/description.txt b/fastlane/metadata/en-CA/description.txt index 36d8d4e3dd..5b53dce2e6 100644 --- a/fastlane/metadata/en-CA/description.txt +++ b/fastlane/metadata/en-CA/description.txt @@ -1,4 +1,4 @@ -The DuckDuckGo browser provides the most comprehensive online privacy protection in one app. Unlike most popular browsers, it has powerful privacy protections by default, including our search engine that doesn’t track your history and over a dozen other built-in protections. Millions of people use DuckDuckGo as their go-to browser to protect their everyday online activities, from searching to browsing, emailing, and more. +DuckDuckGo is a free browser that provides the most comprehensive online privacy protection in one app. Unlike most popular browsers, it has powerful privacy protections by default, including our search engine that doesn’t track your history and over a dozen other built-in protections. Millions of people use DuckDuckGo as their go-to browser to protect their everyday online activities, from searching to browsing, emailing, and more.   FEATURE HIGHLIGHTS • Search Privately by Default - DuckDuckGo Private Search comes built-in, so you can easily search the web without being tracked. @@ -20,9 +20,7 @@ EVERYDAY PRIVACY CONTROLS   • Signal Your Privacy Preference with Global Privacy Control (GPC) - Built in to our app, GPC intends to help you express your opt-out rights automatically by telling websites not to sell or share your personal information. Whether it can be used to enforce your legal rights (for example, current or future CCPA, GDPR requirements) depends on the laws in your jurisdiction.   -Note about our 3rd-Party Tracker Loading Protection following DuckDuckGo ad clicks: Our 3rd-Party Tracker Protection goes above and beyond what you get in most popular browsers by default, but we’re constantly working to make it more comprehensive. Currently, if an advertiser wants to detect conversions on their own website for our private DuckDuckGo search ads, our 3rd-Party Tracker Loading Protection will not stop bat.bing.com requests from loading on the advertiser’s website following DuckDuckGo ad clicks, but those requests are stopped in all other contexts. This is because DuckDuckGo private search advertising is in partnership with Microsoft. As part of that partnership, viewing ads on DuckDuckGo is anonymous and Microsoft has committed to not profile our users’ ad clicks. -  -Read more about this and our free Tracking Protections at https://help.duckduckgo.com/privacy/web-tracking-protections +Read more about our free Tracking Protections at https://help.duckduckgo.com/privacy/web-tracking-protections You don't need to wait to take back your privacy. Join the millions of people using DuckDuckGo and protect many of your everyday online activities with one app. It's privacy, simplified. diff --git a/fastlane/metadata/en-GB/description.txt b/fastlane/metadata/en-GB/description.txt index c0444a66a2..2a1b6998f0 100644 --- a/fastlane/metadata/en-GB/description.txt +++ b/fastlane/metadata/en-GB/description.txt @@ -1,4 +1,4 @@ -The DuckDuckGo browser provides the most comprehensive online privacy protection in one app. Unlike most popular browsers, it has powerful privacy protections by default, including our search engine that doesn’t track your history and over a dozen other built-in protections. Millions of people use DuckDuckGo as their go-to browser to protect their everyday online activities, from searching to browsing, emailing, and more. +DuckDuckGo is a free browser that provides the most comprehensive online privacy protection in one app. Unlike most popular browsers, it has powerful privacy protections by default, including our search engine that doesn’t track your history and over a dozen other built-in protections. Millions of people use DuckDuckGo as their go-to browser to protect their everyday online activities, from searching to browsing, emailing, and more. FEATURE HIGHLIGHTS • Search Privately by Default — DuckDuckGo Private Search comes built-in, so you can easily search the web without being tracked. @@ -20,9 +20,7 @@ EVERYDAY PRIVACY CONTROLS • Signal Your Privacy Preference with Global Privacy Control (GPC) — built-into our app, GPC intends to help you express your opt-out rights automatically by telling websites not to sell or share your personal information. Whether it can be used to enforce your legal rights (for example, current or future CCPA, GDPR requirements) depends on the laws in your jurisdiction. -Note about our 3rd-Party Tracker Loading Protection following DuckDuckGo ad clicks: Our 3rd-Party Tracker Protection goes above and beyond what you get in most popular browsers by default, but we’re constantly working to make it more comprehensive. Currently, if an advertiser wants to detect conversions on their own website for our private DuckDuckGo search ads, our 3rd-Party Tracker Loading Protection will not stop bat.bing.com requests from loading on the advertiser’s website following DuckDuckGo ad clicks, but those requests are stopped in all other contexts. This is because DuckDuckGo private search advertising is in partnership with Microsoft. As part of that partnership, viewing ads on DuckDuckGo is anonymous and Microsoft has committed to not profile our users’ ad clicks. - -Read more about this and our Tracking Protections at https://help.duckduckgo.com/privacy/web-tracking-protections/ +Read more about our free Tracking Protections at https://help.duckduckgo.com/privacy/web-tracking-protections/ You don't need to wait to take back your privacy. Join the millions of people using DuckDuckGo and protect many of your everyday online activities with one app. It's privacy, simplified. diff --git a/fastlane/metadata/en-US/description.txt b/fastlane/metadata/en-US/description.txt index 61bf4cae26..e16d809a31 100644 --- a/fastlane/metadata/en-US/description.txt +++ b/fastlane/metadata/en-US/description.txt @@ -1,4 +1,4 @@ -The DuckDuckGo browser provides the most comprehensive online privacy protection in one app. Unlike most popular browsers, it has powerful privacy protections by default, including our search engine that doesn’t track your history and over a dozen other built-in protections. Millions of people use DuckDuckGo as their go-to browser to protect their everyday online activities, from searching to browsing, emailing, and more. +DuckDuckGo is a free browser that provides the most comprehensive online privacy protection in one app. Unlike most popular browsers, it has powerful privacy protections by default, including our search engine that doesn’t track your history and over a dozen other built-in protections. Millions of people use DuckDuckGo as their go-to browser to protect their everyday online activities, from searching to browsing, emailing, and more.   FEATURE HIGHLIGHTS • Search Privately by Default - DuckDuckGo Private Search comes built-in, so you can easily search the web without being tracked. @@ -19,10 +19,8 @@ EVERYDAY PRIVACY CONTROLS • Tap Fire Button, Burn Data - Clear your tabs and browsing data fast with our Fire Button.   • Signal Your Privacy Preference with Global Privacy Control (GPC) - Built in to our app, GPC intends to help you express your opt-out rights automatically by telling websites not to sell or share your personal information. Whether it can be used to enforce your legal rights (for example, current or future CCPA, GDPR requirements) depends on the laws in your jurisdiction. -  -Note about our 3rd-Party Tracker Loading Protection following DuckDuckGo ad clicks: Our 3rd-Party Tracker Protection goes above and beyond what you get in most popular browsers by default, but we’re constantly working to make it more comprehensive. Currently, if an advertiser wants to detect conversions on their own website for our private DuckDuckGo search ads, our 3rd-Party Tracker Loading Protection will not stop bat.bing.com requests from loading on the advertiser’s website following DuckDuckGo ad clicks, but those requests are stopped in all other contexts. This is because DuckDuckGo private search advertising is in partnership with Microsoft. As part of that partnership, viewing ads on DuckDuckGo is anonymous and Microsoft has committed to not profile our users’ ad clicks. -  -Read more about this and our free Tracking Protections at https://help.duckduckgo.com/privacy/web-tracking-protections + +Read more about our free Tracking Protections at https://help.duckduckgo.com/privacy/web-tracking-protections You don't need to wait to take back your privacy. Join the millions of people using DuckDuckGo and protect many of your everyday online activities with one app. It's privacy, simplified. diff --git a/fastlane/metadata/es-ES/description.txt b/fastlane/metadata/es-ES/description.txt index af71cefc1d..5eba4a2f86 100644 --- a/fastlane/metadata/es-ES/description.txt +++ b/fastlane/metadata/es-ES/description.txt @@ -1,4 +1,3 @@ - La aplicación DuckDuckGo ofrece la protección de la privacidad en línea más completa con solo pulsar un botón. Con una sola descarga, obtienes un nuevo navegador de uso cotidiano que ofrece una protección perfecta mientras realizas búsquedas y navegas y acceso a protecciones de rastreo cuando recibes correo electrónico. Muchas de estas protecciones no se ofrecen en la mayoría de navegadores por defecto. FUNCIONES DESTACADAS @@ -17,8 +16,6 @@ CONTROLES DE PRIVACIDAD COTIDIANOS • Marca tus preferencias de privacidad con Global Privacy Control (GPC): integrado en la aplicación, GPC te ayuda a expresar tus derechos de exclusión voluntaria automáticamente para indicar a los sitios web que no pueden vender o compartir tu información personal. El hecho de que se pueda utilizar para hacer valer tus derechos (por ejemplo, la CCPA actual o futura, los requisitos del RGPD) depende de las leyes de tu jurisdicción. -Nota sobre nuestra protección de carga de rastreadores de terceros tras los clics de los anuncios de DuckDuckGo: nuestra protección de carga de rastreadores de terceros va más allá de lo que obtienes en la mayoría de los navegadores populares por defecto, pero estamos trabajando constantemente para hacerla aún más completa. Actualmente, si un anunciante desea detectar conversiones en su propio sitio web para nuestros anuncios privados de búsqueda de DuckDuckGo, nuestra protección de carga de rastreadores de terceros no impedirá que las solicitudes de bat.bing.com se carguen del sitio web del anunciante tras los clics en anuncios de DuckDuckGo, pero esas solicitudes se detienen en todos los demás contextos. Esto se debe a que la publicidad de DuckDuckGo Private Search está asociada con Microsoft. Como parte de esa asociación, la visualización de anuncios en DuckDuckGo es anónima y Microsoft se ha comprometido a no perfilar los clics en los anuncios de nuestros usuarios. - Lee más sobre esto y sobre nuestras protecciones de rastreo en https://help.duckduckgo.com/privacy/web-tracking-protections/ No tienes que esperar para recuperar tu privacidad. Únete a los millones de personas que usan DuckDuckGo y protege muchas de tus actividades cotidianas en línea con una sola aplicación. Es tan solo privacidad simplificada. diff --git a/fastlane/metadata/fi/description.txt b/fastlane/metadata/fi/description.txt index 6738709a7c..b3fe33787b 100644 --- a/fastlane/metadata/fi/description.txt +++ b/fastlane/metadata/fi/description.txt @@ -1,4 +1,3 @@ - DuckDuckGo-sovellus tarjoaa kattavimman yksityisyydensuojan verkossa napin painalluksella. Yhdellä latauksella saat uuden arkiselaimen, joka tarjoaa saumattoman suojan, kun haet ja selaat, ja seurantasuojaukset, kun vastaanotat sähköposteja. Monia näistä suojauksista ei tarjota oletusarvoisesti suosituimmissa selaimissa. OMINAISUUKSIEN KOHOKOHDAT  @@ -17,8 +16,6 @@ JOKAPÄIVÄISTÄ YKSITYISYYDEN HALLINTAA • Ilmoita yksityisyysasetuksestasi Global Privacy Control (GPC) -toiminnolla – sovellukseemme sisäänrakennettu GPC auttaa sinua ilmaisemaan kieltäytymisoikeutesi automaattisesti käskemällä sivustoja olemaan myymättä tai jakamatta henkilötietojasi. Se, voidaanko sitä käyttää laillisten oikeuksiesi (esimerkiksi nykyisen tai tulevan CCPA:n tai yleisen tietosuoja-asetuksen vaatimusten) täytäntöönpanoon, riippuu oman lainkäyttöalueesi laeista.  -Huomautus kolmannen osapuolen seurantalaitteiden lataussuojauksesta DuckDuckGo-mainosnapsautusten jälkeen: Kolmannen osapuolen seurantasuojauksemme ylittää sen, mitä saat suosituimmissa selaimissa oletusarvoisesti, mutta pyrimme jatkuvasti tekemään siitä kattavamman. Tällä hetkellä, jos mainostaja haluaa havaita yksityisten DuckDuckGo-hakumainosten konversioita omalla sivustollaan, kolmannen osapuolen seurantalaitteiden lataussuojaus ei estä bat.bing.com-pyyntöjen lataamista mainostajan verkkosivustolle DuckDuckGo-mainosklikkausten jälkeen, mutta pyynnöt pysäytetään kaikissa muissa yhteyksissä. Tämä johtuu siitä, että DuckDuckGon Private Search -haun mainonta tapahtuu yhteistyössä Microsoftin kanssa. Osana kumppanuutta, mainosten katselu DuckDuckGossa on nimetöntä, ja Microsoft on sitoutunut olemaan profiloimamatta käyttäjiemme mainosnapsautuksia.    - Lue lisää tästä ja jäljityssuojastamme osoitteessa https://help.duckduckgo.com/privacy/web-tracking-protections/.  Sinun ei tarvitse odottaa tietosuojasi takaisinottoa. Liity miljoonien DuckDuckGo-käyttäjien joukkoon ja suojaa monia päivittäisiä verkkotoimintojasi yhdellä sovelluksella. Se on tietosuojaa, yksinkertaistettuna.   diff --git a/fastlane/metadata/fr-FR/description.txt b/fastlane/metadata/fr-FR/description.txt index 92fada92bd..9ea936cf54 100644 --- a/fastlane/metadata/fr-FR/description.txt +++ b/fastlane/metadata/fr-FR/description.txt @@ -16,8 +16,6 @@ CONTRÔLES QUOTIDIENS DE LA CONFIDENTIALITÉ • Signalez vos préférences en matière de confidentialité avec le Global Privacy Control (GPC) – intégré à notre application, le GPC vous aide à formuler automatiquement vos droits de refus en indiquant aux sites Web de ne pas vendre ou partager vos informations personnelles. Le fait qu'il puisse être utilisé pour faire valoir vos droits légaux (par exemple, CCPA actuel ou futur, conditions du RGPD) dépend des lois de votre juridiction.  -Remarque à propos de notre « 3rd-Party Tracker Loading Protection » suivant les clics sur les publicités DuckDuckGo – notre protection contre les traqueurs tiers va bien au-delà de ce que vous proposent par défaut les navigateurs les plus populaires, mais nous nous efforçons constamment de l'enrichir. À l'heure actuelle, si un annonceur souhaite détecter les conversions sur son propre site Web pour nos publicités de recherche privée DuckDuckGo, notre protection contre le chargement de traqueurs tiers n'empêchera pas les requêtes bat.bing.com de se charger sur le site Web de l'annonceur après les clics sur les publicités DuckDuckGo, mais ces requêtes sont bloquées dans tous les autres contextes. Cela s'explique par le fait que la publicité de recherche privée DuckDuckGo a un partenariat avec Microsoft. Dans le cadre de ce partenariat, le visionnage des publicités sur DuckDuckGo est anonyme et Microsoft s'est engagé à ne pas profiler les clics sur les publicités de nos utilisateurs.    - Découvrez-en plus à ce sujet et sur nos protections contre le suivi sur https://help.duckduckgo.com/privacy/web-tracking-protections/  Inutile d'attendre pour reprendre le contrôle de votre vie privée. La confidentialité, simplifiée.   \ No newline at end of file diff --git a/fastlane/metadata/it/description.txt b/fastlane/metadata/it/description.txt index bc48c3f911..03dc40d5a6 100644 --- a/fastlane/metadata/it/description.txt +++ b/fastlane/metadata/it/description.txt @@ -20,8 +20,6 @@ CONTROLLI QUOTIDIANI SULLA PRIVACY • Global Privacy Control (GPC) per comunicare le proprie preferenze in materia di privacy - Per saperne di più, visita https://help.duckduckgo.com/duckduckgo-help-pages/privacy/gpc/ -Nota sulla nostra protezione dal caricamento dei sistemi di tracciamento di terze parti a seguito dei clic sugli annunci di DuckDuckGo: la nostra protezione dai sistemi di tracciamento di terze parti è superiore a quella disponibile per impostazione predefinita nei browser più diffusi, ma stiamo lavorando costantemente per renderla più completa. Attualmente, se un inserzionista desidera rilevare le conversioni sul proprio sito Web per i nostri annunci di ricerca privati di DuckDuckGo, la nostra protezione dal caricamento del sistemi di tracciamento di terze parti non interrompe il caricamento delle richieste di bat.bing.com sul sito Web dell'inserzionista in seguito ai clic sugli annunci di DuckDuckGo, ma tali richieste vengono interrotte in tutti gli altri contesti. Questo perché la pubblicità Private Search di DuckDuckGo è in collaborazione con Microsoft. Nell'ambito di questa partnership, la visualizzazione degli annunci su DuckDuckGo è anonima e Microsoft si è impegnata a non profilare i clic degli annunci dei nostri utenti. - Maggiori informazioni su questo e sulle nostre protezioni gratuite dal tracciamento sono disponibili all'indirizzo https://help.duckduckgo.com/privacy/web-tracking-protections diff --git a/fastlane/metadata/nl-NL/description.txt b/fastlane/metadata/nl-NL/description.txt index c1b4c2dba5..3c05387304 100644 --- a/fastlane/metadata/nl-NL/description.txt +++ b/fastlane/metadata/nl-NL/description.txt @@ -20,6 +20,4 @@ PRIVACYCONTROLES VOOR ELKE DAG • Geef je privacyvoorkeuren aan met Global Privacy Control (GPC) - Meer informatie vind je op https://help.duckduckgo.com/duckduckgo-help-pages/privacy/gpc/ -Opmerking over het volgen van klikken op DuckDuckGo-advertenties door onze functie voor het beschermen tegen trackers van derden: onze bescherming tegen trackers van derden gaat verder dan wat de meeste populaire browsers je standaard bieden, maar we werken voortdurend aan verdere uitbreiding ervan. Als een adverteerder conversies voor onze DuckDuckGo-advertenties bij privézoekopdrachten op zijn eigen website wil detecteren, zal onze bescherming tegen trackers van derden niet voorkomen dat bat.bing.com-verzoeken op de website van de adverteerder worden geladen nadat er op DuckDuckGo-advertenties is geklikt. Die verzoeken worden echter gestopt in alle andere contexten. Dat komt doordat voor de advertenties bij privézoekopdrachten in DuckDuckGo wordt samengewerkt met Microsoft. Het bekijken van advertenties op DuckDuckGo is als onderdeel van die samenwerking anoniem. Ook heeft Microsoft afgesproken om de advertentieklikken van onze gebruikers niet te gebruiken om profielen te maken. - Lees meer over deze functie en onze gratis trackingbescherming op https://help.duckduckgo.com/privacy/web-tracking-protections diff --git a/fastlane/metadata/nl-NL/it/description.txt b/fastlane/metadata/nl-NL/it/description.txt deleted file mode 100644 index bc48c3f911..0000000000 --- a/fastlane/metadata/nl-NL/it/description.txt +++ /dev/null @@ -1,27 +0,0 @@ -L'app DuckDuckGo offre protezioni della privacy online complete con la semplice pressione di un pulsante. Con un solo download gratuito avrai un browser Internet privato per uso quotidiano che offre una protezione perfetta durante la ricerca e la navigazione e l'accesso alla protezione del tracciamento per le e-mail che ricevi e le app che usi. Molte di queste protezioni non sono offerte automaticamente dai browser web più famosi. - -FUNZIONALITÀ PRINCIPALI -• Private Search disponibile automaticamente - DuckDuckGo Private Search integrato. - -• Blocco dei cookie di tracciamento - Impedisce il tracciamento alla maggior parte dei cookie di terze parti durante la navigazione online da un sito web all'altro. - -• Blocco della maggior parte dei sistemi di tracciamento prima che vengano caricati - Impedisce alle aziende di raccogliere e utilizzare i dati personali raccolti dai sistemi di tracciamento nascosti nei siti web. - -• Crittografia imposta - Obbliga molti siti visitati a utilizzare automaticamente una connessione crittografata (HTTPS). - -• Blocco dei sistemi di tracciamento e-mail (Beta) - Crea un indirizzo e-mail @duck.com personale e usa Email Protection per bloccare la maggior parte dei sistemi di tracciamento della elettronica, nascondere l'indirizzo esistente e inoltrare i messaggi alla posta in arrivo. - -• Elusione del fingerprinting - Aiuta a impedire alle aziende di creare un identificatore univoco per l'utente. - -Offriamo molte protezioni non disponibili nella maggior parte dei browser Internet (anche i browser in incognito), inclusa la protezione dal tracciamento dei link, dal monitoraggio AMP di Google e altro ancora. - -CONTROLLI QUOTIDIANI SULLA PRIVACY -• Fire Button per l'eliminazione dei dati - Cancella rapidamente le schede e i dati di navigazione con il nostro Fire Button. - -• Global Privacy Control (GPC) per comunicare le proprie preferenze in materia di privacy - Per saperne di più, visita https://help.duckduckgo.com/duckduckgo-help-pages/privacy/gpc/ - -Nota sulla nostra protezione dal caricamento dei sistemi di tracciamento di terze parti a seguito dei clic sugli annunci di DuckDuckGo: la nostra protezione dai sistemi di tracciamento di terze parti è superiore a quella disponibile per impostazione predefinita nei browser più diffusi, ma stiamo lavorando costantemente per renderla più completa. Attualmente, se un inserzionista desidera rilevare le conversioni sul proprio sito Web per i nostri annunci di ricerca privati di DuckDuckGo, la nostra protezione dal caricamento del sistemi di tracciamento di terze parti non interrompe il caricamento delle richieste di bat.bing.com sul sito Web dell'inserzionista in seguito ai clic sugli annunci di DuckDuckGo, ma tali richieste vengono interrotte in tutti gli altri contesti. Questo perché la pubblicità Private Search di DuckDuckGo è in collaborazione con Microsoft. Nell'ambito di questa partnership, la visualizzazione degli annunci su DuckDuckGo è anonima e Microsoft si è impegnata a non profilare i clic degli annunci dei nostri utenti. - -Maggiori informazioni su questo e sulle nostre protezioni gratuite dal tracciamento sono disponibili all'indirizzo https://help.duckduckgo.com/privacy/web-tracking-protections - - diff --git a/fastlane/metadata/nl-NL/it/marketing_url.txt b/fastlane/metadata/nl-NL/it/marketing_url.txt deleted file mode 100644 index 07b7d171d9..0000000000 --- a/fastlane/metadata/nl-NL/it/marketing_url.txt +++ /dev/null @@ -1 +0,0 @@ -https://duckduckgo.com/ diff --git a/fastlane/metadata/nl-NL/it/name.txt b/fastlane/metadata/nl-NL/it/name.txt deleted file mode 100644 index d9601ee135..0000000000 --- a/fastlane/metadata/nl-NL/it/name.txt +++ /dev/null @@ -1 +0,0 @@ -DuckDuckGo Private Browser diff --git a/fastlane/metadata/nl-NL/it/privacy_url.txt b/fastlane/metadata/nl-NL/it/privacy_url.txt deleted file mode 100644 index 910a16c034..0000000000 --- a/fastlane/metadata/nl-NL/it/privacy_url.txt +++ /dev/null @@ -1 +0,0 @@ -https://duckduckgo.com/privacy diff --git a/fastlane/metadata/nl-NL/it/promotional_text.txt b/fastlane/metadata/nl-NL/it/promotional_text.txt deleted file mode 100644 index f253621fcf..0000000000 --- a/fastlane/metadata/nl-NL/it/promotional_text.txt +++ /dev/null @@ -1 +0,0 @@ -La privacy semplificata. Un'unica app ricca di funzionalità e svariati metodi per proteggere la privacy. \ No newline at end of file diff --git a/fastlane/metadata/nl-NL/it/subtitle.txt b/fastlane/metadata/nl-NL/it/subtitle.txt deleted file mode 100644 index df074ced85..0000000000 --- a/fastlane/metadata/nl-NL/it/subtitle.txt +++ /dev/null @@ -1 +0,0 @@ -La privacy semplificata. \ No newline at end of file diff --git a/fastlane/metadata/nl-NL/it/support_url.txt b/fastlane/metadata/nl-NL/it/support_url.txt deleted file mode 100644 index a741075b71..0000000000 --- a/fastlane/metadata/nl-NL/it/support_url.txt +++ /dev/null @@ -1 +0,0 @@ -https://duckduckgo.com/feedback diff --git a/fastlane/metadata/no/description.txt b/fastlane/metadata/no/description.txt index 6af807818e..e9546ff48d 100644 --- a/fastlane/metadata/no/description.txt +++ b/fastlane/metadata/no/description.txt @@ -1,4 +1,3 @@ - DuckDuckGo-appen gir det mest omfattende personvernet på nettet med et trykk på en knapp. Med én nedlasting får du en hverdagsnettleser som tilbyr sømløs beskyttelse mens du søker og surfer samt tilgang til sporingsbeskyttelse på e-poster du får. Mange av disse beskyttelsene tilbys ikke som standard i de fleste populære nettleserne.  HØYDEPUNKTER  @@ -19,9 +18,7 @@ Vi har også mange beskyttelser som ikke er tilgjengelige i de fleste nettlesere PERSONVERNKONTROLLER I HVERDAGEN • Trykk på Fire Button-knappen og brenn data – tøm faner og nettleserdata med Fire Button. -• Signaliser personvernreferansene dine med GPC (Global Privacy Control) – som er innebygd i appen og uttrykker automatisk reservasjonsretten din ved å gi beskjed til nettstedene om at de ikke kan selge eller dele personopplysningene dine. Om dette kan brukes til å håndheve dine juridiske rettigheter (for eksempel i henhold til nåværende eller fremtidige CCPA- og GDPR-krav), avhenger av lovene i din jurisdiksjon.  - -Merknad om vår beskyttelse mot tredjeparts sporingslasting etter annonseklikk på DuckDuckGo: I utgangspunktet går vår beskyttelse mot tredjepartssporing langt utover hva du får i de fleste populære nettlesere, men vi jobber uavlatelig med å gjøre det mer omfattende. For øyeblikket er det slik at hvis en annonsør vil registrere konverteringer på sitt eget nettsted for våre private DuckDuckGo-søkeannonser, forhindrer ikke beskyttelsen mot tredjeparts sporingslasting bat.bing.com-forespørsler i å lastes på annonsørens nettsted etter DuckDuckGo-annonseklikk, men disse forespørslene stoppes i alle andre sammenhenger. Dette er fordi DuckDuckGos Private Search-annonsering skjer i samarbeid med Microsoft. Dette samarbeidet innebærer at du ser annonser anonymt på DuckDuckGo, og Microsoft har forpliktet seg til å ikke profilere brukernes annonseklikk.   +• Signaliser personvernreferansene dine med GPC (Global Privacy Control) – som er innebygd i appen og uttrykker automatisk reservasjonsretten din ved å gi beskjed til nettstedene om at de ikke kan selge eller dele personopplysningene dine. Om dette kan brukes til å håndheve dine juridiske rettigheter (for eksempel i henhold til nåværende eller fremtidige CCPA- og GDPR-krav), avhenger av lovene i din jurisdiksjon.     Les mer om dette og sporingsbeskyttelsene våre på https://help.duckduckgo.com/privacy/web-tracking-protections/   diff --git a/fastlane/metadata/pl/description.txt b/fastlane/metadata/pl/description.txt index 52b811bf02..e7f7073a49 100644 --- a/fastlane/metadata/pl/description.txt +++ b/fastlane/metadata/pl/description.txt @@ -20,7 +20,5 @@ CODZIENNA KONTROLA PRYWATNOŚCI • Zasygnalizuj swoje preferencje dotyczące prywatności za pomocą funkcji GPC (Global Privacy Control) - Więcej informacji znajdziesz na stronie https://help.duckduckgo.com/duckduckgo-help-pages/privacy/gpc/ -Informacja o naszym systemie ochrony przed wczytywaniem reklam przez strony trzecie po kliknięciu reklamy DuckDuckGo: nasza funkcja ochrony przed ładowaniem przez strony trzecie wykracza poza to, co jest domyślnie dostępne w większości popularnych przeglądarek, jednak nieustannie pracujemy nad tym, aby była ona jeszcze bardziej wszechstronna. Obecnie, jeśli reklamodawca chce wykryć konwersje na swojej stronie internetowej dla naszych prywatnych reklam wyszukiwania DuckDuckGo, nasza ochrona przed ładowaniem przez trackery stron trzecich nie blokuje ładowania zapytań bat.bing.com na stronie reklamodawcy po kliknięciu reklamy DuckDuckGo, ale zapytania te są blokowane we wszystkich innych kontekstach. Dzieje się tak dlatego, że reklama w wyszukiwarce prywatnej DuckDuckGo współpracuje z firmą Microsoft. W ramach tego partnerstwa oglądanie reklam w DuckDuckGo jest anonimowe, a firma Microsoft zobowiązała się nie profilować kliknięć reklam przez naszych użytkowników. - Więcej informacji na ten temat oraz na temat naszej bezpłatnej ochrony przed śledzeniem można znaleźć na stronie https://help.duckduckgo.com/privacy/web-tracking-protections diff --git a/fastlane/metadata/pt-PT/description.txt b/fastlane/metadata/pt-PT/description.txt index 880030ff87..6a03db6606 100644 --- a/fastlane/metadata/pt-PT/description.txt +++ b/fastlane/metadata/pt-PT/description.txt @@ -1,4 +1,3 @@ - A aplicação DuckDuckGo oferece-te a proteção de privacidade online mais abrangente com um toque num botão. Com uma transferência, obténs um novo navegador para o dia a dia que oferece proteção contínua enquanto pesquisas e navegas, e acesso a proteções contra rastreamento para os e-mails. Muitas dessas proteções não são oferecidas na maioria dos navegadores populares por predefinição. DESTAQUES DAS FUNCIONALIDADES  @@ -16,8 +15,7 @@ CONTROLOS DE PRIVACIDADE PARA O DIA A DIA • Toca no Fire Button, limpa os dados – Fecha os separadores e limpa os dados de navegação com o Fire Button. • Sinaliza a tua preferência de privacidade com o Global Privacy Control (GPC) – O GPC está incorporado na nossa aplicação e ajuda-te a expressares os teus direitos de exclusão automaticamente ao indicar aos sites que não vendam nem partilhem as tuas informações pessoais. Dependendo das leis da tua jurisdição, pode ser possível utilizá-lo para fazer valer os teus direitos legais (por exemplo, os requisitos atuais e futuros da CCPA e do RGPD).  - -Nota sobre a nossa 3rd-Party Tracker Loading Protection após cliques em anúncios da DuckDuckGo: a nossa 3rd-Party Tracker Loading Protection vai muito além do que a maioria dos navegadores populares oferecem por predefinição, mas estamos sempre a tentar melhorá-la. Atualmente, se um anunciante quiser detetar conversões no respetivo site para os nossos anúncios de pesquisa privados do DuckDuckGo, a 3rd-Party Tracker Loading Protection não impedirá o carregamento dos pedidos a stop.bat.bing.com no site do anunciante após cliques em anúncios do DuckDuckGo, mas esses pedidos serão travados em todos os outros contextos. Tal acontece porque a publicidade de pesquisa privada do DuckDuckGo é uma parceria com a Microsoft. No âmbito dessa parceria, a visualização de anúncios no DuckDuckGo é anónima e a Microsoft comprometeu-se a não criar perfis com os cliques em anúncios dos nossos utilizadores.    +    Obtém mais informações sobre estes anúncios e as nossas proteções contra o rastreamento em https://help.duckduckgo.com/privacy/web-tracking-protections/   Não precisa de esperar para recuperar a sua privacidade. Junte-se aos milhões de pessoas que usam o DuckDuckGo e proteja muitas das suas atividades online diárias com uma aplicação. É privacidade, simplificada.   \ No newline at end of file diff --git a/fastlane/metadata/ro/description.txt b/fastlane/metadata/ro/description.txt index c83a7e612e..14c6ebfd0f 100644 --- a/fastlane/metadata/ro/description.txt +++ b/fastlane/metadata/ro/description.txt @@ -20,8 +20,6 @@ CONTROALE DE CONFIDENȚIALITATE ZILNICE • Semnalează-ți preferințele în materie de confidențialitate cu Controlul global al confidențialității (GPC) Află mai multe la adresa https://help.duckduckgo.com/duckduckgo-help-pages/privacy/gpc/ -Notă despre protecția noastră împotriva încărcării tehnologiilor de urmărire terțe după ce ai făcut clic pe reclame DuckDuckGo: Protecția noastră împotriva tehnologiilor de urmărire terțe depășește cu mult ceea ce primești în mod implicit de la cele mai populare browsere și lucrăm constant pentru a o face chiar și mai cuprinzătoare. În prezent, dacă un agent de publicitate dorește să detecteze conversiile pe propriul site pentru reclamele noastre private la căutările pe DuckDuckGo, protecția noastră împotriva încărcării tehnologiilor de urmărire terțe nu va opri încărcarea solicitărilor bat.bing.com pe site-ul agenției de publicitate în urma clicurilor pe reclame DuckDuckGo, dar aceste solicitări sunt oprite în toate celelalte contexte. Acest lucru se datorează faptului că reclamele în cazul căutării private DuckDuckGo sunt oferite în parteneriat cu Microsoft. Ca parte a acestui parteneriat, vizualizarea anunțurilor pe DuckDuckGo este anonimă, iar Microsoft s-a angajat să nu profileze clicurile pe anunțuri ale utilizatorilor noștri. - Citește mai multe despre acest lucru și despre instrumentele noastre gratuite de protecție împotriva urmăririi pe https://help.duckduckgo.com/privacy/web-tracking-protections diff --git a/fastlane/metadata/ru/description.txt b/fastlane/metadata/ru/description.txt index 6e28acf372..7140d77c4b 100644 --- a/fastlane/metadata/ru/description.txt +++ b/fastlane/metadata/ru/description.txt @@ -1,5 +1,3 @@ - - Приложение DuckDuckGo обеспечивает наиболее комплексную защиту конфиденциальности онлайн одним нажатием кнопки. Всего одна загрузка — и ваш новый браузер будет незаметно защищать вас во время поиска и просмотра сайтов и даже блокировать отслеживание входящей почты. Защита конфиденциальности онлайн напоминает защиту своего дома. Если запереть только входную дверь, это не помешает настойчивым злоумышленникам проникнуть внутрь, особенно если вы оставили заднюю дверь и окна открытыми, а запасной ключ — под ковриком. Вот почему мы предлагаем несколько типов защиты конфиденциальности, многие из которых по умолчанию не доступны в большинстве популярных браузеров.   @@ -19,8 +17,6 @@ • Кнопка «Тревога» для удаления данных — теперь закрыть вкладки и очистить данные браузера можно нажатием специальной кнопки «Тревога». • Определите устраивающие вас параметры конфиденциальности с помощью Глобального контроля конфиденциальности (GPC) — в приложение уже встроен GPC, который автоматически выбирает отказ сайтам в праве продавать или передавать вашу персональную информацию. Возможность использования такого контроля для принудительного соблюдения ваших законных прав (согласно требований, например, CCPA и GDPR в действующей или последующей редакции) зависит от законодательства в месте вашего пребывания.  - -Защита от загрузки сторонних трекеров после нажатия рекламных объявлений в DuckDuckGo намного превосходит то, что предлагает большинство популярных браузеров по умолчанию. Тем не менее, мы постоянно работаем над расширением такой защиты. В настоящее время она не предотвращает загрузку запросов bat.bing.com на сайте рекламодателя после перехода на него по рекламному объявлению в DuckDuckGo, когда рекламодатель отслеживает конверсию на своем сайте. Однако такие запросы блокируются во всех других случаях. Это связано с тем, что реклама в поиске DuckDuckGo ведется на условиях партнерства с Microsoft. В рамках этого партнерства просмотр рекламы на DuckDuckGo является анонимным, так как Microsoft обязалась не профилировать наших пользователей при нажатии ими объявлений.    Читайте подробнее о средствах защиты от отслеживания на странице https://help.duckduckgo.com/privacy/web-tracking-protections/.   diff --git a/fastlane/metadata/sv/description.txt b/fastlane/metadata/sv/description.txt index 85c726e5dc..c17da96d4a 100644 --- a/fastlane/metadata/sv/description.txt +++ b/fastlane/metadata/sv/description.txt @@ -1,4 +1,3 @@ - DuckDuckGo-appen ger det mest omfattande integritetsskyddet online med ett enda knapptryck. Med en nedladdning får du en ny vardagswebbläsare som erbjuder sömlöst skydd medan du söker och surfar. Du får även tillgång till spårningsskydd för inkommande e-postmeddelanden. Många av dessa skydd erbjuds inte som standard i de populäraste webbläsarna. FUNKTIONER I FOKUS  @@ -17,9 +16,7 @@ FUNKTIONER I FOKUS  INTEGRITETSKONTROLLER VARJE DAG • Tryck på brännarknappen, bränn data – Rensa dina flikar och webbläsardata med vår brännarknapp. -• Ange dina integritetsinställningar med Global Privacy Control (GPC) – GPC är inbyggd i vår app och är till för att hjälpa dig att automatiskt utnyttja din rätt att välja genom att tala om för webbplatser att de inte får sälja eller dela din personliga information. Huruvida det kan användas för att genomdriva dina juridiska rättigheter (till exempel nuvarande eller framtida CCPA- eller GDPR-krav) beror på lagarna i din jurisdiktion.  - -Anmärkning gällande vårt tredjepartsskydd för inläsning av spårare efter DuckDuckGo-annonsklick: Vårt skydd för spårare från tredje part ger dig mer än vad du får som standard i de flesta populära webbläsare, men vi arbetar ständigt för att utöka det. Om en annonsör vill upptäcka konverteringar på sin egen webbplats för våra privata DuckDuckGo-sökannonser stoppar vårt skydd för laddning av spårare från tredje part inte bat.bing.com-förfrågningar från att läsas in på annonsörens webbplats efter DuckDuckGo-annonsklick, men dessa förfrågningar stoppas i alla andra sammanhang. Detta beror på att DuckDuckGo:s privata sökannonsering sker i samarbete med Microsoft. Enligt denna partnerskapsöverenskommelse är du anonym när du ser annonser på DuckDuckGo, och Microsoft har åtagit sig att inte profilera våra användares annonsklick.    +• Ange dina integritetsinställningar med Global Privacy Control (GPC) – GPC är inbyggd i vår app och är till för att hjälpa dig att automatiskt utnyttja din rätt att välja genom att tala om för webbplatser att de inte får sälja eller dela din personliga information. Huruvida det kan användas för att genomdriva dina juridiska rättigheter (till exempel nuvarande eller framtida CCPA- eller GDPR-krav) beror på lagarna i din jurisdiktion.   Läs mer om detta och vårt skydd mot spårning på https://help.duckduckgo.com/privacy/web-tracking-protections/  diff --git a/fastlane/metadata/tr/description.txt b/fastlane/metadata/tr/description.txt index 60b4d757f1..fa283ba4a0 100644 --- a/fastlane/metadata/tr/description.txt +++ b/fastlane/metadata/tr/description.txt @@ -1,4 +1,3 @@ - DuckDuckGo uygulaması, bir düğmeye basarak en kapsamlı çevrimiçi gizlilik korumasını sağlar. Tek bir indirme ile, arama yaparken ve tarayıcıda gezinirken sürekli bir koruma ve aldığınız e-postalar için izleme korumalarına erişim sağlar. Bu korumaların çoğu en popüler tarayıcılarda varsayılan olarak sunulmaz. ÖNE ÇIKAN ÖZELLİKLER  @@ -21,8 +20,6 @@ GÜNLÜK GİZLİLİK KONTROLLERİ • Küresel Gizlilik Kontrolü (GPC) ile Gizlilik Tercihinizi Belirtin - uygulamamızda yerleşik olarak bulunan GPC, web sitelerine kişisel bilgilerinizi satmamalarını veya paylaşmamalarını söyleyerek devre dışı bırakma haklarınızı otomatik olarak belirtmenize yardımcı olmayı amaçlamaktadır. Yasal haklarınızı uygulamak için kullanılıp kullanılamayacağı (örneğin, mevcut veya gelecekteki CCPA, GDPR gereklilikleri) yargı alanınızdaki yasalara bağlıdır.  -DuckDuckGo reklam tıklamalarını takip eden Üçüncü Taraf İzleyici Yükleme Korumamız hakkında bilgi: Üçüncü Taraf İzleyici Korumamız, varsayılan olarak en popüler tarayıcılara göre çok daha fazlasını yapar; ancak bunu daha kapsamlı hale getirmek için sürekli olarak çalışmaya devam ediyoruz. Şu anda, bir reklam veren özel DuckDuckGo arama reklamlarımız için kendi web sitesinde dönüşümleri tespit etmek isterse, Üçüncü Taraf İzleyici Yükleme Korumamız, DuckDuckGo reklam tıklamalarını takiben bat.bing.com isteklerinin reklam verenin web sitesinde yüklenmesini durdurmaz; ancak bu istekler diğer tüm koşullarda durdurulur. Bunun nedeni, DuckDuckGo özel arama reklamlarının Microsoft ile ortak olmasıdır. Bu ortaklığın bir parçası olarak, DuckDuckGo'da reklamlar anonim olarak görüntülenir ve Microsoft, kullanıcılarımızın reklam tıklamaları için profil oluşturmamayı taahhüt etmiştir.    - Bu konu ve İzlenme Korumalarımız hakkında daha fazla bilgi okumak için tıklayın: https://help.duckduckgo.com/privacy/web-tracking-protections/  Gizliliğinizi geri almak için beklemenize gerek yok. DuckDuckGo kullanan milyonlarca insana katılın ve tek bir uygulama ile internetteki günlük etkinliklerinizin çoğunu koruyun. Gizlilik, basitleştirildi.   \ No newline at end of file