Получить все IP-адреса Ma c, используя Objective- C - PullRequest
1 голос
/ 04 февраля 2020

Я пытаюсь получить все IP-адреса моего Ма c (IPv4 + IPv6), но не вижу ожидаемых результатов. Я использую код из этого сообщения о переполнении стека , чтобы получить массив всех IP-адресов, но отсутствует IP-адрес.

Я использую другой Ma c, чтобы поделиться им inte rnet и создайте сеть NAT64 для Документация Apple по тестированию / поддержке сетей IPv6 .

Например, вот что говорит Системные настройки → Сетевая панель: мой IP-адрес :

enter image description here

... но я вижу, что на самом деле у меня есть два адреса IPv6 при дальнейшей проверке:

enter image description here

... но возвращается только один из них:

"169.254.38.213",
"2001:2:0:aab1:d0ef:646d:f22a:5d83",
"127.0.0.1"

.. при использовании этого:

#include <ifaddrs.h>
#include <arpa/inet.h>
#include <net/if.h>

@interface AppDelegate ()

#define IP_ADDR_IPv4    @"ipv4"
#define IP_ADDR_IPv6    @"ipv6"

@end

@implementation AppDelegate

- (void) applicationDidFinishLaunching: (NSNotification *) aNotification
{
    NSMutableDictionary *addresses = [NSMutableDictionary dictionaryWithCapacity:8];
    NSMutableArray *ipAddressesArray = [[NSMutableArray alloc] init];

    // retrieve the current interfaces - returns 0 on success
    struct ifaddrs *interfaces;

    if (!getifaddrs(&interfaces))
    {
        // Loop through linked list of interfaces
        struct ifaddrs *interface;

        for (interface=interfaces; interface; interface=interface->ifa_next)
        {
            if (!(interface->ifa_flags & IFF_UP) /* || (interface->ifa_flags & IFF_LOOPBACK) */)
            {
                continue; // deeply nested code harder to read
            }

            const struct sockaddr_in *addr = (const struct sockaddr_in*)interface->ifa_addr;
            char addrBuf[ MAX(INET_ADDRSTRLEN, INET6_ADDRSTRLEN) ];

            if (addr && (addr->sin_family==AF_INET || addr->sin_family==AF_INET6))
            {
                NSString *name = [NSString stringWithUTF8String:interface->ifa_name];
                NSString *type;

                if (addr->sin_family == AF_INET)
                {
                    if (inet_ntop(AF_INET, &addr->sin_addr, addrBuf, INET_ADDRSTRLEN))
                    {
                        type = IP_ADDR_IPv4;
                    }
                }
                else
                {
                    const struct sockaddr_in6 *addr6 = (const struct sockaddr_in6*)interface->ifa_addr;

                    if (inet_ntop(AF_INET6, &addr6->sin6_addr, addrBuf, INET6_ADDRSTRLEN))
                    {
                        type = IP_ADDR_IPv6;
                    }
                }

                if (type)
                {
                    NSString *key = [NSString stringWithFormat:@"%@/%@", name, type];
                    addresses[key] = [NSString stringWithUTF8String:addrBuf];
                }
            }
        }

        for (id key in addresses)
        {
            if (![[addresses valueForKey:key] hasPrefix:@"fe80"])
            {
                [ipAddressesArray addObject:[addresses valueForKey:key]];
            }
        }

        // Free memory
        freeifaddrs(interfaces);
    }

    NSLog(@"%@", ipAddressesArray);
}

Есть идеи, что здесь происходит? Это имеет значение? Я пытаюсь выполнить другой код условно на основе сопоставления IP-адресов. Было бы одно дело, если бы единственным возвращенным IPv6-адресом был тот, который отображался пользователю в Системных настройках при первом открытии панели «Сеть», но вместо этого единственным возвращаемым IPv6-адресом был «скрытый», который нужно копать в Advanced. ... раздел, чтобы найти. Заранее спасибо.

1 Ответ

1 голос
/ 10 февраля 2020

Ответ здесь таков: я идиот. Как предложил Роб Нейпир в своем комментарии к моему вопросу, я, по сути, отфильтровывал пропущенные IP-адреса:

NSString *key = [NSString stringWithFormat:@"%@/%@", name, type];
addresses[key] = [NSString stringWithUTF8String:addrBuf];

Каждый интерфейс может иметь более одного IP-адреса, но я, поскольку я использовал тип интерфейса в качестве уникального ключа в моем словаре addresses в словаре присутствовал только один IP-адрес на интерфейс. Я исправил это, возвращая массив словарей, а не один словарь:

if (![[NSString stringWithFormat:@"%s", addrBuf] hasPrefix:@"fe80"])
{
      NSDictionary *addressDict = [NSDictionary dictionaryWithObjectsAndKeys :
                                                   [NSString stringWithFormat:@"%@/%@", name, type], @"Interface",
                                                   [NSString stringWithFormat:@"%s", addrBuf], @"Address",
                                                    nil];

     [addresses addObject:addressDict];
}

return addresses;
...