管理 ifaddrs 以在 Swift 中返回 MAC 地址

Posted

技术标签:

【中文标题】管理 ifaddrs 以在 Swift 中返回 MAC 地址【英文标题】:Manage ifaddrs to return MAC addresses as well in Swift 【发布时间】:2016-03-04 18:07:27 【问题描述】:

我有以下代码可以成功检索连接到我的路由器的所有 IP。但我需要获取每个 IP 的 MAC 地址。

所以地址不是作为数组返回[ips],而是作为字典返回[ips:0000000, mac: 000000]

是否可以通过更改以下代码来实现(来自How to get Ip address in swift)?

func getIFAddresses() -> [String] 


print("GET IF ADDRESSSE")

var addresses = [String]()

// Get list of all interfaces on the local machine:
var ifaddr : UnsafeMutablePointer<ifaddrs> = nil
if getifaddrs(&ifaddr) == 0 

    print("getifaddrs\(getifaddrs)")

    // For each interface ...
    for (var ptr = ifaddr; ptr != nil; ptr = ptr.memory.ifa_next) 
        let flags = Int32(ptr.memory.ifa_flags)
        var addr = ptr.memory.ifa_addr.memory

        print("flags\(flags)")
        print("addr\(addr)")

        // Check for running IPv4, IPv6 interfaces. Skip the loopback interface.
        if (flags & (IFF_UP|IFF_RUNNING|IFF_LOOPBACK)) == (IFF_UP|IFF_RUNNING) 



            if addr.sa_family == UInt8(AF_INET) || addr.sa_family == UInt8(AF_INET6) 

                print("addr.sa_family\(addr.sa_family)")

                // Convert interface address to a human readable string:
                var hostname = [CChar](count: Int(NI_MAXHOST), repeatedValue: 0)

                print("hostname\(hostname)")

                if (getnameinfo(
                    &addr, socklen_t(addr.sa_len),
                    &hostname,
                    socklen_t(hostname.count),
                    nil,
                    socklen_t(0), NI_NUMERICHOST) == 0) 


                        if let address = String.fromCString(hostname) 
                            addresses.append(address)
                        
                
            
        
    
    freeifaddrs(ifaddr)
    print("freeifaddrs\(freeifaddrs)")

print("ADDRESSES \(addresses)")
return addresses

【问题讨论】:

你试过了吗?代码有效吗?如果不是什么问题? @DanBeaulieu 你是对的,正在删除 cmets。我应该知道最好不要尝试解释为什么通常不建议投反对票。是的,我的答案已被复制并用作另一个问题的答案。 【参考方案1】:

(备注/澄清:这是对“在 Swift 中管理 ifaddrs 以返回 MAC 地址”“是否有可能修改 How to get Ip address in swift 的代码以返回 MAC 地址”。 这不是“检索连接到我的路由器的所有 IP”的解决方案 问题正文中也提到了这一点。)

这里是引用代码的扩展,它返回 本地(启动并运行)接口为 (接口名称、IP 地址、MAC 地址)的数组 元组。 MAC 地址是从AF_LINK 类型的接口检索的 它们在接口列表中存储为sockaddr_dl 结构。 这是一个可变长度结构,Swift 的严格类型系统使得一些指针杂耍变得必要。

重要提示:此代码旨在在 Mac 计算机上运行。 在 ios 设备上获取 MAC 地址不起作用。 iOS 有意返回“02:00:00:00:00:00”作为硬件地址 对于出于隐私原因的所有接口,请参见例如 Trouble with MAC address in iOS 7.0.2.)

func getInterfaces() -> [(name : String, addr: String, mac : String)] 

    var addresses = [(name : String, addr: String, mac : String)]()
    var nameToMac = [ String : String ]()

    // Get list of all interfaces on the local machine:
    var ifaddr : UnsafeMutablePointer<ifaddrs> = nil
    if getifaddrs(&ifaddr) == 0 

        // For each interface ...
        var ptr = ifaddr
        while ptr != nil 
            defer  ptr = ptr.memory.ifa_next 

            let flags = Int32(ptr.memory.ifa_flags)
            let addr = ptr.memory.ifa_addr

            if let name = String.fromCString(ptr.memory.ifa_name)  

                // Check for running IPv4, IPv6 interfaces. Skip the loopback interface.
                if (flags & (IFF_UP|IFF_RUNNING|IFF_LOOPBACK)) == (IFF_UP|IFF_RUNNING) 

                    if addr.memory.sa_family == UInt8(AF_LINK) 
                        // Get MAC address from sockaddr_dl structure and store in nameToMac dictionary:
                        let dl = UnsafePointer<sockaddr_dl>(ptr.memory.ifa_addr)
                        let lladdr = UnsafeBufferPointer(start: UnsafePointer<Int8>(dl) + 8 + Int(dl.memory.sdl_nlen),
                                                         count: Int(dl.memory.sdl_alen))
                        if lladdr.count == 6 
                            nameToMac[name] = lladdr.map  String(format:"%02hhx", $0).joinWithSeparator(":")
                        
                    

                    if addr.memory.sa_family == UInt8(AF_INET) || addr.memory.sa_family == UInt8(AF_INET6) 
                        // Convert interface address to a human readable string:
                        var hostname = [CChar](count: Int(NI_MAXHOST), repeatedValue: 0)
                        if (getnameinfo(addr, socklen_t(addr.memory.sa_len), &hostname, socklen_t(hostname.count),
                            nil, socklen_t(0), NI_NUMERICHOST) == 0) 
                            if let address = String.fromCString(hostname) 
                                addresses.append( (name: name, addr: address, mac : "") )
                            
                        
                    
                
            
        
        freeifaddrs(ifaddr)
    

    // Now add the mac address to the tuples:
    for (i, addr) in addresses.enumerate() 
        if let mac = nameToMac[addr.name] 
            addresses[i] = (name: addr.name, addr: addr.addr, mac : mac)
        
    

    return addresses

