// // NETCPInterface.swift // TunnelKit // // Created by Davide De Rosa on 4/15/18. // Copyright © 2018 London Trust Media. All rights reserved. // import Foundation import NetworkExtension import SwiftyBeaver private let log = SwiftyBeaver.self class NETCPSocket: NSObject, GenericSocket { private static var linkContext = 0 private let impl: NWTCPConnection init(impl: NWTCPConnection) { self.impl = impl isActive = false isShutdown = false } // MARK: GenericSocket private weak var queue: DispatchQueue? private var isActive: Bool private(set) var isShutdown: Bool var remoteAddress: String? { return (impl.remoteAddress as? NWHostEndpoint)?.hostname } var hasBetterPath: Bool { return impl.hasBetterPath } weak var delegate: GenericSocketDelegate? func observe(queue: DispatchQueue, activeTimeout: Int) { isActive = false self.queue = queue queue.schedule(after: .milliseconds(activeTimeout)) { [weak self] in guard let _self = self else { return } guard _self.isActive else { _self.delegate?.socketShouldChangeProtocol(_self) _self.delegate?.socketDidTimeout(_self) return } } impl.addObserver(self, forKeyPath: #keyPath(NWTCPConnection.state), options: [.initial, .new], context: &NETCPSocket.linkContext) impl.addObserver(self, forKeyPath: #keyPath(NWTCPConnection.hasBetterPath), options: .new, context: &NETCPSocket.linkContext) } func unobserve() { impl.removeObserver(self, forKeyPath: #keyPath(NWTCPConnection.state), context: &NETCPSocket.linkContext) impl.removeObserver(self, forKeyPath: #keyPath(NWTCPConnection.hasBetterPath), context: &NETCPSocket.linkContext) } func shutdown() { impl.writeClose() impl.cancel() } func upgraded() -> GenericSocket? { guard impl.hasBetterPath else { return nil } return NETCPSocket(impl: NWTCPConnection(upgradeFor: impl)) } func link(withMTU mtu: Int) -> LinkInterface { return NETCPLink(impl: impl) } // MARK: Connection KVO (any queue) override func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) { guard (context == &NETCPSocket.linkContext) else { super.observeValue(forKeyPath: keyPath, of: object, change: change, context: context) return } // if let keyPath = keyPath { // log.debug("KVO change reported (\(anyPointer(object)).\(keyPath))") // } queue?.async { self.observeValueInTunnelQueue(forKeyPath: keyPath, of: object, change: change, context: context) } } private func observeValueInTunnelQueue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) { // if let keyPath = keyPath { // log.debug("KVO change reported (\(anyPointer(object)).\(keyPath))") // } guard let impl = object as? NWTCPConnection, (impl == self.impl) else { log.warning("Discard KVO change from old socket") return } guard let keyPath = keyPath else { return } switch keyPath { case #keyPath(NWTCPConnection.state): if let resolvedEndpoint = impl.remoteAddress { log.debug("Socket state is \(impl.state) (endpoint: \(impl.endpoint) -> \(resolvedEndpoint))") } else { log.debug("Socket state is \(impl.state) (endpoint: \(impl.endpoint) -> in progress)") } switch impl.state { case .connected: guard !isActive else { return } isActive = true delegate?.socketDidBecomeActive(self) case .cancelled: isShutdown = true delegate?.socket(self, didShutdownWithFailure: false) case .disconnected: isShutdown = true delegate?.socket(self, didShutdownWithFailure: true) default: break } case #keyPath(NWTCPConnection.hasBetterPath): guard impl.hasBetterPath else { break } log.debug("Socket has a better path") delegate?.socketHasBetterPath(self) default: break } } } class NETCPLink: LinkInterface { private let impl: NWTCPConnection private let maxPacketSize: Int init(impl: NWTCPConnection, maxPacketSize: Int? = nil) { self.impl = impl self.mtu = .max self.maxPacketSize = maxPacketSize ?? (512 * 1024) } // MARK: LinkInterface let isReliable: Bool = true var remoteAddress: String? { return (impl.remoteAddress as? NWHostEndpoint)?.hostname } let mtu: Int var packetBufferSize: Int { return maxPacketSize } let negotiationTimeout: TimeInterval = 10.0 let hardResetTimeout: TimeInterval = 5.0 func setReadHandler(queue: DispatchQueue, _ handler: @escaping ([Data]?, Error?) -> Void) { loopReadPackets(queue, Data(), handler) } private func loopReadPackets(_ queue: DispatchQueue, _ buffer: Data, _ handler: @escaping ([Data]?, Error?) -> Void) { // WARNING: runs in Network.framework queue impl.readMinimumLength(2, maximumLength: packetBufferSize) { [weak self] (data, error) in guard let _ = self else { return } queue.sync { guard (error == nil), let data = data else { handler(nil, error) return } var newBuffer = buffer newBuffer.append(contentsOf: data) let (until, packets) = CommonPacket.parsed(newBuffer) newBuffer = newBuffer.subdata(in: until.. Void)?) { let stream = CommonPacket.stream(packet) impl.write(stream) { (error) in completionHandler?(error) } } func writePackets(_ packets: [Data], completionHandler: ((Error?) -> Void)?) { let stream = CommonPacket.stream(packets) impl.write(stream) { (error) in completionHandler?(error) } } } extension NETCPSocket { override var description: String { guard let hostEndpoint = impl.endpoint as? NWHostEndpoint else { return impl.endpoint.description } return "\(hostEndpoint.hostname):\(hostEndpoint.port)" } }