Compare commits
2 Commits
master
...
am/default
Author | SHA1 | Date | |
---|---|---|---|
2909fdf41e | |||
3149c50299 |
@ -47,6 +47,12 @@ public class WireGuardAdapter {
|
||||
/// Packet tunnel provider.
|
||||
private weak var packetTunnelProvider: NEPacketTunnelProvider?
|
||||
|
||||
/// KVO observer for `NEProvider.defaultPath`.
|
||||
private var defaultPathObserver: NSKeyValueObservation?
|
||||
|
||||
/// Last known default path.
|
||||
private var currentDefaultPath: NetworkExtension.NWPath?
|
||||
|
||||
/// Log handler closure.
|
||||
private let logHandler: LogHandler
|
||||
|
||||
@ -181,11 +187,7 @@ public class WireGuardAdapter {
|
||||
return
|
||||
}
|
||||
|
||||
let networkMonitor = NWPathMonitor()
|
||||
networkMonitor.pathUpdateHandler = { [weak self] path in
|
||||
self?.didReceivePathUpdate(path: path)
|
||||
}
|
||||
networkMonitor.start(queue: self.workQueue)
|
||||
self.addDefaultPathObserver()
|
||||
|
||||
do {
|
||||
let settingsGenerator = try self.makeSettingsGenerator(with: tunnelConfiguration)
|
||||
@ -198,10 +200,10 @@ public class WireGuardAdapter {
|
||||
try self.startWireGuardBackend(wgConfig: wgConfig),
|
||||
settingsGenerator
|
||||
)
|
||||
self.networkMonitor = networkMonitor
|
||||
|
||||
completionHandler(nil)
|
||||
} catch let error as WireGuardAdapterError {
|
||||
networkMonitor.cancel()
|
||||
self.removeDefaultPathObserver()
|
||||
completionHandler(error)
|
||||
} catch {
|
||||
fatalError()
|
||||
@ -225,8 +227,7 @@ public class WireGuardAdapter {
|
||||
return
|
||||
}
|
||||
|
||||
self.networkMonitor?.cancel()
|
||||
self.networkMonitor = nil
|
||||
self.removeDefaultPathObserver()
|
||||
|
||||
self.state = .stopped
|
||||
|
||||
@ -313,26 +314,29 @@ public class WireGuardAdapter {
|
||||
/// - Returns: `PacketTunnelSettingsGenerator`.
|
||||
private func setNetworkSettings(_ networkSettings: NEPacketTunnelNetworkSettings) throws {
|
||||
var systemError: Error?
|
||||
let condition = NSCondition()
|
||||
|
||||
// Activate the condition
|
||||
condition.lock()
|
||||
defer { condition.unlock() }
|
||||
let dispatchGroup = DispatchGroup()
|
||||
|
||||
dispatchGroup.enter()
|
||||
|
||||
self.packetTunnelProvider?.setTunnelNetworkSettings(networkSettings) { error in
|
||||
systemError = error
|
||||
condition.signal()
|
||||
dispatchGroup.leave()
|
||||
}
|
||||
|
||||
// Packet tunnel's `setTunnelNetworkSettings` times out in certain
|
||||
// scenarios & never calls the given callback.
|
||||
let setTunnelNetworkSettingsTimeout: TimeInterval = 5 // seconds
|
||||
let setTunnelNetworkSettingsTimeout: Int = 5 // seconds
|
||||
|
||||
if condition.wait(until: Date().addingTimeInterval(setTunnelNetworkSettingsTimeout)) {
|
||||
let waitResult = dispatchGroup.wait(wallTimeout: .now() + .seconds(setTunnelNetworkSettingsTimeout))
|
||||
|
||||
switch waitResult {
|
||||
case .success:
|
||||
if let systemError = systemError {
|
||||
throw WireGuardAdapterError.setNetworkSettings(systemError)
|
||||
}
|
||||
} else {
|
||||
|
||||
case .timedOut:
|
||||
self.logHandler(.error, "setTunnelNetworkSettings timed out after 5 seconds; proceeding anyway")
|
||||
}
|
||||
}
|
||||
@ -411,19 +415,48 @@ public class WireGuardAdapter {
|
||||
}
|
||||
}
|
||||
|
||||
/// Helper method used by network path monitor.
|
||||
private func addDefaultPathObserver() {
|
||||
guard let packetTunnelProvider = packetTunnelProvider else { return }
|
||||
|
||||
defaultPathObserver?.invalidate()
|
||||
defaultPathObserver = packetTunnelProvider.observe(\.defaultPath, options: [.new]) { [weak self] _, change in
|
||||
guard let self = self, let defaultPath = change.newValue?.flatMap({ $0 }) else { return }
|
||||
|
||||
self.workQueue.async {
|
||||
self.didReceivePathUpdate(path: defaultPath)
|
||||
}
|
||||
}
|
||||
|
||||
currentDefaultPath = packetTunnelProvider.defaultPath
|
||||
}
|
||||
|
||||
private func removeDefaultPathObserver() {
|
||||
defaultPathObserver?.invalidate()
|
||||
defaultPathObserver = nil
|
||||
currentDefaultPath = nil
|
||||
}
|
||||
|
||||
/// Method invoked by KVO observer when new network path is received.
|
||||
/// - Parameter path: new network path
|
||||
private func didReceivePathUpdate(path: Network.NWPath) {
|
||||
self.logHandler(.verbose, "Network change detected with \(path.status) route and interface order \(path.availableInterfaces)")
|
||||
private func didReceivePathUpdate(path: NetworkExtension.NWPath) {
|
||||
let isSamePath = currentDefaultPath?.isEqual(to: path) ?? false
|
||||
|
||||
currentDefaultPath = path
|
||||
|
||||
self.logHandler(.verbose, "Network change detected with \(path.status)")
|
||||
|
||||
#if os(macOS)
|
||||
if case .started(let handle, _) = self.state {
|
||||
if case .started(let handle, _) = self.state, !isSamePath {
|
||||
wgBumpSockets(handle)
|
||||
}
|
||||
#elseif os(iOS)
|
||||
let isSatisfiable = path.status == .satisfied || path.status == .satisfiable
|
||||
|
||||
switch self.state {
|
||||
case .started(let handle, let settingsGenerator):
|
||||
if path.status.isSatisfiable {
|
||||
if isSatisfiable {
|
||||
guard !isSamePath else { return }
|
||||
|
||||
let (wgConfig, resolutionResults) = settingsGenerator.endpointUapiConfiguration()
|
||||
self.logEndpointResolutionResults(resolutionResults)
|
||||
|
||||
@ -438,7 +471,7 @@ public class WireGuardAdapter {
|
||||
}
|
||||
|
||||
case .temporaryShutdown(let settingsGenerator):
|
||||
guard path.status.isSatisfiable else { return }
|
||||
guard isSatisfiable else { return }
|
||||
|
||||
self.logHandler(.verbose, "Connectivity online, resuming backend.")
|
||||
|
||||
@ -472,16 +505,19 @@ public enum WireGuardLogLevel: Int32 {
|
||||
case error = 1
|
||||
}
|
||||
|
||||
private extension Network.NWPath.Status {
|
||||
/// Returns `true` if the path is potentially satisfiable.
|
||||
var isSatisfiable: Bool {
|
||||
extension NetworkExtension.NWPathStatus: CustomDebugStringConvertible {
|
||||
public var debugDescription: String {
|
||||
switch self {
|
||||
case .requiresConnection, .satisfied:
|
||||
return true
|
||||
case .unsatisfied:
|
||||
return false
|
||||
return "unsatisfied"
|
||||
case .satisfied:
|
||||
return "satisfied"
|
||||
case .satisfiable:
|
||||
return "satisfiable"
|
||||
case .invalid:
|
||||
return "invalid"
|
||||
@unknown default:
|
||||
return true
|
||||
return "unknown (rawValue = \(rawValue))"
|
||||
}
|
||||
}
|
||||
}
|
||||
|
Reference in New Issue
Block a user