tunnelkit/Sources/TunnelKitOpenVPNProtocol/OpenVPNSession.swift

1264 lines
43 KiB
Swift
Raw Normal View History

2018-08-23 08:19:25 +00:00
//
// OpenVPNSession.swift
// TunnelKit
2018-08-23 08:19:25 +00:00
//
// Created by Davide De Rosa on 2/3/17.
2020-12-27 16:29:39 +00:00
// Copyright (c) 2021 Davide De Rosa. All rights reserved.
//
// https://github.com/passepartoutvpn
//
// This file is part of TunnelKit.
//
// TunnelKit 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.
//
// TunnelKit 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 TunnelKit. If not, see <http://www.gnu.org/licenses/>.
//
// This file incorporates work covered by the following copyright and
// permission notice:
//
// Copyright (c) 2018-Present Private Internet Access
//
// Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
//
2018-08-23 08:19:25 +00:00
import Foundation
import SwiftyBeaver
2021-10-25 14:27:27 +00:00
import TunnelKitCore
import TunnelKitOpenVPNCore
import CTunnelKitCore
import CTunnelKitOpenVPNProtocol
2018-08-23 08:19:25 +00:00
private let log = SwiftyBeaver.self
/// Observes major events notified by a `OpenVPNSession`.
2021-06-26 07:43:50 +00:00
public protocol OpenVPNSessionDelegate: AnyObject {
2018-08-23 08:19:25 +00:00
/**
Called after starting a session.
2018-08-23 08:19:25 +00:00
- Parameter remoteAddress: The address of the VPN server.
- Parameter options: The pulled tunnel settings.
2018-08-23 08:19:25 +00:00
*/
func sessionDidStart(_: OpenVPNSession, remoteAddress: String, options: OpenVPN.Configuration)
2018-08-23 08:19:25 +00:00
/**
Called after stopping a session.
- Parameter error: An optional `Error` being the reason of the stop.
2018-08-23 08:19:25 +00:00
- Parameter shouldReconnect: When `true`, the session can/should be restarted. Usually because the stop reason was recoverable.
- Seealso: `OpenVPNSession.reconnect(...)`
2018-08-23 08:19:25 +00:00
*/
func sessionDidStop(_: OpenVPNSession, withError error: Error?, shouldReconnect: Bool)
2018-08-23 08:19:25 +00:00
}
/// Provides methods to set up and maintain an OpenVPN session.
public class OpenVPNSession: Session {
private enum StopMethod {
case shutdown
2018-08-23 08:19:25 +00:00
case reconnect
}
// MARK: Configuration
/// The session base configuration.
public let configuration: OpenVPN.Configuration
/// The optional credentials.
public var credentials: OpenVPN.Credentials?
private var keepAliveInterval: TimeInterval? {
let interval: TimeInterval?
if let negInterval = pushReply?.options.keepAliveInterval, negInterval > 0.0 {
interval = negInterval
} else if let cfgInterval = configuration.keepAliveInterval, cfgInterval > 0.0 {
interval = cfgInterval
} else {
return nil
}
return interval
}
private var keepAliveTimeout: TimeInterval {
if let negTimeout = pushReply?.options.keepAliveTimeout, negTimeout > 0.0 {
return negTimeout
} else if let cfgTimeout = configuration.keepAliveTimeout, cfgTimeout > 0.0 {
return cfgTimeout
} else {
return CoreConfiguration.OpenVPN.pingTimeout
}
}
/// An optional `OpenVPNSessionDelegate` for receiving session events.
public weak var delegate: OpenVPNSessionDelegate?
// MARK: State
2018-08-23 08:19:25 +00:00
private let queue: DispatchQueue
2018-08-23 08:19:25 +00:00
private var tlsObserver: NSObjectProtocol?
private var withLocalOptions: Bool
2018-08-23 08:19:25 +00:00
private var keys: [UInt8: OpenVPN.SessionKey]
2018-08-23 08:19:25 +00:00
private var oldKeys: [OpenVPN.SessionKey]
2018-08-23 08:19:25 +00:00
private var negotiationKeyIdx: UInt8
private var currentKeyIdx: UInt8?
private var isRenegotiating: Bool
private var negotiationKey: OpenVPN.SessionKey {
guard let key = keys[negotiationKeyIdx] else {
fatalError("Keys are empty or index \(negotiationKeyIdx) not found in \(keys.keys)")
}
return key
}
private var currentKey: OpenVPN.SessionKey? {
guard let i = currentKeyIdx else {
return nil
2018-08-23 08:19:25 +00:00
}
return keys[i]
}
private var link: LinkInterface?
private var tunnel: TunnelInterface?
private var isReliableLink: Bool {
return link?.isReliable ?? false
}
2018-08-23 08:19:25 +00:00
private var continuatedPushReplyMessage: String?
2018-08-23 08:19:25 +00:00
private var pushReply: OpenVPN.PushReply?
private var nextPushRequestDate: Date?
private var connectedDate: Date?
2018-10-07 14:21:22 +00:00
private var lastPing: BidirectionalState<Date>
private(set) var isStopping: Bool
/// The optional reason why the session stopped.
public private(set) var stopError: Error?
// MARK: Control
private var controlChannel: OpenVPN.ControlChannel
private var authenticator: OpenVPN.Authenticator?
// MARK: Init
/**
Creates a VPN session.
- Parameter queue: The `DispatchQueue` where to run the session loop.
- Parameter configuration: The `Configuration` to use for this session.
*/
public init(queue: DispatchQueue, configuration: OpenVPN.Configuration) throws {
guard let _ = configuration.ca else {
throw ConfigurationError.missingConfiguration(option: "ca")
}
self.queue = queue
self.configuration = configuration
withLocalOptions = true
keys = [:]
oldKeys = []
negotiationKeyIdx = 0
isRenegotiating = false
lastPing = BidirectionalState(withResetValue: Date.distantPast)
isStopping = false
if let tlsWrap = configuration.tlsWrap {
switch tlsWrap.strategy {
case .auth:
controlChannel = try OpenVPN.ControlChannel(withAuthKey: tlsWrap.key, digest: configuration.fallbackDigest)
case .crypt:
controlChannel = try OpenVPN.ControlChannel(withCryptKey: tlsWrap.key)
}
} else {
controlChannel = OpenVPN.ControlChannel()
}
}
deinit {
cleanup()
}
// MARK: Session
2018-08-23 08:19:25 +00:00
public func setLink(_ link: LinkInterface) {
guard (self.link == nil) else {
log.warning("Link interface already set!")
return
2018-08-23 08:19:25 +00:00
}
log.debug("Starting VPN session")
// WARNING: runs in notification source queue (we know it's "queue", but better be safe than sorry)
tlsObserver = NotificationCenter.default.addObserver(forName: .TLSBoxPeerVerificationError, object: nil, queue: nil) { (notification) in
let error = notification.userInfo?[OpenVPNErrorKey] as? Error
self.queue.async {
self.deferStop(.shutdown, error)
2018-08-23 08:19:25 +00:00
}
}
self.link = link
start()
}
public func canRebindLink() -> Bool {
// return (pushReply?.peerId != nil)
// FIXME: floating is currently unreliable
return false
}
public func rebindLink(_ link: LinkInterface) {
guard let _ = pushReply?.options.peerId else {
log.warning("Session doesn't support link rebinding!")
return
2018-08-23 08:19:25 +00:00
}
isStopping = false
stopError = nil
2018-08-23 08:19:25 +00:00
log.debug("Rebinding VPN session to a new link")
self.link = link
loopLink()
}
2018-08-23 08:19:25 +00:00
public func setTunnel(tunnel: TunnelInterface) {
guard (self.tunnel == nil) else {
log.warning("Tunnel interface already set!")
return
2018-08-23 08:19:25 +00:00
}
self.tunnel = tunnel
loopTunnel()
}
2018-08-23 08:19:25 +00:00
public func dataCount() -> (Int, Int)? {
guard let _ = link else {
return nil
}
return controlChannel.currentDataCount()
}
public func serverConfiguration() -> Any? {
return pushReply?.options
}
public func shutdown(error: Error?) {
guard !isStopping else {
log.warning("Ignore stop request, already stopping!")
return
2018-08-23 08:19:25 +00:00
}
deferStop(.shutdown, error)
}
public func reconnect(error: Error?) {
guard !isStopping else {
log.warning("Ignore stop request, already stopping!")
return
2018-08-23 08:19:25 +00:00
}
deferStop(.reconnect, error)
}
// Ruby: cleanup
public func cleanup() {
log.info("Cleaning up...")
if let observer = tlsObserver {
NotificationCenter.default.removeObserver(observer)
tlsObserver = nil
2018-08-23 08:19:25 +00:00
}
keys.removeAll()
oldKeys.removeAll()
negotiationKeyIdx = 0
currentKeyIdx = nil
isRenegotiating = false
nextPushRequestDate = nil
connectedDate = nil
authenticator = nil
continuatedPushReplyMessage = nil
pushReply = nil
link = nil
if !(tunnel?.isPersistent ?? false) {
tunnel = nil
2018-08-23 08:19:25 +00:00
}
isStopping = false
stopError = nil
}
2018-08-23 08:19:25 +00:00
// MARK: Loop
// Ruby: start
private func start() {
loopLink()
hardReset()
}
private func loopNegotiation() {
guard let link = link else {
return
}
guard !keys.isEmpty else {
return
}
guard !negotiationKey.didHardResetTimeOut(link: link) else {
doReconnect(error: OpenVPNError.negotiationTimeout)
return
}
guard !negotiationKey.didNegotiationTimeOut(link: link) else {
doShutdown(error: OpenVPNError.negotiationTimeout)
return
}
2018-08-23 08:19:25 +00:00
pushRequest()
if !isReliableLink {
flushControlQueue()
}
guard negotiationKey.controlState == .connected else {
queue.asyncAfter(deadline: .now() + CoreConfiguration.OpenVPN.tickInterval) { [weak self] in
self?.loopNegotiation()
2018-08-23 08:19:25 +00:00
}
return
}
// let loop die when negotiation is complete
}
2018-08-23 08:19:25 +00:00
// Ruby: udp_loop
private func loopLink() {
let loopedLink = link
loopedLink?.setReadHandler(queue: queue) { [weak self] (newPackets, error) in
guard self?.link === loopedLink else {
log.warning("Ignoring read from outdated LINK")
2018-08-23 08:19:25 +00:00
return
}
if let error = error {
log.error("Failed LINK read: \(error)")
// XXX: why isn't the tunnel shutting down at this point?
2018-08-23 08:19:25 +00:00
return
}
if let packets = newPackets, !packets.isEmpty {
self?.maybeRenegotiate()
// log.verbose("Received \(packets.count) packets from LINK")
self?.receiveLink(packets: packets)
2018-08-23 08:19:25 +00:00
}
}
}
// Ruby: tun_loop
private func loopTunnel() {
tunnel?.setReadHandler(queue: queue) { [weak self] (newPackets, error) in
if let error = error {
log.error("Failed TUN read: \(error)")
2018-08-23 08:19:25 +00:00
return
}
if let packets = newPackets, !packets.isEmpty {
2019-12-12 08:34:08 +00:00
// log.verbose("Received \(packets.count) packets from TUN")
self?.receiveTunnel(packets: packets)
}
2018-08-23 08:19:25 +00:00
}
}
2018-08-23 08:19:25 +00:00
// Ruby: recv_link
private func receiveLink(packets: [Data]) {
guard shouldHandlePackets() else {
2019-12-12 08:32:34 +00:00
log.warning("Discarding \(packets.count) LINK packets (should not handle)")
return
2018-08-23 08:19:25 +00:00
}
lastPing.inbound = Date()
2018-08-23 08:19:25 +00:00
var dataPacketsByKey = [UInt8: [Data]]()
for packet in packets {
// log.verbose("Received data from LINK (\(packet.count) bytes): \(packet.toHex())")
2018-08-23 08:19:25 +00:00
guard let firstByte = packet.first else {
log.warning("Dropped malformed packet (missing opcode)")
continue
2018-08-23 08:19:25 +00:00
}
let codeValue = firstByte >> 3
guard let code = PacketCode(rawValue: codeValue) else {
log.warning("Dropped malformed packet (unknown code: \(codeValue))")
continue
2018-08-23 08:19:25 +00:00
}
// log.verbose("Parsed packet with code \(code)")
var offset = 1
if (code == .dataV2) {
guard packet.count >= offset + PacketPeerIdLength else {
log.warning("Dropped malformed packet (missing peerId)")
continue
}
offset += PacketPeerIdLength
}
2018-08-23 08:19:25 +00:00
if (code == .dataV1) || (code == .dataV2) {
let key = firstByte & 0b111
guard let _ = keys[key] else {
log.warning("Key with id \(key) not found")
// deferStop(.shutdown, OpenVPNError.badKey)
continue // JK: This used to be return, but we'd see connections that would stay in Connecting state forever
2018-08-23 08:19:25 +00:00
}
// XXX: improve with array reference
var dataPackets = dataPacketsByKey[key] ?? [Data]()
dataPackets.append(packet)
dataPacketsByKey[key] = dataPackets
2018-08-23 08:19:25 +00:00
continue
}
2018-08-23 08:19:25 +00:00
let controlPacket: ControlPacket
do {
let parsedPacket = try controlChannel.readInboundPacket(withData: packet, offset: 0)
handleAcks()
if parsedPacket.code == .ackV1 {
continue
}
controlPacket = parsedPacket
} catch let e {
log.warning("Dropped malformed packet: \(e)")
continue
// deferStop(.shutdown, e)
// return
}
switch code {
case .hardResetServerV2:
// HARD_RESET coming during a SOFT_RESET handshake (before connecting)
guard !isRenegotiating else {
deferStop(.shutdown, OpenVPNError.staleSession)
return
}
case .softResetV1:
if !isRenegotiating {
softReset(isServerInitiated: true)
}
default:
break
}
sendAck(for: controlPacket)
2018-08-23 08:19:25 +00:00
let pendingInboundQueue = controlChannel.enqueueInboundPacket(packet: controlPacket)
for inboundPacket in pendingInboundQueue {
handleControlPacket(inboundPacket)
2018-08-23 08:19:25 +00:00
}
}
2018-08-23 08:19:25 +00:00
// send decrypted packets to tunnel all at once
for (keyId, dataPackets) in dataPacketsByKey {
guard let sessionKey = keys[keyId] else {
log.warning("Accounted a data packet for which the cryptographic key hadn't been found")
continue
2018-08-23 08:19:25 +00:00
}
handleDataPackets(dataPackets, key: sessionKey)
2018-08-23 08:19:25 +00:00
}
}
// Ruby: recv_tun
private func receiveTunnel(packets: [Data]) {
guard shouldHandlePackets() else {
2019-12-12 08:32:34 +00:00
log.warning("Discarding \(packets.count) TUN packets (should not handle)")
return
}
sendDataPackets(packets)
}
// Ruby: ping
private func ping() {
guard currentKey?.controlState == .connected else {
return
2018-08-23 08:19:25 +00:00
}
let now = Date()
guard now.timeIntervalSince(lastPing.inbound) <= keepAliveTimeout else {
deferStop(.shutdown, OpenVPNError.pingTimeout)
return
}
2018-08-23 08:19:25 +00:00
// is keep-alive enabled?
if let _ = keepAliveInterval {
log.debug("Send ping")
sendDataPackets([OpenVPN.DataPacket.pingString])
lastPing.outbound = Date()
}
// schedule even just to check for ping timeout
scheduleNextPing()
}
private func scheduleNextPing() {
let interval: TimeInterval
if let keepAliveInterval = keepAliveInterval {
interval = keepAliveInterval
2021-01-27 00:36:48 +00:00
log.verbose("Schedule ping after \(interval.asTimeString)")
} else {
interval = CoreConfiguration.OpenVPN.pingTimeoutCheckInterval
2021-01-27 00:36:48 +00:00
log.verbose("Schedule ping timeout check after \(interval.asTimeString)")
2018-08-23 08:19:25 +00:00
}
queue.asyncAfter(deadline: .now() + interval) { [weak self] in
2019-12-12 12:29:20 +00:00
log.verbose("Running ping block")
self?.ping()
2018-08-23 08:19:25 +00:00
}
}
// MARK: Handshake
// Ruby: reset_ctrl
private func resetControlChannel(forNewSession: Bool) {
authenticator = nil
do {
try controlChannel.reset(forNewSession: forNewSession)
} catch let e {
deferStop(.shutdown, e)
}
}
// Ruby: hard_reset
private func hardReset() {
log.debug("Send hard reset")
resetControlChannel(forNewSession: true)
continuatedPushReplyMessage = nil
pushReply = nil
negotiationKeyIdx = 0
let newKey = OpenVPN.SessionKey(id: UInt8(negotiationKeyIdx), timeout: CoreConfiguration.OpenVPN.negotiationTimeout)
keys[negotiationKeyIdx] = newKey
log.debug("Negotiation key index is \(negotiationKeyIdx)")
let payload = hardResetPayload() ?? Data()
negotiationKey.state = .hardReset
guard !keys.isEmpty else {
fatalError("Main loop must follow hard reset, keys are empty!")
}
loopNegotiation()
enqueueControlPackets(code: .hardResetClientV2, key: UInt8(negotiationKeyIdx), payload: payload)
}
private func hardResetPayload() -> Data? {
guard !(configuration.usesPIAPatches ?? false) else {
guard let ca = configuration.ca else {
log.error("Configuration doesn't have a CA")
return nil
}
let caMD5: String
do {
caMD5 = try TLSBox.md5(forCertificatePEM: ca.pem)
} catch {
log.error("CA MD5 could not be computed, skipping custom HARD_RESET")
return nil
2018-10-23 20:44:35 +00:00
}
log.debug("CA MD5 is: \(caMD5)")
return try? PIAHardReset(
caMd5Digest: caMD5,
cipher: configuration.fallbackCipher,
digest: configuration.fallbackDigest
).encodedData()
2018-08-23 08:19:25 +00:00
}
return nil
}
// Ruby: soft_reset
private func softReset(isServerInitiated: Bool) {
guard !isRenegotiating else {
log.warning("Renegotiation already in progress")
return
}
if isServerInitiated {
log.debug("Handle soft reset")
} else {
log.debug("Send soft reset")
2018-08-23 08:19:25 +00:00
}
resetControlChannel(forNewSession: false)
negotiationKeyIdx = max(1, (negotiationKeyIdx + 1) % OpenVPN.ProtocolMacros.numberOfKeys)
let newKey = OpenVPN.SessionKey(id: UInt8(negotiationKeyIdx), timeout: CoreConfiguration.OpenVPN.softNegotiationTimeout)
keys[negotiationKeyIdx] = newKey
log.debug("Negotiation key index is \(negotiationKeyIdx)")
negotiationKey.state = .softReset
isRenegotiating = true
loopNegotiation()
if !isServerInitiated {
enqueueControlPackets(code: .softResetV1, key: UInt8(negotiationKeyIdx), payload: Data())
2018-08-23 08:19:25 +00:00
}
}
// Ruby: on_tls_connect
private func onTLSConnect() {
log.debug("TLS.connect: Handshake is complete")
negotiationKey.controlState = .preAuth
2018-08-23 08:19:25 +00:00
do {
authenticator = try OpenVPN.Authenticator(credentials?.username, pushReply?.options.authToken ?? credentials?.password)
authenticator?.withLocalOptions = withLocalOptions
try authenticator?.putAuth(into: negotiationKey.tls, options: configuration)
} catch let e {
deferStop(.shutdown, e)
return
}
2018-08-23 08:19:25 +00:00
let cipherTextOut: Data
do {
cipherTextOut = try negotiationKey.tls.pullCipherText()
} catch let e {
if let _ = e.openVPNErrorCode() {
log.error("TLS.auth: Failed pulling ciphertext (error: \(e))")
shutdown(error: e)
2018-08-23 08:19:25 +00:00
return
}
log.verbose("TLS.auth: Still can't pull ciphertext")
return
}
2018-08-23 08:19:25 +00:00
log.debug("TLS.auth: Pulled ciphertext (\(cipherTextOut.count) bytes)")
enqueueControlPackets(code: .controlV1, key: negotiationKey.id, payload: cipherTextOut)
}
// Ruby: push_request
private func pushRequest() {
guard negotiationKey.controlState == .preIfConfig else {
return
}
guard let targetDate = nextPushRequestDate, Date() > targetDate else {
return
2018-08-23 08:19:25 +00:00
}
log.debug("TLS.ifconfig: Put plaintext (PUSH_REQUEST)")
try? negotiationKey.tls.putPlainText("PUSH_REQUEST\0")
let cipherTextOut: Data
do {
cipherTextOut = try negotiationKey.tls.pullCipherText()
} catch let e {
if let _ = e.openVPNErrorCode() {
log.error("TLS.auth: Failed pulling ciphertext (error: \(e))")
shutdown(error: e)
return
2018-08-23 08:19:25 +00:00
}
log.verbose("TLS.ifconfig: Still can't pull ciphertext")
return
}
log.debug("TLS.ifconfig: Send pulled ciphertext (\(cipherTextOut.count) bytes)")
enqueueControlPackets(code: .controlV1, key: negotiationKey.id, payload: cipherTextOut)
if isRenegotiating {
completeConnection()
isRenegotiating = false
}
nextPushRequestDate = Date().addingTimeInterval(CoreConfiguration.OpenVPN.pushRequestInterval)
}
private func maybeRenegotiate() {
guard let renegotiatesAfter = configuration.renegotiatesAfter, renegotiatesAfter > 0 else {
return
}
guard (negotiationKeyIdx == currentKeyIdx) else {
return
}
let elapsed = -negotiationKey.startTime.timeIntervalSinceNow
if (elapsed > renegotiatesAfter) {
2021-01-27 00:36:48 +00:00
log.debug("Renegotiating after \(elapsed.asTimeString)")
softReset(isServerInitiated: false)
}
}
private func completeConnection() {
setupEncryption()
authenticator?.reset()
negotiationKey.controlState = .connected
connectedDate = Date()
transitionKeys()
}
// MARK: Control
// Ruby: handle_ctrl_pkt
private func handleControlPacket(_ packet: ControlPacket) {
guard packet.key == negotiationKey.id else {
log.error("Bad key in control packet (\(packet.key) != \(negotiationKey.id))")
// deferStop(.shutdown, OpenVPNError.badKey)
return
}
guard let ca = configuration.ca else {
log.error("Configuration doesn't have a CA")
return
}
// start new TLS handshake
if ((packet.code == .hardResetServerV2) && (negotiationKey.state == .hardReset)) ||
((packet.code == .softResetV1) && (negotiationKey.state == .softReset)) {
if negotiationKey.state == .hardReset {
controlChannel.remoteSessionId = packet.sessionId
}
guard let remoteSessionId = controlChannel.remoteSessionId else {
log.error("No remote sessionId (never set)")
deferStop(.shutdown, OpenVPNError.missingSessionId)
return
}
guard packet.sessionId == remoteSessionId else {
log.error("Packet session mismatch (\(packet.sessionId.toHex()) != \(remoteSessionId.toHex()))")
deferStop(.shutdown, OpenVPNError.sessionMismatch)
return
}
negotiationKey.state = .tls
log.debug("Start TLS handshake")
let tls = TLSBox(
ca: ca.pem,
clientCertificate: configuration.clientCertificate?.pem,
clientKey: configuration.clientKey?.pem,
checksEKU: configuration.checksEKU ?? false,
checksSANHost: configuration.checksSANHost ?? false,
hostname: configuration.sanHost
)
if let tlsSecurityLevel = configuration.tlsSecurityLevel {
tls.securityLevel = tlsSecurityLevel
}
negotiationKey.tlsOptional = tls
do {
try negotiationKey.tls.start()
} catch let e {
deferStop(.shutdown, e)
return
}
let cipherTextOut: Data
do {
cipherTextOut = try negotiationKey.tls.pullCipherText()
} catch let e {
if let _ = e.openVPNErrorCode() {
log.error("TLS.connect: Failed pulling ciphertext (error: \(e))")
shutdown(error: e)
return
}
deferStop(.shutdown, e)
return
2018-08-23 08:19:25 +00:00
}
log.debug("TLS.connect: Pulled ciphertext (\(cipherTextOut.count) bytes)")
enqueueControlPackets(code: .controlV1, key: negotiationKey.id, payload: cipherTextOut)
}
// exchange TLS ciphertext
else if ((packet.code == .controlV1) && (negotiationKey.state == .tls)) {
guard let remoteSessionId = controlChannel.remoteSessionId else {
log.error("No remote sessionId found in packet (control packets before server HARD_RESET)")
deferStop(.shutdown, OpenVPNError.missingSessionId)
return
}
guard packet.sessionId == remoteSessionId else {
log.error("Packet session mismatch (\(packet.sessionId.toHex()) != \(remoteSessionId.toHex()))")
deferStop(.shutdown, OpenVPNError.sessionMismatch)
return
}
guard let cipherTextIn = packet.payload else {
log.warning("TLS.connect: Control packet with empty payload?")
return
}
log.debug("TLS.connect: Put received ciphertext (\(cipherTextIn.count) bytes)")
try? negotiationKey.tls.putCipherText(cipherTextIn)
let cipherTextOut: Data
do {
cipherTextOut = try negotiationKey.tls.pullCipherText()
log.debug("TLS.connect: Send pulled ciphertext (\(cipherTextOut.count) bytes)")
enqueueControlPackets(code: .controlV1, key: negotiationKey.id, payload: cipherTextOut)
} catch let e {
if let _ = e.openVPNErrorCode() {
log.error("TLS.connect: Failed pulling ciphertext (error: \(e))")
shutdown(error: e)
return
}
log.verbose("TLS.connect: No available ciphertext to pull")
}
if negotiationKey.shouldOnTLSConnect() {
onTLSConnect()
}
do {
while true {
let controlData = try controlChannel.currentControlData(withTLS: negotiationKey.tls)
handleControlData(controlData)
}
} catch _ {
}
2018-08-23 08:19:25 +00:00
}
}
2018-08-23 08:19:25 +00:00
// Ruby: handle_ctrl_data
private func handleControlData(_ data: ZeroingData) {
guard let auth = authenticator else {
return
}
if CoreConfiguration.logsSensitiveData {
log.debug("Pulled plain control data (\(data.count) bytes): \(data.toHex())")
} else {
log.debug("Pulled plain control data (\(data.count) bytes)")
}
2018-08-23 08:19:25 +00:00
auth.appendControlData(data)
if (negotiationKey.controlState == .preAuth) {
2018-08-23 08:19:25 +00:00
do {
guard try auth.parseAuthReply() else {
2018-08-23 08:19:25 +00:00
return
}
} catch let e {
deferStop(.shutdown, e)
return
}
negotiationKey.controlState = .preIfConfig
nextPushRequestDate = Date()
pushRequest()
nextPushRequestDate?.addTimeInterval(isRenegotiating ? CoreConfiguration.OpenVPN.pushRequestInterval : CoreConfiguration.OpenVPN.retransmissionLimit)
2018-08-23 08:19:25 +00:00
}
for message in auth.parseMessages() {
if CoreConfiguration.logsSensitiveData {
log.debug("Parsed control message (\(message.count) bytes): \"\(message)\"")
} else {
log.debug("Parsed control message (\(message.count) bytes)")
}
handleControlMessage(message)
2018-08-23 08:19:25 +00:00
}
}
// Ruby: handle_ctrl_msg
private func handleControlMessage(_ message: String) {
if CoreConfiguration.logsSensitiveData {
log.debug("Received control message: \"\(message)\"")
}
// disconnect on authentication failure
guard !message.hasPrefix("AUTH_FAILED") else {
// XXX: retry without client options
if authenticator?.withLocalOptions ?? false {
log.warning("Authentication failure, retrying without local options")
withLocalOptions = false
deferStop(.reconnect, OpenVPNError.badCredentials)
return
}
deferStop(.shutdown, OpenVPNError.badCredentials)
return
}
2018-08-23 08:19:25 +00:00
// disconnect on remote server restart (--explicit-exit-notify)
guard !message.hasPrefix("RESTART") else {
log.debug("Disconnecting due to server shutdown")
deferStop(.shutdown, OpenVPNError.serverShutdown)
return
}
// handle authentication from now on
guard negotiationKey.controlState == .preIfConfig else {
return
}
let completeMessage: String
if let continuated = continuatedPushReplyMessage {
completeMessage = "\(continuated),\(message)"
} else {
completeMessage = message
}
let reply: OpenVPN.PushReply
do {
guard let optionalReply = try OpenVPN.PushReply(message: completeMessage) else {
2018-08-23 08:19:25 +00:00
return
}
reply = optionalReply
log.debug("Received PUSH_REPLY: \"\(reply.maskedDescription)\"")
if let framing = reply.options.compressionFraming, let compression = reply.options.compressionAlgorithm {
switch compression {
case .disabled:
break
case .LZO:
2021-10-25 14:27:27 +00:00
if !LZOFactory.isSupported() {
log.error("Server has LZO compression enabled and this was not built into the library (framing=\(framing))")
throw OpenVPNError.serverCompression
}
case .other:
log.error("Server has non-LZO compression enabled and this is currently unsupported (framing=\(framing))")
throw OpenVPNError.serverCompression
}
}
} catch OpenVPNError.continuationPushReply {
continuatedPushReplyMessage = completeMessage.replacingOccurrences(of: "push-continuation", with: "")
// FIXME: strip "PUSH_REPLY" and "push-continuation 2"
return
} catch let e {
deferStop(.shutdown, e)
return
}
pushReply = reply
guard reply.options.ipv4 != nil || reply.options.ipv6 != nil else {
deferStop(.shutdown, OpenVPNError.noRouting)
return
}
completeConnection()
2018-08-23 08:19:25 +00:00
guard let remoteAddress = link?.remoteAddress else {
fatalError("Could not resolve link remote address")
}
delegate?.sessionDidStart(self, remoteAddress: remoteAddress, options: reply.options)
scheduleNextPing()
}
// Ruby: transition_keys
private func transitionKeys() {
if let key = currentKey {
oldKeys.append(key)
}
currentKeyIdx = negotiationKeyIdx
cleanKeys()
}
// Ruby: clean_keys
private func cleanKeys() {
while (oldKeys.count > 1) {
let key = oldKeys.removeFirst()
keys.removeValue(forKey: key.id)
}
}
// Ruby: q_ctrl
private func enqueueControlPackets(code: PacketCode, key: UInt8, payload: Data) {
guard let _ = link else {
log.warning("Not writing to LINK, interface is down")
return
}
controlChannel.enqueueOutboundPackets(withCode: code, key: key, payload: payload, maxPacketSize: 1000)
flushControlQueue()
}
// Ruby: flush_ctrl_q_out
private func flushControlQueue() {
let rawList: [Data]
do {
rawList = try controlChannel.writeOutboundPackets()
} catch let e {
log.warning("Failed control packet serialization: \(e)")
deferStop(.shutdown, e)
return
}
for raw in rawList {
log.debug("Send control packet (\(raw.count) bytes): \(raw.toHex())")
}
// WARNING: runs in Network.framework queue
let writeLink = link
link?.writePackets(rawList) { [weak self] (error) in
self?.queue.sync {
guard self?.link === writeLink else {
log.warning("Ignoring write from outdated LINK")
return
}
if let error = error {
log.error("Failed LINK write during control flush: \(error)")
self?.deferStop(.shutdown, OpenVPNError.failedLinkWrite)
return
}
2018-08-23 08:19:25 +00:00
}
}
}
// Ruby: setup_keys
private func setupEncryption() {
guard let auth = authenticator else {
fatalError("Setting up encryption without having authenticated")
}
guard let sessionId = controlChannel.sessionId else {
fatalError("Setting up encryption without a local sessionId")
}
guard let remoteSessionId = controlChannel.remoteSessionId else {
fatalError("Setting up encryption without a remote sessionId")
}
guard let serverRandom1 = auth.serverRandom1, let serverRandom2 = auth.serverRandom2 else {
fatalError("Setting up encryption without server randoms")
}
guard let pushReply = pushReply else {
fatalError("Setting up encryption without a former PUSH_REPLY")
}
2018-09-01 23:39:02 +00:00
if CoreConfiguration.logsSensitiveData {
log.debug("Set up encryption from the following components:")
log.debug("\tpreMaster: \(auth.preMaster.toHex())")
log.debug("\trandom1: \(auth.random1.toHex())")
log.debug("\trandom2: \(auth.random2.toHex())")
log.debug("\tserverRandom1: \(serverRandom1.toHex())")
log.debug("\tserverRandom2: \(serverRandom2.toHex())")
log.debug("\tsessionId: \(sessionId.toHex())")
log.debug("\tremoteSessionId: \(remoteSessionId.toHex())")
} else {
log.debug("Set up encryption")
2018-09-01 23:39:02 +00:00
}
let pushedCipher = pushReply.options.cipher
if let negCipher = pushedCipher {
log.info("\tNegotiated cipher: \(negCipher.rawValue)")
}
let pushedFraming = pushReply.options.compressionFraming
if let negFraming = pushedFraming {
log.info("\tNegotiated compression framing: \(negFraming)")
}
let pushedCompression = pushReply.options.compressionAlgorithm
if let negCompression = pushedCompression {
log.info("\tNegotiated compression algorithm: \(negCompression)")
}
if let negPing = pushReply.options.keepAliveInterval {
2021-01-27 00:36:48 +00:00
log.info("\tNegotiated keep-alive interval: \(negPing.asTimeString)")
}
if let negPingRestart = pushReply.options.keepAliveTimeout {
2021-01-27 00:36:48 +00:00
log.info("\tNegotiated keep-alive timeout: \(negPingRestart.asTimeString)")
}
2018-08-23 08:19:25 +00:00
let bridge: OpenVPN.EncryptionBridge
do {
bridge = try OpenVPN.EncryptionBridge(
pushedCipher ?? configuration.fallbackCipher,
configuration.fallbackDigest,
auth,
sessionId,
remoteSessionId
)
} catch let e {
deferStop(.shutdown, e)
return
}
2018-08-23 08:19:25 +00:00
negotiationKey.dataPath = DataPath(
encrypter: bridge.encrypter(),
decrypter: bridge.decrypter(),
peerId: pushReply.options.peerId ?? PacketPeerIdDisabled,
compressionFraming: (pushedFraming ?? configuration.fallbackCompressionFraming).native,
compressionAlgorithm: (pushedCompression ?? configuration.compressionAlgorithm ?? .disabled).native,
maxPackets: link?.packetBufferSize ?? 200,
usesReplayProtection: CoreConfiguration.OpenVPN.usesReplayProtection
)
}
// MARK: Data
// Ruby: handle_data_pkt
private func handleDataPackets(_ packets: [Data], key: OpenVPN.SessionKey) {
controlChannel.addReceivedDataCount(packets.flatCount)
do {
guard let decryptedPackets = try key.decrypt(packets: packets) else {
log.warning("Could not decrypt packets, is SessionKey properly configured (dataPath, peerId)?")
return
2018-08-23 08:19:25 +00:00
}
guard !decryptedPackets.isEmpty else {
2018-08-23 08:19:25 +00:00
return
}
tunnel?.writePackets(decryptedPackets, completionHandler: nil)
} catch let e {
guard !e.isOpenVPNError() else {
deferStop(.shutdown, e)
return
2018-08-23 08:19:25 +00:00
}
deferStop(.reconnect, e)
2018-08-23 08:19:25 +00:00
}
}
// Ruby: send_data_pkt
private func sendDataPackets(_ packets: [Data]) {
guard let key = currentKey else {
return
2018-08-23 08:19:25 +00:00
}
do {
guard let encryptedPackets = try key.encrypt(packets: packets) else {
log.warning("Could not encrypt packets, is SessionKey properly configured (dataPath, peerId)?")
return
}
guard !encryptedPackets.isEmpty else {
2018-08-23 08:19:25 +00:00
return
}
// WARNING: runs in Network.framework queue
controlChannel.addSentDataCount(encryptedPackets.flatCount)
let writeLink = link
link?.writePackets(encryptedPackets) { [weak self] (error) in
self?.queue.sync {
guard self?.link === writeLink else {
log.warning("Ignoring write from outdated LINK")
return
}
if let error = error {
log.error("Data: Failed LINK write during send data: \(error)")
self?.deferStop(.shutdown, OpenVPNError.failedLinkWrite)
return
2018-08-23 08:19:25 +00:00
}
// log.verbose("Data: \(encryptedPackets.count) packets successfully written to LINK")
2018-08-23 08:19:25 +00:00
}
}
} catch let e {
guard !e.isOpenVPNError() else {
deferStop(.shutdown, e)
return
2018-08-23 08:19:25 +00:00
}
deferStop(.reconnect, e)
}
}
// MARK: Acks
private func handleAcks() {
}
// Ruby: send_ack
private func sendAck(for controlPacket: ControlPacket) {
log.debug("Send ack for received packetId \(controlPacket.packetId)")
let raw: Data
do {
raw = try controlChannel.writeAcks(
withKey: controlPacket.key,
ackPacketIds: [controlPacket.packetId],
ackRemoteSessionId: controlPacket.sessionId
)
} catch let e {
deferStop(.shutdown, e)
return
2018-08-23 08:19:25 +00:00
}
// WARNING: runs in Network.framework queue
let writeLink = link
link?.writePacket(raw) { [weak self] (error) in
self?.queue.sync {
guard self?.link === writeLink else {
log.warning("Ignoring write from outdated LINK")
return
}
if let error = error {
log.error("Failed LINK write during send ack for packetId \(controlPacket.packetId): \(error)")
self?.deferStop(.shutdown, OpenVPNError.failedLinkWrite)
return
}
log.debug("Ack successfully written to LINK for packetId \(controlPacket.packetId)")
}
}
}
// MARK: Stop
private func shouldHandlePackets() -> Bool {
2019-12-12 08:32:34 +00:00
return !isStopping && !keys.isEmpty
}
private func deferStop(_ method: StopMethod, _ error: Error?) {
guard !isStopping else {
return
}
isStopping = true
let completion = { [weak self] in
switch method {
case .shutdown:
self?.doShutdown(error: error)
case .reconnect:
self?.doReconnect(error: error)
}
}
// shut down after sending exit notification if socket is unreliable (normally UDP)
if let link = link, !link.isReliable {
do {
guard let packets = try currentKey?.encrypt(packets: [OpenVPN.OCCPacket.exit.serialized()]) else {
completion()
return
}
link.writePackets(packets) { [weak self] (error) in
self?.queue.sync {
2019-03-20 17:01:57 +00:00
completion()
}
}
} catch {
completion()
}
} else {
completion()
2018-08-23 08:19:25 +00:00
}
}
private func doShutdown(error: Error?) {
if let error = error {
log.error("Trigger shutdown (error: \(error))")
} else {
log.info("Trigger shutdown on request")
2018-08-23 08:19:25 +00:00
}
stopError = error
delegate?.sessionDidStop(self, withError: error, shouldReconnect: false)
}
private func doReconnect(error: Error?) {
if let error = error {
log.error("Trigger reconnection (error: \(error))")
} else {
log.info("Trigger reconnection on request")
2018-08-23 08:19:25 +00:00
}
stopError = error
delegate?.sessionDidStop(self, withError: error, shouldReconnect: true)
2018-08-23 08:19:25 +00:00
}
}