获取接口的最新全局ipv6地址

时间:2018-08-29 15:19:40

标签: python ipv6

我正在尝试获取python接口的当前公共ipv6地址。通过搜索,我发现了netifaces软件包,该软件包可以为我提供设备的所有 ipv6地址列表。

这将包括

  • 本地链接地址(我可以用一个简单的技巧将其过滤掉,虽然不太好,但是...)
  • 多个全局ipv6地址,其中之一是当前地址

现在我所说的电流是什么?通过隐私扩展(RFC 4941),定期将新地址分配给接口。旧的全局ipv6地址仍然有效,但标记为已弃用。我想要这个最新的。

这就是我尝试过的(出于安全原因,混淆了地址的一部分):

>>> import netifaces 
>>> addrs = netifaces.ifaddresses('en0')
>>> for i in addrs[netifaces.AF_INET6]:
...     print(i)
...
{'addr': 'fe80::caa:452a:b8cb:862e%en0', 'netmask': 'ffff:ffff:ffff:ffff::/64', 'flags': 1024}
{'addr': '2a02:____:____:9f1:cd8:e__e:e57b:eaa8', 'netmask': 'ffff:ffff:ffff:ffff::/64', 'flags': 1088}
{'addr': '2a02:____:____:9f1:3489:b__7:fb34:101', 'netmask': 'ffff:ffff:ffff:ffff::/64', 'flags': 192}

很明显,第一个是本地链接地址。第二个和第三个是全局标志,它们有一个标志,我不知道这意味着什么。

是否有任何方法(也许通过标志或python中的另一种方法)找出当前(临时的,不推荐使用的)全局地址?

在Linux中,我知道我可以解析ip -o -6 addr show dev interface的输出并解析该输出,但是那样行不通,例如在Mac上,如果有python软件包,我不喜欢解析其他工具的输出。

编辑: 这些标志似乎并非在每个OS上都存在。上面的输出是在Mac上。这是同一网络中Linux计算机上的输出(正常运行时间更长):

{'addr': '2a02:____:____:9f1:940c:f__f:f9c0:c274', 'netmask': 'ffff:ffff:ffff:ffff::/64'}
{'addr': '2a02:____:____:9f1:395a:6__d:650a:3a11', 'netmask': 'ffff:ffff:ffff:ffff::/64'}
{'addr': '2a02:____:____:9f1:d08a:6__d:d967:d28a', 'netmask': 'ffff:ffff:ffff:ffff::/64'}
{'addr': '2a02:____:____:9f1:9d20:b__7:3edc:96a4', 'netmask': 'ffff:ffff:ffff:ffff::/64'}
{'addr': '2a02:____:____:9f1:b584:d__b:16bf:5dd2', 'netmask': 'ffff:ffff:ffff:ffff::/64'}
{'addr': '2a02:____:____:9f1:61ba:d__b:78d8:197f', 'netmask': 'ffff:ffff:ffff:ffff::/64'}
{'addr': '2a02:____:____:9f1:c1df:7__7:236d:4219', 'netmask': 'ffff:ffff:ffff:ffff::/64'}
{'addr': '2a02:____:____:9f1:d021:4__3:1a44:4947', 'netmask': 'ffff:ffff:ffff:ffff::/64'}
{'addr': 'fe80::17ed:fbf5:a707:2a42%enp4s0', 'netmask': 'ffff:ffff:ffff:ffff::/64'}

编辑2:

由于“当前”地址的含义有些混乱,所以这就是我的意思。在上一个示例的Linux计算机上:

% ip -o -6 addr show dev enp4s0
2: enp4s0    inet6 2a02:____:____:9f1:940c:f__f:f9c0:c274/64 scope global temporary dynamic \       valid_lft 530760sec preferred_lft 12247sec
2: enp4s0    inet6 2a02:____:____:9f1:395a:6__d:650a:3a11/64 scope global temporary deprecated dynamic \       valid_lft 444473sec preferred_lft 0sec
2: enp4s0    inet6 2a02:____:____:9f1:d08a:6__d:d967:d28a/64 scope global temporary deprecated dynamic \       valid_lft 358187sec preferred_lft 0sec
2: enp4s0    inet6 2a02:____:____:9f1:9d20:b__7:3edc:96a4/64 scope global temporary deprecated dynamic \       valid_lft 271901sec preferred_lft 0sec
2: enp4s0    inet6 2a02:____:____:9f1:b584:d__b:16bf:5dd2/64 scope global temporary deprecated dynamic \       valid_lft 185616sec preferred_lft 0sec
2: enp4s0    inet6 2a02:____:____:9f1:61ba:d__b:78d8:197f/64 scope global temporary deprecated dynamic \       valid_lft 99330sec preferred_lft 0sec
2: enp4s0    inet6 2a02:____:____:9f1:c1df:7__7:236d:4219/64 scope global temporary deprecated dynamic \       valid_lft 13044sec preferred_lft 0sec
2: enp4s0    inet6 2a02:____:____:9f1:d021:4__3:1a44:4947/64 scope global dynamic mngtmpaddr noprefixroute \       valid_lft 2591735sec preferred_lft 604535sec
2: enp4s0    inet6 fe80::17ed:fbf5:a707:2a42/64 scope link noprefixroute \       valid_lft forever preferred_lft forever

很明显,所有“临时”地址都是有效地址。但是只有一个未被标记为不推荐使用。该地址将用于出站连接。这就是我想要的。

2 个答案:

答案 0 :(得分:0)

如果你想要一个公共 IPv6 地址,那么试试这个。

socket.getaddrinfo(socket.gethostname(), None, socket.AF_INET6)[1][4][0]

答案 1 :(得分:-1)

socket.getaddrinfo(socket.gethostname(), None, socket.AF_INET6)
socket.getaddrinfo(socket.gethostname(), None, socket.AF_INET6)[0][4][0]