gpt4 book ai didi

c++ - 不支持 C/C++ getnameinfo ai_family

转载 作者:太空宇宙 更新时间:2023-11-04 07:01:13 26 4
gpt4 key购买 nike

我使用一个库来做一些网络工作,当客户端连接时,这个库提供了一个“struct sockaddr *”来保存客户端套接字。我只是想提取 IP 和端口,目前我是这样做的:

#include <sys/socket.h>
#include <netdb.h>

const std::string Client::prepareIPandPort(struct sockaddr *hostaddr) {
assert(hostaddr != nullptr);

std::string ipport;
char clienthost[NI_MAXHOST];
char clientport[NI_MAXSERV];
int result = getnameinfo(hostaddr, sizeof(*hostaddr),
clienthost, sizeof(clienthost),
clientport, sizeof(clientport),
NI_NUMERICHOST | NI_NUMERICSERV);

if (result != 0) {
std::cerr << "Error: " << gai_strerror(result) << std::endl;
ipport = "unknown";
} else {
switch (hostaddr->sa_family) {
case AF_INET:
ipport = std::string {clienthost} + ":"
+ std::string {clientport};
break;
case AF_INET6:
ipport = "[" + std::string {clienthost} + "]:"
+ std::string {clientport};
break;
default:
ipport = "unknown";
}
}

return ipport;
}

在我的 Mac 上使用 IPv4 时,它可以正常工作。如果我在完全支持 IPv6 的 Gentoo Linux 服务器上使用这个应用程序,我只会得到:

错误:不支持 ai_family

连接客户端有 AAAA 和 IP6 记录。

我添加了一些 cout 并打印了 hostaddr->sa_family,它返回 10,即 AF_INET6。

为什么这不起作用? :-)

最佳答案

您不能使用 sizeof(*hostaddr) 因为 hostaddr 是一个通用的 sockaddr* 指针。不同的地址族使用不同的sockaddr_... 类型,这些类型与sockaddr 本身的大小不尽相同。 getnameinfo() 需要知道 sockaddr_... 结构的 true 大小,hostaddr 实际上指向, 基于其地址族。

根据 Linux getnameinfo() documentation :

EAI_FAMILY
The address family was not recognized, or the address length was invalid for the specified family

sockaddr_in (IPv4) 与 sockaddr 大小相同,这就是为什么 getnameinfo() 对 IPv4 “有效”。但是 sockaddr_in6 (IPv6) 比 sockaddr 大,这就是 getnameinfo() 失败的原因。

最好的解决方案是让调用者传入正确的尺寸:

const std::string Client::prepareIPandPort(struct sockaddr *hostaddr, int hostaddrlen) {
...
int result = getnameinfo(hostaddr, hostaddrlen, ...);
...
}

sockaddr_in ipv4host;
...
client.prepareIPandPort((sockaddr*)&ipv4host, sizeof(ipv4host));

sockaddr_in6 ipv6host;
...
client.prepareIPandPort((sockaddr*)&ipv6host, sizeof(ipv6host));

否则,你必须计算它:

const std::string Client::prepareIPandPort(struct sockaddr *hostaddr) {
...
int hostaddrlen;
switch (hostaddr->sa_family) {
case AF_INET:
hostaddrlen = sizeof(sockaddr_in);
break;
case AF_INET6:
hostaddrlen = sizeof(sockaddr_in6);
break;
default:
std::cerr << "Error: " << gai_strerror(EAI_FAMILY) << std::endl;
return "unknown";
}
int result = getnameinfo(hostaddr, hostaddrlen, ...);
...
}

关于c++ - 不支持 C/C++ getnameinfo ai_family,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38132577/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com