diff --git a/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved b/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved index 6f25eeb8b6..daf82f57e6 100644 --- a/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved +++ b/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved @@ -40,8 +40,8 @@ "kind" : "remoteSourceControl", "location" : "https://github.com/duckduckgo/content-scope-scripts", "state" : { - "revision" : "f6241631fc14cc2d0f47950bfdc4d6c30bf90130", - "version" : "5.4.0" + "revision" : "edd96481d49b094c260f9a79e078abdbdd3a83fb", + "version" : "5.6.0" } }, { @@ -164,7 +164,7 @@ { "identity" : "trackerradarkit", "kind" : "remoteSourceControl", - "location" : "https://github.com/duckduckgo/TrackerRadarKit", + "location" : "https://github.com/duckduckgo/TrackerRadarKit.git", "state" : { "revision" : "a6b7ba151d9dc6684484f3785293875ec01cc1ff", "version" : "1.2.2" diff --git a/DuckDuckGo/Application/AppDelegate.swift b/DuckDuckGo/Application/AppDelegate.swift index 1afd63ec5d..fdaaca695a 100644 --- a/DuckDuckGo/Application/AppDelegate.swift +++ b/DuckDuckGo/Application/AppDelegate.swift @@ -261,19 +261,10 @@ final class AppDelegate: NSObject, NSApplicationDelegate, FileDownloadManagerDel Task { let accountManager = AccountManager() - do { - try accountManager.migrateAccessTokenToNewStore() - } catch { - if let error = error as? AccountManager.MigrationError { - switch error { - case AccountManager.MigrationError.migrationFailed: - os_log(.default, log: .subscription, "Access token migration failed") - case AccountManager.MigrationError.noMigrationNeeded: - os_log(.default, log: .subscription, "No access token migration needed") - } - } + if let token = accountManager.accessToken { + _ = await SubscriptionService.getSubscription(accessToken: token, cachePolicy: .reloadIgnoringLocalCacheData) + _ = await accountManager.fetchEntitlements(cachePolicy: .reloadIgnoringLocalCacheData) } - await accountManager.checkSubscriptionState() } #endif diff --git a/DuckDuckGo/Autofill/ContentOverlayViewController.swift b/DuckDuckGo/Autofill/ContentOverlayViewController.swift index bbc73f1f3a..d90b35f71d 100644 --- a/DuckDuckGo/Autofill/ContentOverlayViewController.swift +++ b/DuckDuckGo/Autofill/ContentOverlayViewController.swift @@ -253,6 +253,7 @@ extension ContentOverlayViewController: SecureVaultManagerDelegate { promptUserToAutofillCredentialsForDomain domain: String, withAccounts accounts: [SecureVaultModels.WebsiteAccount], withTrigger trigger: AutofillUserScript.GetTriggerType, + onAccountSelected account: @escaping (SecureVaultModels.WebsiteAccount?) -> Void, completionHandler: @escaping (SecureVaultModels.WebsiteAccount?) -> Void) { // no-op on macOS } diff --git a/DuckDuckGo/LoginItems/LoginItemsManager.swift b/DuckDuckGo/LoginItems/LoginItemsManager.swift index 8fef5f953b..e0a44a4fb4 100644 --- a/DuckDuckGo/LoginItems/LoginItemsManager.swift +++ b/DuckDuckGo/LoginItems/LoginItemsManager.swift @@ -59,6 +59,12 @@ final class LoginItemsManager { } } + func isAnyEnabled(_ items: Set) -> Bool { + return items.contains(where: { item in + item.status == .enabled + }) + } + private func handleError(for item: LoginItem, action: Action, error: NSError) { let event = Pixel.Event.Debug.loginItemUpdateError( loginItemBundleID: item.agentBundleID, diff --git a/DuckDuckGo/NetworkProtection/AppTargets/BothAppTargets/NetworkProtectionAppEvents.swift b/DuckDuckGo/NetworkProtection/AppTargets/BothAppTargets/NetworkProtectionAppEvents.swift index 2d240b6c0d..e639721f17 100644 --- a/DuckDuckGo/NetworkProtection/AppTargets/BothAppTargets/NetworkProtectionAppEvents.swift +++ b/DuckDuckGo/NetworkProtection/AppTargets/BothAppTargets/NetworkProtectionAppEvents.swift @@ -85,19 +85,15 @@ final class NetworkProtectionAppEvents { } private func restartNetworkProtectionIfVersionChanged(using loginItemsManager: LoginItemsManager) { - let versionStore = NetworkProtectionLastVersionRunStore() - - // should‘ve been run at least once with NetP enabled - guard versionStore.lastVersionRun != nil else { - os_log(.info, log: .networkProtection, "No last version found for the NetP login items, skipping update") - return - } - // We want to restart the VPN menu app to make sure it's always on the latest. restartNetworkProtectionMenu(using: loginItemsManager) } private func restartNetworkProtectionMenu(using loginItemsManager: LoginItemsManager) { + guard loginItemsManager.isAnyEnabled(LoginItemsManager.networkProtectionLoginItems) else { + return + } + loginItemsManager.restartLoginItems(LoginItemsManager.networkProtectionLoginItems, log: .networkProtection) } diff --git a/DuckDuckGo/NetworkProtection/AppTargets/BothAppTargets/NetworkProtectionNavBarButtonModel.swift b/DuckDuckGo/NetworkProtection/AppTargets/BothAppTargets/NetworkProtectionNavBarButtonModel.swift index 057f5b622c..b31c8f007f 100644 --- a/DuckDuckGo/NetworkProtection/AppTargets/BothAppTargets/NetworkProtectionNavBarButtonModel.swift +++ b/DuckDuckGo/NetworkProtection/AppTargets/BothAppTargets/NetworkProtectionNavBarButtonModel.swift @@ -100,9 +100,11 @@ final class NetworkProtectionNavBarButtonModel: NSObject, ObservableObject { } private func setupIconSubscription() { - iconPublisherCancellable = iconPublisher.$icon.sink { [weak self] icon in - self?.buttonImage = self?.buttonImageFromWaitlistState(icon: icon) - } + iconPublisherCancellable = iconPublisher.$icon + .receive(on: DispatchQueue.main) + .sink { [weak self] icon in + self?.buttonImage = self?.buttonImageFromWaitlistState(icon: icon) + } } /// Temporary override used for the NetP waitlist beta, as a different asset is used for users who are invited to join the beta but haven't yet accepted. diff --git a/DuckDuckGo/PrivacyDashboard/View/PrivacyDashboardViewController.swift b/DuckDuckGo/PrivacyDashboard/View/PrivacyDashboardViewController.swift index 6074ac2343..4a5d5113f3 100644 --- a/DuckDuckGo/PrivacyDashboard/View/PrivacyDashboardViewController.swift +++ b/DuckDuckGo/PrivacyDashboard/View/PrivacyDashboardViewController.swift @@ -258,11 +258,13 @@ extension PrivacyDashboardViewController: PrivacyDashboardReportBrokenSiteDelega didRequestSubmitBrokenSiteReportWithCategory category: String, description: String) { let source: BrokenSiteReport.Source = privacyDashboardController.initDashboardMode == .report ? .appMenu : .dashboard - do { - let report = try makeBrokenSiteReport(category: category, description: description, source: source) - try brokenSiteReporter.report(report, reportMode: .regular) - } catch { - os_log("Failed to generate or send the broken site report: \(error.localizedDescription)", type: .error) + Task { @MainActor in + do { + let report = try await makeBrokenSiteReport(category: category, description: description, source: source) + try brokenSiteReporter.report(report, reportMode: .regular) + } catch { + os_log("Failed to generate or send the broken site report: \(error.localizedDescription)", type: .error) + } } } @@ -281,13 +283,15 @@ extension PrivacyDashboardViewController: PrivacyDashboardToggleReportDelegate { didRequestSubmitToggleReportWithSource source: BrokenSiteReport.Source, didOpenReportInfo: Bool, toggleReportCounter: Int?) { - do { - let report = try makeBrokenSiteReport(source: source, - didOpenReportInfo: didOpenReportInfo, - toggleReportCounter: toggleReportCounter) - try toggleProtectionsOffReporter.report(report, reportMode: .toggle) - } catch { - os_log("Failed to generate or send the broken site report: %@", type: .error, error.localizedDescription) + Task { @MainActor in + do { + let report = try await makeBrokenSiteReport(source: source, + didOpenReportInfo: didOpenReportInfo, + toggleReportCounter: toggleReportCounter) + try toggleProtectionsOffReporter.report(report, reportMode: .toggle) + } catch { + os_log("Failed to generate or send the broken site report: %@", type: .error, error.localizedDescription) + } } } @@ -301,11 +305,25 @@ extension PrivacyDashboardViewController { case failedToFetchTheCurrentURL } + private func calculateWebVitals(performanceMetrics: PerformanceMetricsSubfeature?, privacyConfig: PrivacyConfiguration) async -> [Double]? { + var webVitalsResult: [Double]? + if privacyConfig.isEnabled(featureKey: .performanceMetrics) { + webVitalsResult = await withCheckedContinuation({ continuation in + guard let performanceMetrics else { continuation.resume(returning: nil); return } + performanceMetrics.notifyHandler { result in + continuation.resume(returning: result) + } + }) + } + + return webVitalsResult + } + private func makeBrokenSiteReport(category: String = "", description: String = "", source: BrokenSiteReport.Source, didOpenReportInfo: Bool = false, - toggleReportCounter: Int? = nil) throws -> BrokenSiteReport { + toggleReportCounter: Int? = nil) async throws -> BrokenSiteReport { // ⚠️ To limit privacy risk, site URL is trimmed to not include query and fragment guard let currentTab = tabViewModel?.tab, @@ -321,12 +339,14 @@ extension PrivacyDashboardViewController { let configuration = ContentBlocking.shared.privacyConfigurationManager.privacyConfig let protectionsState = configuration.isFeature(.contentBlocking, enabledForDomain: currentTab.content.url?.host) + let webVitals = await calculateWebVitals(performanceMetrics: currentTab.performanceMetrics, privacyConfig: configuration) + var errors: [Error]? var statusCodes: [Int]? - if let error = tabViewModel?.tab.lastWebError { + if let error = currentTab.lastWebError { errors = [error] } - if let httpStatusCode = tabViewModel?.tab.lastHttpStatusCode { + if let httpStatusCode = currentTab.lastHttpStatusCode { statusCodes = [httpStatusCode] } @@ -346,6 +366,10 @@ extension PrivacyDashboardViewController { reportFlow: source, errors: errors, httpStatusCodes: statusCodes, + openerContext: currentTab.inferredOpenerContext, + vpnOn: currentTab.tunnelController?.isConnected ?? false, + jsPerformance: webVitals, + userRefreshCount: currentTab.refreshCountSinceLoad, didOpenReportInfo: didOpenReportInfo, toggleReportCounter: toggleReportCounter) return websiteBreakage diff --git a/DuckDuckGo/Tab/Model/Tab.swift b/DuckDuckGo/Tab/Model/Tab.swift index accb86d699..ad05b12141 100644 --- a/DuckDuckGo/Tab/Model/Tab.swift +++ b/DuckDuckGo/Tab/Model/Tab.swift @@ -25,6 +25,7 @@ import Navigation import UserScript import WebKit import History +import PrivacyDashboard #if SUBSCRIPTION import Subscription @@ -344,7 +345,7 @@ protocol NewWindowPolicyDecisionMaker { let pinnedTabsManager: PinnedTabsManager #if NETWORK_PROTECTION - private var tunnelController: NetworkProtectionIPCTunnelController? + private(set) var tunnelController: NetworkProtectionIPCTunnelController? #endif private let webViewConfiguration: WKWebViewConfiguration @@ -779,6 +780,10 @@ protocol NewWindowPolicyDecisionMaker { @Published private(set) var lastWebError: Error? @Published private(set) var lastHttpStatusCode: Int? + @Published private(set) var inferredOpenerContext: BrokenSiteReport.OpenerContext? + @Published private(set) var refreshCountSinceLoad: Int = 0 + private (set) var performanceMetrics: PerformanceMetricsSubfeature? + @Published private(set) var isLoading: Bool = false @Published private(set) var loadingProgress: Double = 0.0 @@ -1028,6 +1033,8 @@ protocol NewWindowPolicyDecisionMaker { return nil } + refreshCountSinceLoad += 1 + self.content = content.forceReload() if webView.url == nil, content.isUrl { // load from cache or interactionStateData when called by lazy loader @@ -1287,6 +1294,9 @@ extension Tab: UserContentControllerDelegate { userScripts.faviconScript.delegate = self userScripts.pageObserverScript.delegate = self userScripts.printingUserScript.delegate = self + + performanceMetrics = PerformanceMetricsSubfeature(targetWebview: webView) + userScripts.contentScopeUserScriptIsolated.registerSubfeature(delegate: performanceMetrics!) } } @@ -1357,11 +1367,32 @@ extension Tab/*: NavigationResponder*/ { // to be moved to Tab+Navigation.swift committedURL = navigation.url } + func resetRefreshCountIfNeeded(action: NavigationAction) { + switch action.navigationType { + case .reload, .other: + break + default: + refreshCountSinceLoad = 0 + } + } + + func setOpenerContextIfNeeded(action: NavigationAction) { + switch action.navigationType { + case .linkActivated, .formSubmitted: + inferredOpenerContext = .navigation + default: + break + } + } + @MainActor func decidePolicy(for navigationAction: NavigationAction, preferences: inout NavigationPreferences) async -> NavigationActionPolicy? { // allow local file navigations if navigationAction.url.isFileURL { return .allow } + resetRefreshCountIfNeeded(action: navigationAction) + setOpenerContextIfNeeded(action: navigationAction) + // when navigating to a URL with basic auth username/password, cache it and redirect to a trimmed URL if let mainFrame = navigationAction.mainFrameTarget, let credential = navigationAction.url.basicAuthCredential { @@ -1416,6 +1447,10 @@ extension Tab/*: NavigationResponder*/ { // to be moved to Tab+Navigation.swift error = nil } + if inferredOpenerContext != .external { + inferredOpenerContext = nil + } + invalidateInteractionStateData() } @@ -1425,6 +1460,12 @@ extension Tab/*: NavigationResponder*/ { // to be moved to Tab+Navigation.swift webViewDidFinishNavigationPublisher.send() statisticsLoader?.refreshRetentionAtb(isSearch: navigation.url.isDuckDuckGoSearch) + Task { @MainActor in + if await webView.isCurrentSiteReferredFromDuckDuckGo { + inferredOpenerContext = .serp + } + } + #if NETWORK_PROTECTION if navigation.url.isDuckDuckGoSearch, tunnelController?.isConnected == true { DailyPixel.fire(pixel: .networkProtectionEnabledOnSearch, frequency: .dailyAndCount) diff --git a/DuckDuckGo/Tab/TabExtensions/AutofillTabExtension.swift b/DuckDuckGo/Tab/TabExtensions/AutofillTabExtension.swift index 74974a747d..5a973d76cc 100644 --- a/DuckDuckGo/Tab/TabExtensions/AutofillTabExtension.swift +++ b/DuckDuckGo/Tab/TabExtensions/AutofillTabExtension.swift @@ -109,6 +109,7 @@ extension AutofillTabExtension: SecureVaultManagerDelegate { promptUserToAutofillCredentialsForDomain domain: String, withAccounts accounts: [SecureVaultModels.WebsiteAccount], withTrigger trigger: AutofillUserScript.GetTriggerType, + onAccountSelected account: @escaping (SecureVaultModels.WebsiteAccount?) -> Void, completionHandler: @escaping (SecureVaultModels.WebsiteAccount?) -> Void) { // no-op on macOS } diff --git a/DuckDuckGo/Tab/View/BrowserTabViewController.swift b/DuckDuckGo/Tab/View/BrowserTabViewController.swift index eccbde153a..ad103788cb 100644 --- a/DuckDuckGo/Tab/View/BrowserTabViewController.swift +++ b/DuckDuckGo/Tab/View/BrowserTabViewController.swift @@ -199,7 +199,9 @@ final class BrowserTabViewController: NSViewController { #if DBP @objc private func onDBPFeatureDisabled(_ notification: Notification) { - tabCollectionViewModel.removeAll(with: .dataBrokerProtection) + Task { @MainActor in + tabCollectionViewModel.removeAll(with: .dataBrokerProtection) + } } @objc diff --git a/DuckDuckGo/VPNFeedbackForm/VPNMetadataCollector.swift b/DuckDuckGo/VPNFeedbackForm/VPNMetadataCollector.swift index 58849d4211..e1c071de4e 100644 --- a/DuckDuckGo/VPNFeedbackForm/VPNMetadataCollector.swift +++ b/DuckDuckGo/VPNFeedbackForm/VPNMetadataCollector.swift @@ -31,7 +31,8 @@ struct VPNMetadata: Encodable { struct AppInfo: Encodable { let appVersion: String - let lastVersionRun: String + let lastAgentVersionRun: String + let lastExtensionVersionRun: String let isInternalUser: Bool let isInApplicationsDirectory: Bool } @@ -154,13 +155,14 @@ final class DefaultVPNMetadataCollector: VPNMetadataCollector { private func collectAppInfoMetadata() -> VPNMetadata.AppInfo { let appVersion = AppVersion.shared.versionAndBuildNumber - let versionStore = NetworkProtectionLastVersionRunStore() + let versionStore = NetworkProtectionLastVersionRunStore(userDefaults: .netP) let isInternalUser = NSApp.delegateTyped.internalUserDecider.isInternalUser let isInApplicationsDirectory = Bundle.main.isInApplicationsDirectory return .init( appVersion: appVersion, - lastVersionRun: versionStore.lastVersionRun ?? "Unknown", + lastAgentVersionRun: versionStore.lastAgentVersionRun ?? "none", + lastExtensionVersionRun: versionStore.lastExtensionVersionRun ?? "none", isInternalUser: isInternalUser, isInApplicationsDirectory: isInApplicationsDirectory ) diff --git a/DuckDuckGoVPN/NetworkExtensionController.swift b/DuckDuckGoVPN/NetworkExtensionController.swift index 49960fdb8a..d850bfbb80 100644 --- a/DuckDuckGoVPN/NetworkExtensionController.swift +++ b/DuckDuckGoVPN/NetworkExtensionController.swift @@ -17,6 +17,7 @@ // import Foundation +import NetworkProtection import NetworkProtectionUI #if NETP_SYSTEM_EXTENSION @@ -32,11 +33,13 @@ final class NetworkExtensionController { #if NETP_SYSTEM_EXTENSION private let systemExtensionManager: SystemExtensionManager + private let defaults: UserDefaults #endif - init(extensionBundleID: String) { + init(extensionBundleID: String, defaults: UserDefaults = .netP) { #if NETP_SYSTEM_EXTENSION systemExtensionManager = SystemExtensionManager(extensionBundleID: extensionBundleID) + self.defaults = defaults #endif } @@ -46,9 +49,11 @@ extension NetworkExtensionController { func activateSystemExtension(waitingForUserApproval: @escaping () -> Void) async throws { #if NETP_SYSTEM_EXTENSION - try await systemExtensionManager.activate( + let extensionVersion = try await systemExtensionManager.activate( waitingForUserApproval: waitingForUserApproval) + NetworkProtectionLastVersionRunStore(userDefaults: defaults).lastExtensionVersionRun = extensionVersion + try? await Task.sleep(nanoseconds: 300 * NSEC_PER_MSEC) #endif } diff --git a/DuckDuckGoVPN/VPNAppEventsHandler.swift b/DuckDuckGoVPN/VPNAppEventsHandler.swift index 0dc1b0364f..5ac88e4a2b 100644 --- a/DuckDuckGoVPN/VPNAppEventsHandler.swift +++ b/DuckDuckGoVPN/VPNAppEventsHandler.swift @@ -30,20 +30,20 @@ final class VPNAppEventsHandler { func appDidFinishLaunching() { let currentVersion = AppVersion.shared.versionAndBuildNumber - let versionStore = NetworkProtectionLastVersionRunStore() + let versionStore = NetworkProtectionLastVersionRunStore(userDefaults: .netP) defer { - versionStore.lastVersionRun = currentVersion + versionStore.lastAgentVersionRun = currentVersion } let restartTunnel = { - os_log(.info, log: .networkProtection, "App updated from %{public}s to %{public}s: updating login items", versionStore.lastVersionRun ?? "null", currentVersion) + os_log(.info, log: .networkProtection, "App updated from %{public}s to %{public}s: updating login items", versionStore.lastAgentVersionRun ?? "null", currentVersion) self.restartTunnel() } #if DEBUG || REVIEW // Since DEBUG and REVIEW builds may not change version No. we want them to always reset. restartTunnel() #else - if versionStore.lastVersionRun != currentVersion { + if versionStore.lastAgentVersionRun != currentVersion { restartTunnel() } #endif diff --git a/LocalPackages/DataBrokerProtection/Package.swift b/LocalPackages/DataBrokerProtection/Package.swift index af3abd24c7..d604410732 100644 --- a/LocalPackages/DataBrokerProtection/Package.swift +++ b/LocalPackages/DataBrokerProtection/Package.swift @@ -29,7 +29,7 @@ let package = Package( targets: ["DataBrokerProtection"]) ], dependencies: [ - .package(url: "https://github.com/duckduckgo/BrowserServicesKit", exact: "127.0.0"), + .package(url: "https://github.com/duckduckgo/BrowserServicesKit", exact: "129.0.0"), .package(path: "../PixelKit"), .package(path: "../SwiftUIExtensions"), .package(path: "../XPCHelper"), diff --git a/LocalPackages/NetworkProtectionMac/Package.swift b/LocalPackages/NetworkProtectionMac/Package.swift index 9c95668e6d..6cca70cf1d 100644 --- a/LocalPackages/NetworkProtectionMac/Package.swift +++ b/LocalPackages/NetworkProtectionMac/Package.swift @@ -31,7 +31,7 @@ let package = Package( .library(name: "NetworkProtectionUI", targets: ["NetworkProtectionUI"]) ], dependencies: [ - .package(url: "https://github.com/duckduckgo/BrowserServicesKit", exact: "127.0.0"), + .package(url: "https://github.com/duckduckgo/BrowserServicesKit", exact: "129.0.0"), .package(path: "../XPCHelper"), .package(path: "../SwiftUIExtensions"), .package(path: "../LoginItems"), diff --git a/LocalPackages/SubscriptionUI/Package.swift b/LocalPackages/SubscriptionUI/Package.swift index ccf7c0509a..fea582b151 100644 --- a/LocalPackages/SubscriptionUI/Package.swift +++ b/LocalPackages/SubscriptionUI/Package.swift @@ -12,7 +12,7 @@ let package = Package( targets: ["SubscriptionUI"]), ], dependencies: [ - .package(url: "https://github.com/duckduckgo/BrowserServicesKit", exact: "127.0.0"), + .package(url: "https://github.com/duckduckgo/BrowserServicesKit", exact: "129.0.0"), .package(path: "../SwiftUIExtensions") ], targets: [ diff --git a/LocalPackages/SubscriptionUI/Sources/SubscriptionUI/Preferences/PreferencesSubscriptionModel.swift b/LocalPackages/SubscriptionUI/Sources/SubscriptionUI/Preferences/PreferencesSubscriptionModel.swift index c2cd52eba6..ff2628ae29 100644 --- a/LocalPackages/SubscriptionUI/Sources/SubscriptionUI/Preferences/PreferencesSubscriptionModel.swift +++ b/LocalPackages/SubscriptionUI/Sources/SubscriptionUI/Preferences/PreferencesSubscriptionModel.swift @@ -95,35 +95,10 @@ public final class PreferencesSubscriptionModel: ObservableObject { self.isUserAuthenticated = accountManager.isUserAuthenticated - if let token = accountManager.accessToken { + if accountManager.isUserAuthenticated { Task { - let subscriptionResult = await SubscriptionService.getSubscription(accessToken: token, cachePolicy: .returnCacheDataElseLoad) - if case .success(let subscription) = subscriptionResult { - self.updateDescription(for: subscription.expiresOrRenewsAt, status: subscription.status, period: subscription.billingPeriod) - self.subscriptionPlatform = subscription.platform - self.subscriptionStatus = subscription.status - } - - switch await self.accountManager.hasEntitlement(for: .networkProtection, cachePolicy: .returnCacheDataElseLoad) { - case let .success(result): - self.hasAccessToVPN = result - case .failure: - self.hasAccessToVPN = false - } - - switch await self.accountManager.hasEntitlement(for: .dataBrokerProtection, cachePolicy: .returnCacheDataElseLoad) { - case let .success(result): - self.hasAccessToDBP = result - case .failure: - self.hasAccessToDBP = false - } - - switch await self.accountManager.hasEntitlement(for: .identityTheftRestoration, cachePolicy: .returnCacheDataElseLoad) { - case let .success(result): - self.hasAccessToITR = result - case .failure: - self.hasAccessToITR = false - } + await self.updateSubscription(with: .returnCacheDataElseLoad) + await self.updateAllEntitlement(with: .returnCacheDataElseLoad) } } @@ -264,6 +239,8 @@ public final class PreferencesSubscriptionModel: ObservableObject { @MainActor func fetchAndUpdateSubscriptionDetails() { + self.isUserAuthenticated = accountManager.isUserAuthenticated + guard fetchSubscriptionDetailsTask == nil else { return } fetchSubscriptionDetailsTask = Task { [weak self] in @@ -271,43 +248,53 @@ public final class PreferencesSubscriptionModel: ObservableObject { self?.fetchSubscriptionDetailsTask = nil } - guard let token = self?.accountManager.accessToken else { return } + await self?.updateSubscription(with: .reloadIgnoringLocalCacheData) + await self?.updateAllEntitlement(with: .reloadIgnoringLocalCacheData) + } + } - let subscriptionResult = await SubscriptionService.getSubscription(accessToken: token, cachePolicy: .reloadIgnoringLocalCacheData) + @MainActor + private func updateSubscription(with cachePolicy: SubscriptionService.CachePolicy) async { + guard let token = accountManager.accessToken else { + SubscriptionService.signOut() + return + } - if case .success(let subscription) = subscriptionResult { - self?.updateDescription(for: subscription.expiresOrRenewsAt, status: subscription.status, period: subscription.billingPeriod) - self?.subscriptionPlatform = subscription.platform - self?.subscriptionStatus = subscription.status - } else { - self?.accountManager.signOut() - } + switch await SubscriptionService.getSubscription(accessToken: token, cachePolicy: cachePolicy) { + case .success(let subscription): + updateDescription(for: subscription.expiresOrRenewsAt, status: subscription.status, period: subscription.billingPeriod) + subscriptionPlatform = subscription.platform + subscriptionStatus = subscription.status + case .failure: + break + } + } - if let self { - switch await self.accountManager.hasEntitlement(for: .networkProtection) { - case let .success(result): - hasAccessToVPN = result - case .failure: - hasAccessToVPN = false - } + @MainActor + private func updateAllEntitlement(with cachePolicy: AccountManager.CachePolicy) async { + switch await self.accountManager.hasEntitlement(for: .networkProtection, cachePolicy: cachePolicy) { + case let .success(result): + hasAccessToVPN = result + case .failure: + hasAccessToVPN = false + } - switch await self.accountManager.hasEntitlement(for: .dataBrokerProtection) { - case let .success(result): - hasAccessToDBP = result - case .failure: - hasAccessToDBP = false - } + switch await self.accountManager.hasEntitlement(for: .dataBrokerProtection, cachePolicy: cachePolicy) { + case let .success(result): + hasAccessToDBP = result + case .failure: + hasAccessToDBP = false + } - switch await self.accountManager.hasEntitlement(for: .identityTheftRestoration) { - case let .success(result): - hasAccessToITR = result - case .failure: - hasAccessToITR = false - } - } + switch await self.accountManager.hasEntitlement(for: .identityTheftRestoration, cachePolicy: cachePolicy) { + case let .success(result): + hasAccessToITR = result + case .failure: + hasAccessToITR = false } } + @MainActor func updateDescription(for date: Date, status: Subscription.Status, period: Subscription.BillingPeriod) { let formattedDate = dateFormatter.string(from: date) diff --git a/LocalPackages/SubscriptionUI/Sources/SubscriptionUI/Preferences/PreferencesSubscriptionView.swift b/LocalPackages/SubscriptionUI/Sources/SubscriptionUI/Preferences/PreferencesSubscriptionView.swift index 5424f3920b..52a723b011 100644 --- a/LocalPackages/SubscriptionUI/Sources/SubscriptionUI/Preferences/PreferencesSubscriptionView.swift +++ b/LocalPackages/SubscriptionUI/Sources/SubscriptionUI/Preferences/PreferencesSubscriptionView.swift @@ -185,8 +185,9 @@ public struct PreferencesSubscriptionView: View { TextMenuItemHeader(model.subscriptionDetails ?? UserText.preferencesSubscriptionInactiveHeader) TextMenuItemCaption(UserText.preferencesSubscriptionExpiredCaption) } buttons: { - Button(UserText.viewPlansButtonTitle) { model.purchaseAction() } - .buttonStyle(DefaultActionButtonStyle(enabled: true)) + // We need to improve re-purchase flow + /* Button(UserText.viewPlansButtonTitle) { model.purchaseAction() } + .buttonStyle(DefaultActionButtonStyle(enabled: true)) */ Menu { Button(UserText.removeFromThisDeviceButton, action: { model.userEventHandler(.removeSubscriptionClick) diff --git a/LocalPackages/SystemExtensionManager/Sources/SystemExtensionManager/SystemExtensionManager.swift b/LocalPackages/SystemExtensionManager/Sources/SystemExtensionManager/SystemExtensionManager.swift index 956b79e3af..6b9f61a851 100644 --- a/LocalPackages/SystemExtensionManager/Sources/SystemExtensionManager/SystemExtensionManager.swift +++ b/LocalPackages/SystemExtensionManager/Sources/SystemExtensionManager/SystemExtensionManager.swift @@ -44,7 +44,9 @@ public struct SystemExtensionManager { self.workspace = workspace } - public func activate(waitingForUserApproval: @escaping () -> Void) async throws { + /// - Returns: The system extension version. + /// + public func activate(waitingForUserApproval: @escaping () -> Void) async throws -> String { /// Documenting a workaround for the issue discussed in https://app.asana.com/0/0/1205275221447702/f /// Background: For a lot of users, the system won't show the system-extension-blocked alert if there's a previous request /// to activate the extension. You can see active requests in your console using command `systemextensionsctl list`. @@ -62,11 +64,14 @@ public struct SystemExtensionManager { openSystemSettingsSecurity() } - return try await SystemExtensionRequest.activationRequest( + let activationRequest = SystemExtensionRequest.activationRequest( forExtensionWithIdentifier: extensionBundleID, manager: manager, waitingForUserApproval: waitingForUserApproval) - .submit() + + try await activationRequest.submit() + + return activationRequest.version ?? "unknown" } public func deactivate() async throws { @@ -113,6 +118,7 @@ final class SystemExtensionRequest: NSObject { private let request: OSSystemExtensionRequest private let manager: OSSystemExtensionManager private let waitingForUserApproval: (() -> Void)? + private(set) var version: String? private var continuation: CheckedContinuation? @@ -144,12 +150,30 @@ final class SystemExtensionRequest: NSObject { manager.submitRequest(request) } } + + private func updateVersion(to version: String) { + self.version = version + } + + private func updateVersionNumberIfMissing() { + guard version == nil, + let versionString = Bundle.main.infoDictionary?["CFBundleShortVersionString"] as? String else { + return + } + + var extensionVersion = versionString + + if let buildString = Bundle.main.infoDictionary?[kCFBundleVersionKey as String] as? String { + extensionVersion = extensionVersion + "." + buildString + } + } } extension SystemExtensionRequest: OSSystemExtensionRequestDelegate { func request(_ request: OSSystemExtensionRequest, actionForReplacingExtension existing: OSSystemExtensionProperties, withExtension ext: OSSystemExtensionProperties) -> OSSystemExtensionRequest.ReplacementAction { + updateVersion(to: ext.bundleShortVersion + "." + ext.bundleVersion) return .replace } @@ -160,6 +184,7 @@ extension SystemExtensionRequest: OSSystemExtensionRequestDelegate { func request(_ request: OSSystemExtensionRequest, didFinishWithResult result: OSSystemExtensionRequest.Result) { switch result { case .completed: + updateVersionNumberIfMissing() continuation?.resume() case .willCompleteAfterReboot: continuation?.resume(throwing: SystemExtensionRequestError.willActivateAfterReboot) diff --git a/UnitTests/DataExport/MockSecureVault.swift b/UnitTests/DataExport/MockSecureVault.swift index d9dd3ad32d..74124c8d5f 100644 --- a/UnitTests/DataExport/MockSecureVault.swift +++ b/UnitTests/DataExport/MockSecureVault.swift @@ -77,6 +77,12 @@ final class MockSecureVault: AutofillSecureVault { return accountID } + func updateLastUsedFor(accountId: Int64) throws { + if var account = storedAccounts.first(where: { $0.id == String(accountId) }) { + account.lastUsed = Date() + } + } + func deleteWebsiteCredentialsFor(accountId: Int64) throws { storedCredentials[accountId] = nil } @@ -298,6 +304,12 @@ class MockDatabaseProvider: AutofillDatabaseProvider { return _accounts } + func updateLastUsedForAccountId(_ accountId: Int64) throws { + if var account = _accounts.first(where: { $0.id == String(accountId) }) { + account.lastUsed = Date() + } + } + func deleteWebsiteCredentialsForAccountId(_ accountId: Int64) throws { self._accounts = self._accounts.filter { $0.id != String(accountId) } } diff --git a/UnitTests/PrivacyReferenceTests/BrokenSiteReportingReferenceTests.swift b/UnitTests/PrivacyReferenceTests/BrokenSiteReportingReferenceTests.swift index 7adfb26758..bc52a33041 100644 --- a/UnitTests/PrivacyReferenceTests/BrokenSiteReportingReferenceTests.swift +++ b/UnitTests/PrivacyReferenceTests/BrokenSiteReportingReferenceTests.swift @@ -90,6 +90,10 @@ final class BrokenSiteReportingReferenceTests: XCTestCase { reportFlow: .appMenu, errors: errors, httpStatusCodes: test.httpErrorCodes ?? [], + openerContext: nil, + vpnOn: false, + jsPerformance: nil, + userRefreshCount: 0, didOpenReportInfo: false, toggleReportCounter: nil) diff --git a/UnitTests/VPNFeedbackForm/VPNFeedbackFormViewModelTests.swift b/UnitTests/VPNFeedbackForm/VPNFeedbackFormViewModelTests.swift index e4794c1972..5ad28ed209 100644 --- a/UnitTests/VPNFeedbackForm/VPNFeedbackFormViewModelTests.swift +++ b/UnitTests/VPNFeedbackForm/VPNFeedbackFormViewModelTests.swift @@ -87,7 +87,8 @@ private class MockVPNMetadataCollector: VPNMetadataCollector { let appInfo = VPNMetadata.AppInfo( appVersion: "1.2.3", - lastVersionRun: "1.2.3", + lastAgentVersionRun: "1.2.3", + lastExtensionVersionRun: "1.2.3", isInternalUser: false, isInApplicationsDirectory: true ) diff --git a/UnitTests/WebsiteBreakageReport/WebsiteBreakageReportTests.swift b/UnitTests/WebsiteBreakageReport/WebsiteBreakageReportTests.swift index 6848c4c481..c5c389feac 100644 --- a/UnitTests/WebsiteBreakageReport/WebsiteBreakageReportTests.swift +++ b/UnitTests/WebsiteBreakageReport/WebsiteBreakageReportTests.swift @@ -47,6 +47,10 @@ class WebsiteBreakageReportTests: XCTestCase { reportFlow: .appMenu, errors: nil, httpStatusCodes: nil, + openerContext: nil, + vpnOn: false, + jsPerformance: nil, + userRefreshCount: 0, didOpenReportInfo: false, toggleReportCounter: nil ) @@ -92,6 +96,10 @@ class WebsiteBreakageReportTests: XCTestCase { reportFlow: .appMenu, errors: nil, httpStatusCodes: nil, + openerContext: nil, + vpnOn: false, + jsPerformance: nil, + userRefreshCount: 0, didOpenReportInfo: false, toggleReportCounter: nil )