使用带有AI_PASSIVE的getaddrinfo()

时间:2011-11-13 18:49:24

标签: sockets serversocket getaddrinfo

getaddrinfo()函数不仅允许客户端程序有效地找到用于创建给定主机的套接字的正确数据,而且还允许服务器绑定到正确的套接字 - 理论上。

我只是learned about that并开始通过Python玩它:

from socket import *
for i in getaddrinfo(None, 22, AF_UNSPEC, SOCK_STREAM, IPPROTO_IP, AI_PASSIVE): i

产量

(2, 1, 6, '', ('0.0.0.0', 22))
(10, 1, 6, '', ('::', 22, 0, 0))
是什么让我想知道是否有什么不对劲。

我应该对这些答案究竟做些什么?我应该

  • 制作所有这些答案的listen()套接字,或者我
  • 选择真正有效的第一个?

manpage中的示例建议我只接受第一个并且如果它没有错误就对它感到满意,但是我只能通过IPv4获得连接。

但是,如果我尝试所有这些,我不得不担心2个服务器套接字,这是不必要的,因为如果满足某些条件(OS,套接字标志等),IPv6服务器套接字也会侦听IPv4。 / p>

我在哪里想错了?


编辑:显然,我没想错,但我的电脑做错了。我使用OpenSUSE附带的默认/etc/gai.conf。如果有人能指出我朝着正确的方向前进,那就太好了。

编辑2:在给定的情况下,strace在阅读/etc/gai.conf之后在内部进行了以下调用(现在使用端口54321,因为我认为使用端口22可能会产生一些不良影响,这是不是这样的):

socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP) = 3
connect(3, {sa_family=AF_INET6, sin6_port=htons(54321), inet_pton(AF_INET6, "::", &sin6_addr), sin6_flowinfo=0, sin6_scope_id=0}, 28) = 0
getsockname(3, {sa_family=AF_INET6, sin6_port=htons(38289), inet_pton(AF_INET6, "::1", &sin6_addr), sin6_flowinfo=0, sin6_scope_id=0}, [28]) = 0
connect(3, {sa_family=AF_UNSPEC, sa_data="\0\0\0\0\0\0\0\0\0\0\0\0\0\0"}, 16) = 0
connect(3, {sa_family=AF_INET, sin_port=htons(54321), sin_addr=inet_addr("0.0.0.0")}, 16) = 0
getsockname(3, {sa_family=AF_INET6, sin6_port=htons(60866), inet_pton(AF_INET6, "::ffff:127.0.0.1", &sin6_addr), sin6_flowinfo=0, sin6_scope_id=0}, [28]) = 0
close(3)                                = 0

显然,该决定是根据getsockname()电话的结果进行的......

BTW:https://bugs.launchpad.net/ubuntu/+source/eglibc/+bug/673708以及其他提到的错误报告证实了我的观察结果。有几个人声称新行为是正确的,所以我显然坚持使用AF_INET6 ...: - (

2 个答案:

答案 0 :(得分:3)

由于某种原因,您的getaddrinfo返回了错误的结果。它应该首先返回IPv6套接字。我唯一能想到的是,如果您的操作系统检测到您的系统具有低prio IPv6(6to4或Teredo)并且避免它们,那么在这种情况下IMO就是错误的。编辑:刚刚注意到我自己的电脑做了同样的事情,我使用6to4。

但是,您可以同时收听它们,也可以使用AF_INET6代替AF_UNSPEC。然后你可以使用setsockopt来禁用IPV6_V6ONLY

getaddrinfo在这里做了合理的事情,并返回所有适用的结果(尽管顺序错误,正如我所提到的)。一个和两个侦听套接字都是有效的方法,具体取决于您的应用程序。

答案 1 :(得分:0)

JFTR:现在看来,联机帮助页中给出的程序是错误的。

有两种可能的方法可以监听这两种IP类型:

  1. 仅创建IPv6套接字并关闭仅限v6的标志:

    from socket import *
    s = socket(AF_INET6, SOCK_STREAM)
    s.setsockopt(IPPROTO_IPV6, IPV6_V6ONLY, 0)
    s.bind(...)
    

    RESP。

    from socket import *
    ai = getaddrinfo(None, ..., AF_INET6, SOCK_STREAM, 0, AI_PASSIVE)[0]
    s = socket(ai[0], ai[1], ai[2])
    s.setsockopt(IPPROTO_IPV6, IPV6_V6ONLY, 0)
    s.bind(ai[4])
    

    优点:

    • 更易于处理

    缺点:

    • 在XP(AFAIK)下不起作用 - 有两种不同的协议栈
  2. 使用两个套接字并打开v6only标志:

    from socket import *
    aii = getaddrinfo(None, ..., AF_UNSPEC, SOCK_STREAM, 0, AI_PASSIVE)
    sl = []
    for ai in aii:
        s = socket(ai[0], ai[1], ai[2])
        if ai[0] == AF_INET6: s.setsockopt(IPPROTO_IPV6, IPV6_V6ONLY, 1)
        s.bind(ai[4])
        sl.append(s)
    

    并处理sl中接受循环的所有套接字(使用select()或非阻塞IO来执行此操作)

    优点:

    • 使用({近乎)协议独立处理getaddrinfo()
    • 在XP下工作

    缺点:

    • 难以处理