From 9155824706903a81216e3dd61e5c27255e806682 Mon Sep 17 00:00:00 2001 From: Diego Rey Mendez Date: Thu, 7 Mar 2024 11:55:34 +0100 Subject: [PATCH 1/3] Adds pixels to track main VPN funnels (#2543) Task/Issue URL: https://app.asana.com/0/0/1206737255908394/f macOS PR: https://github.com/duckduckgo/macos-browser/pull/2304 BSK PR: https://github.com/duckduckgo/BrowserServicesKit/pull/698 ## Description Add pixels to better track all main VPN funnels: - VPN Controller Start - VPN Tunnel Start - VPN Tunnel Update (change location, etc) --- Core/Pixel.swift | 15 ++++++++- Core/PixelEvent.swift | 25 ++++++++++++++ DuckDuckGo.xcodeproj/project.pbxproj | 2 +- .../xcshareddata/swiftpm/Package.resolved | 4 +-- .../NetworkProtectionTunnelController.swift | 5 +++ ...etworkProtectionPacketTunnelProvider.swift | 33 +++++++++++++++++-- 6 files changed, 78 insertions(+), 6 deletions(-) diff --git a/Core/Pixel.swift b/Core/Pixel.swift index dd0a8e77a6..edf6d6611b 100644 --- a/Core/Pixel.swift +++ b/Core/Pixel.swift @@ -249,6 +249,16 @@ extension Pixel { } } +/// NSError supports this through `NSUnderlyingError`, but there's no support for this for Swift's `Error`. This protocol does that. +/// +/// The reason why this protocol returns a code and a domain instead of just an `Error` or `NSError` is so that the error implementing +/// this protocol has full control over these values, and is able to override them as it best sees fit. +/// +protocol ErrorWithUnderlyingError: Error { + var underlyingErrorCode: Int { get } + var underlyingErrorDomain: String { get } +} + extension Dictionary where Key == String, Value == String { mutating func appendErrorPixelParams(error: Error) { let nsError = error as NSError @@ -256,7 +266,10 @@ extension Dictionary where Key == String, Value == String { self[PixelParameters.errorCode] = "\(nsError.code)" self[PixelParameters.errorDomain] = nsError.domain - if let underlyingError = nsError.userInfo["NSUnderlyingError"] as? NSError { + if let underlyingError = error as? ErrorWithUnderlyingError { + self[PixelParameters.underlyingErrorCode] = "\(underlyingError.underlyingErrorCode)" + self[PixelParameters.underlyingErrorDomain] = underlyingError.underlyingErrorDomain + } else if let underlyingError = nsError.userInfo["NSUnderlyingError"] as? NSError { self[PixelParameters.underlyingErrorCode] = "\(underlyingError.code)" self[PixelParameters.underlyingErrorDomain] = underlyingError.domain } else if let sqlErrorCode = nsError.userInfo["NSSQLiteErrorDomain"] as? NSNumber { diff --git a/Core/PixelEvent.swift b/Core/PixelEvent.swift index 527ce9c495..8925cb5716 100644 --- a/Core/PixelEvent.swift +++ b/Core/PixelEvent.swift @@ -266,6 +266,18 @@ extension Pixel { case networkProtectionActiveUser case networkProtectionNewUser + case networkProtectionControllerStartAttempt + case networkProtectionControllerStartSuccess + case networkProtectionControllerStartFailure + + case networkProtectionTunnelStartAttempt + case networkProtectionTunnelStartSuccess + case networkProtectionTunnelStartFailure + + case networkProtectionTunnelUpdateAttempt + case networkProtectionTunnelUpdateSuccess + case networkProtectionTunnelUpdateFailure + case networkProtectionEnableAttemptConnecting case networkProtectionEnableAttemptSuccess case networkProtectionEnableAttemptFailure @@ -280,6 +292,8 @@ extension Pixel { case networkProtectionBreakageReport + case networkProtectionRekeyAttempt + case networkProtectionRekeyFailure case networkProtectionRekeyCompleted case networkProtectionTunnelConfigurationNoServerRegistrationInfo @@ -785,6 +799,15 @@ extension Pixel.Event { case .networkProtectionActiveUser: return "m_netp_daily_active_d" case .networkProtectionNewUser: return "m_netp_daily_active_u" + case .networkProtectionControllerStartAttempt: return "m_netp_controller_start_attempt" + case .networkProtectionControllerStartSuccess: return "m_netp_controller_start_success" + case .networkProtectionControllerStartFailure: return "m_netp_controller_start_failure" + case .networkProtectionTunnelStartAttempt: return "m_netp_tunnel_start_attempt" + case .networkProtectionTunnelStartSuccess: return "m_netp_tunnel_start_success" + case .networkProtectionTunnelStartFailure: return "m_netp_tunnel_start_failure" + case .networkProtectionTunnelUpdateAttempt: return "m_netp_tunnel_update_attempt" + case .networkProtectionTunnelUpdateSuccess: return "m_netp_tunnel_update_success" + case .networkProtectionTunnelUpdateFailure: return "m_netp_tunnel_update_failure" case .networkProtectionEnableAttemptConnecting: return "m_netp_ev_enable_attempt" case .networkProtectionEnableAttemptSuccess: return "m_netp_ev_enable_attempt_success" case .networkProtectionEnableAttemptFailure: return "m_netp_ev_enable_attempt_failure" @@ -792,7 +815,9 @@ extension Pixel.Event { case .networkProtectionTunnelFailureRecovered: return "m_netp_ev_tunnel_failure_recovered" case .networkProtectionLatency(let quality): return "m_netp_ev_\(quality.rawValue)_latency" case .networkProtectionLatencyError: return "m_netp_ev_latency_error_d" + case .networkProtectionRekeyAttempt: return "m_mac_netp_rekey_attempt" case .networkProtectionRekeyCompleted: return "m_netp_rekey_completed" + case .networkProtectionRekeyFailure: return "m_netp_rekey_failure" case .networkProtectionEnabledOnSearch: return "m_netp_ev_enabled_on_search" case .networkProtectionBreakageReport: return "m_vpn_breakage_report" case .networkProtectionTunnelConfigurationNoServerRegistrationInfo: return "m_netp_tunnel_config_error_no_server_registration_info" diff --git a/DuckDuckGo.xcodeproj/project.pbxproj b/DuckDuckGo.xcodeproj/project.pbxproj index 09fa88e566..00e2248435 100644 --- a/DuckDuckGo.xcodeproj/project.pbxproj +++ b/DuckDuckGo.xcodeproj/project.pbxproj @@ -9981,7 +9981,7 @@ repositoryURL = "https://github.com/DuckDuckGo/BrowserServicesKit"; requirement = { kind = exactVersion; - version = 114.1.0; + version = "114.1.0-1"; }; }; 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 018b6e092f..6d70031d01 100644 --- a/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved +++ b/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved @@ -23,8 +23,8 @@ "kind" : "remoteSourceControl", "location" : "https://github.com/DuckDuckGo/BrowserServicesKit", "state" : { - "revision" : "045a8782c3dbbf79fc088b38120dea1efadc13e1", - "version" : "114.1.0" + "revision" : "d9de416d3b77082f818a91b065bee1a2025c8e46", + "version" : "114.1.0-1" } }, { diff --git a/DuckDuckGo/NetworkProtectionTunnelController.swift b/DuckDuckGo/NetworkProtectionTunnelController.swift index 77a09a12e1..2ba3add399 100644 --- a/DuckDuckGo/NetworkProtectionTunnelController.swift +++ b/DuckDuckGo/NetworkProtectionTunnelController.swift @@ -49,9 +49,14 @@ final class NetworkProtectionTunnelController: TunnelController { /// Starts the VPN connection used for Network Protection /// func start() async { + Pixel.fire(pixel: .networkProtectionControllerStartAttempt) + do { try await startWithError() + Pixel.fire(pixel: .networkProtectionControllerStartSuccess) } catch { + Pixel.fire(pixel: .networkProtectionControllerStartFailure, error: error) + #if DEBUG errorStore.lastErrorMessage = error.localizedDescription #endif diff --git a/PacketTunnelProvider/NetworkProtection/NetworkProtectionPacketTunnelProvider.swift b/PacketTunnelProvider/NetworkProtection/NetworkProtectionPacketTunnelProvider.swift index f9e7fa92f7..feadafba5c 100644 --- a/PacketTunnelProvider/NetworkProtection/NetworkProtectionPacketTunnelProvider.swift +++ b/PacketTunnelProvider/NetworkProtection/NetworkProtectionPacketTunnelProvider.swift @@ -28,6 +28,8 @@ import NetworkExtension import NetworkProtection import Subscription +// swiftlint:disable type_body_length + // Initial implementation for initial Network Protection tests. Will be fleshed out with https://app.asana.com/0/1203137811378537/1204630829332227/f final class NetworkProtectionPacketTunnelProvider: PacketTunnelProvider { @@ -68,8 +70,33 @@ final class NetworkProtectionPacketTunnelProvider: PacketTunnelProvider { guard quality != .unknown else { return } DailyPixel.fireDailyAndCount(pixel: .networkProtectionLatency(quality: quality)) } - case .rekeyCompleted: - Pixel.fire(pixel: .networkProtectionRekeyCompleted) + case .rekeyAttempt(let step): + switch step { + case .begin: + DailyPixel.fireDailyAndCount(pixel: .networkProtectionRekeyAttempt) + case .failure(let error): + DailyPixel.fireDailyAndCount(pixel: .networkProtectionRekeyFailure, error: error) + case .success: + DailyPixel.fireDailyAndCount(pixel: .networkProtectionRekeyCompleted) + } + case .tunnelStartAttempt(let step): + switch step { + case .begin: + DailyPixel.fireDailyAndCount(pixel: .networkProtectionTunnelStartAttempt) + case .failure(let error): + DailyPixel.fireDailyAndCount(pixel: .networkProtectionTunnelStartFailure, error: error) + case .success: + DailyPixel.fireDailyAndCount(pixel: .networkProtectionTunnelStartSuccess) + } + case .tunnelUpdateAttempt(let step): + switch step { + case .begin: + DailyPixel.fireDailyAndCount(pixel: .networkProtectionTunnelUpdateAttempt) + case .failure(let error): + DailyPixel.fireDailyAndCount(pixel: .networkProtectionTunnelUpdateFailure, error: error) + case .success: + DailyPixel.fireDailyAndCount(pixel: .networkProtectionTunnelUpdateSuccess) + } } } @@ -287,4 +314,6 @@ final class NetworkProtectionPacketTunnelProvider: PacketTunnelProvider { } } +// swiftlint:enable type_body_length + #endif From 6f79d9638d35552e177723fb32cadf190f9a7cc4 Mon Sep 17 00:00:00 2001 From: Sam Symons Date: Thu, 7 Mar 2024 13:56:58 -0800 Subject: [PATCH 2/3] Prevent redeeming invite codes after the VPN test ends (#2559) Task/Issue URL: https://app.asana.com/0/414235014887631/1206789264380520/f Tech Design URL: CC: Description: This is the iOS equivalent of duckduckgo/macos-browser#2347. --- DuckDuckGo/AppDelegate+Waitlists.swift | 12 +++++++++--- 1 file changed, 9 insertions(+), 3 deletions(-) diff --git a/DuckDuckGo/AppDelegate+Waitlists.swift b/DuckDuckGo/AppDelegate+Waitlists.swift index db54ce6826..7fae089578 100644 --- a/DuckDuckGo/AppDelegate+Waitlists.swift +++ b/DuckDuckGo/AppDelegate+Waitlists.swift @@ -22,6 +22,7 @@ import Core import BackgroundTasks import NetworkProtection import Waitlist +import BrowserServicesKit extension AppDelegate { @@ -51,12 +52,17 @@ extension AppDelegate { VPNWaitlist.shared.fetchInviteCodeIfAvailable { [weak self] error in guard error == nil else { -#if !DEBUG + if error == .alreadyHasInviteCode, UIApplication.shared.applicationState == .active { // If the user already has an invite code but their auth token has gone missing, attempt to redeem it again. let tokenStore = NetworkProtectionKeychainTokenStore() let waitlistStorage = VPNWaitlist.shared.waitlistStorage - if let inviteCode = waitlistStorage.getWaitlistInviteCode(), !tokenStore.isFeatureActivated { + let configManager = ContentBlocking.shared.privacyConfigurationManager + let waitlistBetaActive = configManager.privacyConfig.isSubfeatureEnabled(NetworkProtectionSubfeature.waitlistBetaActive) + + if let inviteCode = waitlistStorage.getWaitlistInviteCode(), + !tokenStore.isFeatureActivated, + waitlistBetaActive { let pixel: Pixel.Event = .networkProtectionWaitlistRetriedInviteCodeRedemption do { @@ -72,7 +78,7 @@ extension AppDelegate { self?.fetchVPNWaitlistAuthToken(inviteCode: inviteCode) } } -#endif + return } From 066711a7e722dda17aab3ea35e1c0524dd3a78fc Mon Sep 17 00:00:00 2001 From: Juan Manuel Pereira Date: Thu, 7 Mar 2024 18:58:59 -0300 Subject: [PATCH 3/3] Point to BSK hotfix (#2558) Co-authored-by: Diego Rey Mendez --- DuckDuckGo.xcodeproj/project.pbxproj | 2 +- .../xcshareddata/swiftpm/Package.resolved | 8 ++++---- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/DuckDuckGo.xcodeproj/project.pbxproj b/DuckDuckGo.xcodeproj/project.pbxproj index 00e2248435..126d8db9f4 100644 --- a/DuckDuckGo.xcodeproj/project.pbxproj +++ b/DuckDuckGo.xcodeproj/project.pbxproj @@ -9981,7 +9981,7 @@ repositoryURL = "https://github.com/DuckDuckGo/BrowserServicesKit"; requirement = { kind = exactVersion; - version = "114.1.0-1"; + version = "114.1.0-2"; }; }; 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 6d70031d01..e5049fd873 100644 --- a/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved +++ b/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved @@ -23,8 +23,8 @@ "kind" : "remoteSourceControl", "location" : "https://github.com/DuckDuckGo/BrowserServicesKit", "state" : { - "revision" : "d9de416d3b77082f818a91b065bee1a2025c8e46", - "version" : "114.1.0-1" + "revision" : "b8763fdca7848d110b92e1a719e0ba84c2e7b114", + "version" : "114.1.0-2" } }, { @@ -41,8 +41,8 @@ "kind" : "remoteSourceControl", "location" : "https://github.com/duckduckgo/content-scope-scripts", "state" : { - "revision" : "a3690b7666a3617693383d948cb492513f6aa569", - "version" : "5.0.0" + "revision" : "f6241631fc14cc2d0f47950bfdc4d6c30bf90130", + "version" : "5.4.0" } }, {