我正在构建一个 Swift 桌面应用程序,它使用 GCDAsyncUDPSocket 通过 UDP 与网络上的多个硬件设备进行通信。对于初始扫描,我创建一个套接字并发送一个广播消息并监听响应。当我收到来自设备的响应时,会为该设备创建一个唯一的非广播套接字,用于所有其他通信。
通过我的有线以太网时,这几乎可以完美运行。但是,当通过 WiFi 时,我看到 Wireshark 中不断返回 ARP 请求,其中许多请求的响应是错误的 MAC 地址,用于应该接收消息的接口。
我一直在寻找两天来试图找到一个经过验证的真正解决 ARP 问题的方法,但没有任何运气。
我在 GCDAsync 文档中看到,使用 sendData toAddress 而不是 toHost 将包含 MAC 地址,但无法弄清楚如何使用我的接口详细信息创建地址 NSData 对象来尝试这个,甚至不确定这是否将解决 ARP 问题。
这是我的广播套接字的当前代码:
class BroadcastNetworkSocket: NSObject, GCDAsyncUdpSocketDelegate{
var socket:GCDAsyncUdpSocket!
let broadcastIP = "255.255.255.255"
let broadcastPort:UInt16 = 22202
let PORT:UInt16 = 0
override init (){
super.init()
}
init(whichInterface: String){
super.init()
setupConnection(whichInterface)
}
func setupConnection(whichInterface: String){
var error : NSError?
socket = GCDAsyncUdpSocket(delegate: self, delegateQueue: dispatch_get_main_queue())
socket.setPreferIPv4()
socket.enableBroadcast(true, error: &error)
socket.beginReceiving(&error)
}
func send(message:String, toAddress:String, toPort:UInt16){
let data = message.dataUsingEncoding(NSUTF8StringEncoding)
socket!.sendData(data, toHost: toAddress, port: toPort, withTimeout: 2, tag: 0)
}
func sendNetworkScan(){
let message = "!??#"
let data = message.dataUsingEncoding(NSUTF8StringEncoding)
socket!.sendData(data, toHost: broadcastIP, port: broadcastPort, withTimeout: 2, tag: 0)
println("outgoing message: \(message)");
}
func closeSocket(){
socket!.close()
}
func udpSocket(sock: GCDAsyncUdpSocket!, didReceiveData data: NSData!, fromAddress address: NSData!, withFilterContext filterContext: AnyObject!) {
let theMessage = NSString(data: data, encoding: NSUTF8StringEncoding)
var host: NSString?
var port1: UInt16 = 0
GCDAsyncUdpSocket.getHost(&host, port: &port1, fromAddress: address)
println("incoming message: \(theMessage!)")
println("incoming address: \(host!)")
NSNotificationCenter.defaultCenter().postNotificationName(broadcastNetworkMessageReceivedNotificationKey, object: self, userInfo: ["message":theMessage!, "incomingIP":host!])
}
func verifyIPAddress(incomingText: String)-> Bool{
let validIpAddressRegex = "^(([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\\.){3}([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])$"
if incomingText.rangeOfString(validIpAddressRegex, options: .RegularExpressionSearch) != nil{
return true
}
else {
return false
}
}
}