C++ 套接字:当客户端调用 connect() 时,accept() 挂起,但 accept() 响应 HTTP GET 请求

Posted

技术标签:

【中文标题】C++ 套接字:当客户端调用 connect() 时,accept() 挂起,但 accept() 响应 HTTP GET 请求【英文标题】:C++ sockets: accept() hangs when client calls connect(), but accept() responds to HTTP GET request 【发布时间】:2022-01-17 11:17:37 【问题描述】:

我正在尝试用 C++ 编写一个演示服务器/客户端程序。我首先在我的 Macbook 上运行我的服务器程序,打开 ngrok 并将 Internet 上的公共地址转发到我机器上的本地地址。当我尝试运行我的客户端程序以连接到服务器时,我看到了一些我不理解的东西:

    如果客户端尝试连接到服务器程序中定义的本地端口的localhost,则服务器按预期接受并且客户端成功连接, 如果客户端尝试连接到 ngrok 服务器地址的 80 端口(默认为 ngrok),则客户端连接,但服务器在接受调用时仍被阻止。 (这个我不明白!) 如果我向 ngrok 服务器地址发送 HTTP GET 请求,则服务器成功接受连接。

为什么我会看到这些?在理想情况下,我希望我的服务器接受来自我的客户端程序的连接,而不仅仅是响应 HTTP GET 请求。

如果有帮助,这是我的代码:对于客户,

#include "helpers.hh"
#include <cstdio>
#include <netdb.h>

// usage: -h [host] -p [port]
int main(int argc, char** argv) 
    const char* host = "x.xx.xx.xx"; // use the server's ip here.
    const char* port = "80";

    // parse arguments
    int opt;
    while ((opt = getopt(argc, argv, "h:p:")) >= 0) 
        if (opt == 'h') 
            host = optarg;
         else if (opt == 'p') 
            port = optarg;
        
    

    // look up host and port
    struct addrinfo hints, *ais;
    memset(&hints, 0, sizeof(hints));
    hints.ai_family = AF_UNSPEC;        // use IPv4 or IPv6
    hints.ai_socktype = SOCK_STREAM;    // use TCP
    hints.ai_flags = AI_NUMERICSERV;
    if (strcmp(host, "ngrok") == 0) 
        host = "xxxx-xxxx-xxxx-1011-2006-00-27b9.ngrok.io";
    
    int r = getaddrinfo(host, port, &hints, &ais);
    if (r != 0) 
        fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(r));
        exit(1);
    

    // connect to server
    int fd = -1;
    for (auto ai = ais; ai && fd < 0; ai = ai->ai_next) 
        fd = socket(ai->ai_family, ai->ai_socktype, 0);
        if (fd < 0) 
            perror("socket");
            exit(1);
        

        r = connect(fd, ai->ai_addr, ai->ai_addrlen);
        if (r < 0) 
            close(fd);
            fd = -1;
        
    
    if (fd < 0) 
        perror("connect");
        exit(1);
    
    freeaddrinfo(ais);

    // 
    printf("Connection established at fd %d\n", fd);
    FILE* f = fdopen(fd, "a+");
    fwrite("!", 1, 1, f);
    fclose(f);
    while (true) 
    
    

对于服务器:

#include "helpers.hh"

void handle_connection(int cfd, std::string remote) 
    (void) remote;
    printf("Received incoming connection at cfd: %d\n", cfd);
    usleep(1000000);
    printf("Exiting\n");



int main(int argc, char** argv) 
    int port = 6162;
    if (argc >= 2) 
        port = strtol(argv[1], nullptr, 0);
        assert(port > 0 && port <= 65535);
    

    // Prepare listening socket
    int fd = open_listen_socket(port);
    assert(fd >= 0);
    fprintf(stderr, "Listening on port %d...\n", port);

    while (true) 
        struct sockaddr addr;
        socklen_t addrlen = sizeof(addr);

        // Accept connection on listening socket
        int cfd = accept(fd, &addr, &addrlen);
        if (cfd < 0) 
            perror("accept");
            exit(1);
        

        // Handle connection
        handle_connection(cfd, unparse_sockaddr(&addr, addrlen));
    

【问题讨论】:

老实说,我认为问题在于我的客户端程序从未真正能够正确定位 ngrok 生成的地址。在我的代码中,我试图让客户端程序使用 getaddrinfo 设置主机和端口,而这一次客户端甚至没有连接就卡住了。也许有人应该告诉我如何正确使用 ngrok... 【参考方案1】:

与在本地路由器中完成的典型端口转发相反,ngrok 不是传输级别 (TCP) 的端口转发器,而是 HTTP 级别的请求转发器。

因此,如果客户端通过 TCP 连接到外部 ngrok 服务器,则不会转发任何内容。只有在客户端发送 HTTP 请求后,目的地才会被确定,然后这个请求将被发送到内部机器上的 ngrok 连接器,然后它会启动到内部服务器的连接并转发请求。

【讨论】:

该死!谢谢,这实际上很有意义。抱歉,我对这些概念还是很陌生,因为我实际上是在参加系统编程课程而不是网络课程之后编写这些代码......所以,根据你的回答,我可以在连接后发送 HTTP 请求() 从客户端?也许用一些允许 TCP 级别转发的工具替换 ngrok 对我来说可能更容易?我可以在这里使用什么来使我的代码在本地网络之外工作? - 呃,也许我应该打开一个新问题或在线搜索后一个问题。 顺便说一下,我用的是大学网络,所以恐怕我无法篡改这里的路由器:( @Macrophage:例如,请参阅Poor man's ngrok with tcp proxy and ssh reverse tunnel,了解当您拥有具有 SSH 访问权限的外部服务器时如何进行 TCP 级别转发。

以上是关于C++ 套接字:当客户端调用 connect() 时,accept() 挂起,但 accept() 响应 HTTP GET 请求的主要内容,如果未能解决你的问题,请参考以下文章

C++ 套接字连接问题

connect函数

C++ 套接字 recv() 系统调用返回 -1

TCP套接字编程常用函数

listen函数

TCP客户端服务器编程模型