2018-08-15 22:34:16 +00:00
|
|
|
//
|
2018-09-06 05:42:23 +00:00
|
|
|
// Copyright © 2018 WireGuard LLC. All rights reserved.
|
2018-08-15 22:34:16 +00:00
|
|
|
//
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
|
|
|
|
extension Interface {
|
|
|
|
|
2018-09-16 20:38:32 +00:00
|
|
|
var publicKey: String? {
|
|
|
|
if let privateKeyString = privateKey, let privateKey = Data(base64Encoded: privateKeyString) {
|
|
|
|
var publicKey = Data(count: 32)
|
|
|
|
privateKey.withUnsafeBytes({ (privateKeyBytes) -> Void in
|
|
|
|
publicKey.withUnsafeMutableBytes({ (mutableBytes) -> Void in
|
|
|
|
curve25519_derive_public_key(mutableBytes, privateKeyBytes)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
return publicKey.base64EncodedString()
|
|
|
|
} else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-15 22:34:16 +00:00
|
|
|
func validate() throws {
|
|
|
|
guard let privateKey = privateKey, !privateKey.isEmpty else {
|
|
|
|
throw InterfaceValidationError.emptyPrivateKey
|
|
|
|
}
|
|
|
|
|
|
|
|
guard privateKey.isBase64() else {
|
|
|
|
throw InterfaceValidationError.invalidPrivateKey
|
|
|
|
}
|
|
|
|
|
2018-08-16 20:03:40 +00:00
|
|
|
try addresses?.commaSeparatedToArray().forEach { address in
|
2018-08-15 22:34:16 +00:00
|
|
|
do {
|
|
|
|
try _ = CIDRAddress(stringRepresentation: address)
|
|
|
|
} catch {
|
|
|
|
throw InterfaceValidationError.invalidAddress(cause: error)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-16 20:03:40 +00:00
|
|
|
try dns?.commaSeparatedToArray().forEach { address in
|
2018-08-15 22:34:16 +00:00
|
|
|
do {
|
2018-08-16 20:41:45 +00:00
|
|
|
try _ = Endpoint(endpointString: address, needsPort: false)
|
2018-08-15 22:34:16 +00:00
|
|
|
} catch {
|
|
|
|
throw InterfaceValidationError.invalidDNSServer(cause: error)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-21 16:00:41 +00:00
|
|
|
func parse(attribute: Attribute) throws {
|
|
|
|
switch attribute.key {
|
|
|
|
case .address:
|
|
|
|
addresses = attribute.stringValue
|
|
|
|
case .dns:
|
|
|
|
dns = attribute.stringValue
|
|
|
|
case .listenPort:
|
|
|
|
if let port = Int16(attribute.stringValue) {
|
|
|
|
listenPort = port
|
|
|
|
}
|
|
|
|
case .mtu:
|
|
|
|
if let mtu = Int32(attribute.stringValue) {
|
|
|
|
self.mtu = mtu
|
|
|
|
}
|
|
|
|
case .privateKey:
|
|
|
|
privateKey = attribute.stringValue
|
|
|
|
default:
|
|
|
|
throw TunnelParseError.invalidLine(attribute.line)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-29 13:01:59 +00:00
|
|
|
func export() -> String {
|
|
|
|
var exportString = "[Interface]\n"
|
|
|
|
if let privateKey = privateKey {
|
2018-08-31 19:40:24 +00:00
|
|
|
exportString.append("PrivateKey=\(privateKey)\n")
|
|
|
|
}
|
|
|
|
if let addresses = addresses {
|
|
|
|
exportString.append("Address=\(addresses)\n")
|
|
|
|
}
|
|
|
|
if let dns = dns {
|
|
|
|
exportString.append("DNS=\(dns)\n")
|
|
|
|
}
|
|
|
|
if mtu > 0 {
|
|
|
|
exportString.append("MTU=\(mtu)\n")
|
2018-08-29 13:01:59 +00:00
|
|
|
}
|
|
|
|
if listenPort > 0 {
|
2018-08-31 19:40:24 +00:00
|
|
|
exportString.append("ListenPort=\(listenPort)\n")
|
2018-08-29 13:01:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
exportString.append("\n")
|
|
|
|
|
|
|
|
return exportString
|
|
|
|
}
|
|
|
|
|
2018-08-15 22:34:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
enum InterfaceValidationError: Error {
|
|
|
|
case emptyPrivateKey
|
|
|
|
case invalidPrivateKey
|
|
|
|
case invalidAddress(cause: Error)
|
|
|
|
case invalidDNSServer(cause: Error)
|
|
|
|
}
|