// // ExtendedTunnel.swift // Passepartout // // Created by Davide De Rosa on 9/7/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 Foundation import PassepartoutKit @MainActor public final class ExtendedTunnel: ObservableObject { private let tunnel: Tunnel private let environment: TunnelEnvironment private let interval: TimeInterval public func value(forKey key: TunnelEnvironmentKey) -> T? where T: Decodable { environment.environmentValue(forKey: key) } @Published public private(set) var lastErrorCode: PassepartoutError.Code? { didSet { pp_log(.app, .info, "ExtendedTunnel.lastErrorCode -> \(lastErrorCode?.rawValue ?? "nil")") } } @Published public private(set) var dataCount: DataCount? private var subscriptions: Set public init( tunnel: Tunnel, environment: TunnelEnvironment, interval: TimeInterval ) { self.tunnel = tunnel self.environment = environment self.interval = interval subscriptions = [] } public func observeObjects() { tunnel .$status .receive(on: DispatchQueue.main) .sink { [weak self] in guard let self else { return } switch $0 { case .activating: lastErrorCode = nil default: lastErrorCode = value(forKey: TunnelEnvironmentKeys.lastErrorCode) } if $0 != .active { dataCount = nil } } .store(in: &subscriptions) tunnel .$currentProfile .receive(on: DispatchQueue.main) .sink { [weak self] _ in self?.objectWillChange.send() } .store(in: &subscriptions) Timer .publish(every: interval, on: .main, in: .common) .autoconnect() .sink { [weak self] _ in guard let self else { return } guard tunnel.status == .active else { return } dataCount = value(forKey: TunnelEnvironmentKeys.dataCount) } .store(in: &subscriptions) } } extension ExtendedTunnel { public var status: TunnelStatus { tunnel.status } public var connectionStatus: TunnelStatus { var status = tunnel.status if status == .active, let environmentConnectionStatus { if environmentConnectionStatus == .connected { status = .active } else { status = .activating } } return status } private var environmentConnectionStatus: ConnectionStatus? { value(forKey: TunnelEnvironmentKeys.connectionStatus) } } extension ExtendedTunnel { public var currentProfile: TunnelCurrentProfile? { tunnel.currentProfile } public func prepare(purge: Bool) async throws { try await tunnel.prepare(purge: purge) } public func install(_ profile: Profile, processor: ProfileProcessor) async throws { let newProfile = try processor.processed(profile) try await tunnel.install(newProfile, connect: false, title: processor.title) } public func connect(with profile: Profile, processor: ProfileProcessor) async throws { let newProfile = try processor.processed(profile) try await tunnel.install(newProfile, connect: true, title: processor.title) } public func disconnect() async throws { try await tunnel.disconnect() } public func currentLog(parameters: Constants.Log) async -> [String] { let output = try? await tunnel.sendMessage(.localLog( sinceLast: parameters.sinceLast, maxLevel: parameters.maxLevel )) switch output { case .debugLog(let log): return log.lines.map(parameters.formatter.formattedLine) default: return [] } } }