2016-07-05 12 views
5

Staram Bonjour w szybkim 3Swift 3 jak rozwiązać NetService IP?

tu jest mój kod, mogę odbierać delegata

func netServiceDidResolveAddress(_ sender: NetService) { 
print("netServiceDidResolveAddress service name \(sender.name) of type \(sender.type)," + 
       "port \(sender.port), addresses \(sender.addresses)") 
} 

A oto mój wynik

nazwa usługi

netServiceDidResolveAddress Webbera Mac mini typu _myapp ._tcp., port 5678, adresy Opcjonalne ([< 1002162e c0a80205 00000000 00000000>, < 1c1e162e 00000000 fe800000 00000000 00bce7ad 24b4b7e8 08000000>])

c0a80205 jest IP I szuka => 192.168.2.5

oraz adres to [Dane], Apple mówi

adresy usług. Jest to instancja NSArray dla NSData, z których każda zawiera pojedynczą strukturę sockaddr odpowiednią do użycia z connect (2). Jeśli żaden adres nie zostanie rozwiązany dla usługi lub usługa nie została jeszcze rozwiązana, zwracana jest pusta tablica NSArray .

Nadal mylę się, dlaczego dane nie mogą korzystać z .btyes? Jak mówi Apple "To jest instancja NSAD z NSData" Ale nie mogę jej użyć jak NSData

A jak rozwiązać ten adres jako czytelny ciąg IP?

próbuję tego wcześniej, ale nie uzyskać wynik jak I z wyjątkiem ...

let thedata = NSData(bytes: sender.addresses, length: (sender.addresses?.count)!) 
var storage = sockaddr_storage() 
thedata.getBytes(&storage, length: sizeof(sockaddr_storage)) 

if Int32(storage.ss_family) == AF_INET { 
    let addr4 = withUnsafePointer(&storage) {UnsafePointer<sockaddr_in>($0).pointee } 
    print(inet_ntoa(addr4.sin_addr)); 
} 

Każda propozycja będzie pomoc, Dzięki

Odpowiedz

2

OK ... nie jest to inteligentna odpowiedź, przynajmniej można uzyskać czytelny IP

prostu użyć tej func dostać String IP

let bonjourDevices = [NetService]() 

let bonjourDevice = bonjourDevices[0] 

let host = self.getIPV4StringfromAddress(address:bonjourDevice.addresses!) 


func getIPV4StringfromAddress(address: [Data] , port : Int) -> String{ 

     let data = address.first! as NSData; 

     var ip1 = UInt8(0) 
     data.getBytes(&ip1, range: NSMakeRange(4, 1)) 

     var ip2 = UInt8(0) 
     data.getBytes(&ip2, range: NSMakeRange(5, 1)) 

     var ip3 = UInt8(0) 
     data.getBytes(&ip3, range: NSMakeRange(6, 1)) 

     var ip4 = UInt8(0) 
     data.getBytes(&ip4, range: NSMakeRange(7, 1)) 

     let ipStr = String(format: "%d.%d.%d.%d:%d",ip1,ip2,ip3,ip4,port); 

     return ipStr; 
    } 
2

mogę „T sprawiają, że praca z Data, ale przy użyciu NSData, chciałbym użyć tego:

extension NSData { 
    func castToCPointer<T>() -> T { 
     let mem = UnsafeMutablePointer<T>.allocate(capacity: MemoryLayout<T.Type>.size) 
     self.getBytes(mem, length: MemoryLayout<T.Type>.size) 
     return mem.move() 
    } 
} 

Więc mamy netServiceDidResolveAddress:

func netServiceDidResolveAddress(_ sender: NetService) { 
    if let addresses = sender.addresses, addresses.count > 0 { 
     for address in addresses { 
      let data = address as NSData 

      let inetAddress: sockaddr_in = data.castToCPointer() 
      if inetAddress.sin_family == __uint8_t(AF_INET) { 
       if let ip = String(cString: inet_ntoa(inetAddress.sin_addr), encoding: .ascii) { 
        // IPv4 
        print(ip) 
       } 
      } else if inetAddress.sin_family == __uint8_t(AF_INET6) { 
       let inetAddress6: sockaddr_in6 = data.castToCPointer() 
       let ipStringBuffer = UnsafeMutablePointer<Int8>.allocate(capacity: Int(INET6_ADDRSTRLEN)) 
       var addr = inetAddress6.sin6_addr 

       if let ipString = inet_ntop(Int32(inetAddress6.sin6_family), &addr, ipStringBuffer, __uint32_t(INET6_ADDRSTRLEN)) { 
        if let ip = String(cString: ipString, encoding: .ascii) { 
         // IPv6 
         print(ip) 
        } 
       } 

       ipStringBuffer.deallocate(capacity: Int(INET6_ADDRSTRLEN)) 
      } 
     } 
    } 
} 

Mam następujący wynik (zapisywanie adresów IP w tablicy przed wyświetlaczem) :

["172.16.10.120", "172.16.8.251", "::", "::82c9:d9a5:2eed:1c87"] 

Kod zainspirowany https://gist.github.com/agrippa1994/d8c66a2ded74fb2dd801 pisemnej w Swift 2.3 i przystosowany do Swift 3.0

7

Oto jak to zrobiłem w Swift 3.

func netServiceDidResolveAddress(_ sender: NetService) { 
    var hostname = [CChar](repeating: 0, count: Int(NI_MAXHOST)) 
    guard let data = sender.addresses?.first else { return } 
    data.withUnsafeBytes { (pointer:UnsafePointer<sockaddr>) -> Void in 
     guard getnameinfo(pointer, socklen_t(data.count), &hostname, socklen_t(hostname.count), nil, 0, NI_NUMERICHOST) == 0 else { 
      return 
     } 
    } 
    let ipAddress = String(cString:hostname) 
    print(ipAddress) 
}