0

我正在构建一个 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
    }   
}
}
4

0 回答 0