你必须添加

#include <ifaddrs.h>
#include <net/if_dl.h>

到桥接头文件进行编译。

示例用法:

for addr in getInterfaces() 
   print(addr)

// ("en0", "fe80::1234:7fff:fe2e:8765%en0", "a9:55:6f:2e:57:78")
// ("en0", "192.168.2.108", "a9:55:6f:2e:57:78")
// ...

Swift 3 (Xcode 8) 更新:

func getInterfaces() -> [(name : String, addr: String, mac : String)] 

    var addresses = [(name : String, addr: String, mac : String)]()
    var nameToMac = [ String: String ]()

    // Get list of all interfaces on the local machine:
    var ifaddr : UnsafeMutablePointer<ifaddrs>?
    guard getifaddrs(&ifaddr) == 0 else  return [] 
    guard let firstAddr = ifaddr else  return [] 

    // For each interface ...
    for ptr in sequence(first: firstAddr, next:  $0.pointee.ifa_next ) 
        let flags = Int32(ptr.pointee.ifa_flags)
        if let addr = ptr.pointee.ifa_addr 
            let name = String(cString: ptr.pointee.ifa_name)

            // Check for running IPv4, IPv6 interfaces. Skip the loopback interface.
            if (flags & (IFF_UP|IFF_RUNNING|IFF_LOOPBACK)) == (IFF_UP|IFF_RUNNING) 
                switch Int32(addr.pointee.sa_family) 
                case AF_LINK:
                    // Get MAC address from sockaddr_dl structure and store in nameToMac dictionary:
                    addr.withMemoryRebound(to: sockaddr_dl.self, capacity: 1)  dl in
                        dl.withMemoryRebound(to: Int8.self, capacity: 8 + Int(dl.pointee.sdl_nlen + dl.pointee.sdl_alen)) 
                            let lladdr = UnsafeBufferPointer(start: $0 + 8 + Int(dl.pointee.sdl_nlen),
                                                             count: Int(dl.pointee.sdl_alen))
                            if lladdr.count == 6 
                                nameToMac[name] = lladdr.map  String(format:"%02hhx", $0).joined(separator: ":")
                            
                        
                    
                case AF_INET, AF_INET6:
                    // Convert interface address to a human readable string:
                    var hostname = [CChar](repeating: 0, count: Int(NI_MAXHOST))
                    if (getnameinfo(addr, socklen_t(addr.pointee.sa_len),
                                    &hostname, socklen_t(hostname.count),
                                    nil, socklen_t(0), NI_NUMERICHOST) == 0) 
                        let address = String(cString: hostname)
                        addresses.append( (name: name, addr: address, mac : "") )
                    
                default:
                    break
                
            
        
    

    freeifaddrs(ifaddr)

    // Now add the mac address to the tuples:
    for (i, addr) in addresses.enumerated() 
        if let mac = nameToMac[addr.name] 
            addresses[i] = (name: addr.name, addr: addr.addr, mac : mac)
        
    

    return addresses

【讨论】:

感谢您花时间研究它。我很感激。很抱歉在使用您的代码时没有将您的积分添加到我的问题中。 此代码旨在让所有设备连接到 wifi 网络或我的本地机器信息? @GuiSoySauce:这适用于本地机器上的接口(参考代码***.com/questions/25626117/…和您上一个问题中的代码chrisandtennille.com/code/IPAddress.c也是如此)。 @Martin R 如何在 swift 框架的模块映射中添加 if_dl.h 出现此错误:“必须从模块 'Darwin.POSIX.sys.types' 之前导入'u_char'声明是必需的“ @Martin R 我解决了这个问题,我在这个答案链接中添加了:***.com/questions/39036255/…【参考方案2】:

如果你将在框架中使用它,你必须在其中添加一个带有此配置的 .modelmap 文件

module ifaddrs [system]  [extern_c] 
    header
    "/Applications/Xcode_7.3.1.app/Contents/Developer/Platforms/iPhoneOS.platform/Developer/SDKs/iPhoneOS.sdk/usr/include/ifaddrs.h"
    export *


module net [system]  [extern_c] 

     module types 
         header "/usr/include/sys/types.h"
         export *
     

     module if_dl 

         header
         "/usr/include/net/if_dl.h"
         export *
     

 

然后在你的 .swift 文件中

import ifaddrs
import net.if_dl

【讨论】:

以上是关于管理 ifaddrs 以在 Swift 中返回 MAC 地址的主要内容,如果未能解决你的问题,请参考以下文章

从 json 会话返回值以在会话外使用

无法重新生成桥头以在目标 C 中使用 swift

swift:重新加载视图以在图表中显示新数据

下拉以在 swift UI 中刷新数据

如何重新加载 UIPageViewController 以在 Swift 中重新加载其视图

更改约束以在 swift iOS 中扩展视图