// // AppContext.swift // Passepartout // // Created by Davide De Rosa on 8/29/24. // Copyright (c) 2024 Davide De Rosa. All rights reserved. // // https://github.com/passepartoutvpn // // This file is part of Passepartout. // // Passepartout is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // Passepartout is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with Passepartout. If not, see . // import Combine import CommonLibrary import CommonUtils import Foundation import PassepartoutKit @MainActor public final class AppContext: ObservableObject { public let iapManager: IAPManager public let profileManager: ProfileManager public let tunnel: ExtendedTunnel public let registry: Registry public let providerManager: ProviderManager private var isActivating = false private var subscriptions: Set public init( iapManager: IAPManager, profileManager: ProfileManager, tunnel: ExtendedTunnel, registry: Registry, providerManager: ProviderManager ) { self.iapManager = iapManager self.profileManager = profileManager self.tunnel = tunnel self.registry = registry self.providerManager = providerManager subscriptions = [] observeObjects() } public func onApplicationActive() { guard !isActivating else { return } isActivating = true Task { do { pp_log(.app, .notice, "Application became active") await iapManager.reloadReceipt() pp_log(.app, .notice, "Prepare tunnel and purge stale data...") try await tunnel.prepare(purge: true) } catch { pp_log(.app, .fault, "Unable to prepare tunnel: \(error)") } isActivating = false } } } // MARK: - Observation private extension AppContext { func observeObjects() { profileManager.observeObjects() profileManager .didChange .sink { [weak self] event in switch event { case .save(let profile): self?.syncTunnelIfCurrentProfile(profile) default: break } } .store(in: &subscriptions) } } private extension AppContext { func syncTunnelIfCurrentProfile(_ profile: Profile) { guard profile.id == tunnel.currentProfile?.id else { return } Task { guard [.active, .activating].contains(tunnel.status) else { return } if profile.isInteractive { try await tunnel.disconnect() return } do { try await tunnel.connect(with: profile) } catch { try await tunnel.disconnect() } } } }