passepartout-apple/Passepartout/Library/Sources/UILibrary/Business/AppContext.swift

120 lines
3.3 KiB
Swift
Raw Normal View History

2024-09-23 13:02:26 +00:00
//
// 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 <http://www.gnu.org/licenses/>.
//
import Combine
import CommonLibrary
import CommonUtils
2024-09-23 13:02:26 +00:00
import Foundation
import PassepartoutKit
@MainActor
public final class AppContext: ObservableObject {
public let iapManager: IAPManager
public let profileManager: ProfileManager
public let tunnel: ExtendedTunnel
2024-09-23 13:02:26 +00:00
public let registry: Registry
public let providerManager: ProviderManager
2024-09-23 13:02:26 +00:00
private var subscriptions: Set<AnyCancellable>
public init(
iapManager: IAPManager,
profileManager: ProfileManager,
tunnel: ExtendedTunnel,
2024-09-23 13:02:26 +00:00
registry: Registry,
providerManager: ProviderManager
2024-09-23 13:02:26 +00:00
) {
self.iapManager = iapManager
self.profileManager = profileManager
self.tunnel = tunnel
self.registry = registry
self.providerManager = providerManager
subscriptions = []
2024-09-23 13:02:26 +00:00
observeObjects()
2024-09-23 13:02:26 +00:00
}
public func onApplicationActive() {
Task {
do {
pp_log(.app, .notice, "Application became active")
pp_log(.app, .notice, "Reload IAP receipt...")
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)")
}
}
}
2024-09-23 13:02:26 +00:00
}
// MARK: - Observation
2024-09-23 13:02:26 +00:00
private extension AppContext {
func observeObjects() {
profileManager.observeObjects()
2024-09-23 13:02:26 +00:00
profileManager
.didChange
.sink { [weak self] event in
switch event {
case .save(let profile):
self?.syncTunnelIfCurrentProfile(profile)
default:
break
2024-09-23 13:02:26 +00:00
}
}
.store(in: &subscriptions)
}
}
2024-09-23 13:02:26 +00:00
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()
}
}
2024-09-23 13:02:26 +00:00
}
}