gpt4 book ai didi

c++ - 无法从本地网络外部连接到C++ Linux套接字程序

转载 作者:太空宇宙 更新时间:2023-11-04 04:58:23 25 4
gpt4 key购买 nike

我有一个 ubuntu Linux 服务器,它运行一个 C++ 套接字程序,该程序应该在端口 6555 上监听传入连接并发送“欢迎来到服务器!”给他们。当我从本地主机远程登录时它可以工作,但是当我从网络外部远程登录时它不起作用。

本地网络管理员告诉我,他已将端口 6555 转发到服务器的本地 IP,就像他之前对 80 所做的那样。

我还有一个在 80 端口上运行的 Web 服务器,我可以从任何地方远程登录到该服务器,所以我不知道问题是什么。

我尝试暂时禁用服务器上的 iptables (ufw) 防火墙并尝试连接,但没有任何区别。

打开端口检查器表示端口已关闭。这是否意味着转发不起作用?这是用 g++ 编译的代码:

#include <iostream>
#include <sys/types.h>
#include <sys/socket.h>
#include <netdb.h>
#include <cstring>
#include <netinet/in.h>
#include <unistd.h>

#include "../include/netutils.hh"

int main() {
sockaddr_storage clt_addr;
socklen_t sin_size;
int cltSock = -1;
const std::string port = "6555";
int servSock = -1;
try {
servSock = getSrvSocket(port, 10);
while (true) {
sin_size = sizeof clt_addr;
cltSock = accept(servSock, (sockaddr*)&clt_addr, &sin_size);
if (cltSock == -1) continue;
if (!fork()) {
close(servSock);
send(cltSock, "Welcome to the server!\n", 23, 0);
while(true);
}
close(cltSock);
}

}
catch(std::string excpt)
{
std::cerr << excpt << '\n';
return -1;
}
return 0;

}

在 netutils.cc 中使用 getSrvSocket():

int getSrvSocket(std::string port, int BACKLOG) {
int yes = 1;
addrinfo *p;
addrinfo hints;
addrinfo *servinfo;
memset(&hints, 0, sizeof hints);
hints.ai_family = AF_UNSPEC; //Can use either IPv4 or IPv6
hints.ai_socktype = SOCK_STREAM;
hints.ai_flags = AI_PASSIVE; //Automatically set IP address
int sock = -1;
if (getaddrinfo(NULL, port.c_str(), &hints, &servinfo) !=0) {
throw std::string("Error: Could not resolve address info");
}

for (p = servinfo; p != NULL; p = p->ai_next) {
sock = socket(p->ai_family, p->ai_socktype, p->ai_protocol);
if (sock == -1) continue;
if (setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &yes,
sizeof(int)) == -1) {
throw std::string("Error: setsockopt() failure");
}
if (bind(sock, p->ai_addr, p->ai_addrlen) == -1) {
close(sock);
continue;
}
break;
}
freeaddrinfo(servinfo);
if (NULL == p) throw std::string("Error: Server failed to bind");
if (listen(sock, BACKLOG) == -1) {
throw std::string("Error: Failed to to start listen()");
}
struct sigaction sa;
sa.sa_handler = sigchld_handler;
sigemptyset(&sa.sa_mask);
sa.sa_flags = SA_RESTART;

if (sigaction(SIGCHLD, &sa, NULL) == -1) {
throw std::string("Error: sigaction() failed");
}
return sock;
}

最佳答案

getSrvSocket()在调用getaddrinfo()时指定AF_UNSPEC,这意味着它可以输出IPv4和IPv6地址。但你只绑定(bind)到一个地址,然后就停止了。因此,您可能实际上绑定(bind)到 IPv6 地址,然后您的客户端尝试使用 IPv4 连接到它(反之亦然),这显然是行不通的。

如果您想同时支持 IPv4 和 IPv6,则应将单独的监听套接字绑定(bind)到 getaddrinfo() 返回的每个地址。您还应该记录成功绑定(bind)的地址,以便您知道客户端可以连接到哪些地址(以及您需要告诉网络管理员转发到哪些地址)。

否则,请勿将 AF_UNSPECgetaddrinfo() 一起使用。请改用 AF_INET (IPv4) 或 AF_INET6 (IPv6)。

关于c++ - 无法从本地网络外部连接到C++ Linux套接字程序,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40273719/

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