如何使用套接字编程在两台远程计算机之间发送数据?

时间:2018-10-25 01:17:38

标签: c++ sockets tcp port firewall

我正在尝试将数据从一台远程计算机(Ubuntu)发送到我的家用计算机(High Sierra)。我在此处读过a couple of questions,但他们似乎并没有为我的问题提供解决方案。我使用ssh连接到远程计算机,并在两台计算机上创建并成功编译了以下程序。

client.cpp

#include <iostream>
#include <string>

#include <arpa/inet.h>
#include <unistd.h>
#include <errno.h>


int main(int argc, char* argv[])
{
    std::string address = argv[1];
    int port = 38473;
    int success;

    // http://man7.org/linux/man-pages/man2/socket.2.html
    int server_socket = socket(AF_INET, SOCK_STREAM, 0);
    if (server_socket == -1)
        return errno;

    sockaddr_in socket_address{};
    socket_address.sin_family = AF_INET;
    socket_address.sin_port = htons(port);  // htons - Host to Network Short. Flip endianness to match machine.
    success = inet_pton(AF_INET, address.c_str(), &socket_address.sin_addr);  // Pointer (to String) to Number.
    if (success <= 0)
        return errno;

    success = connect(server_socket, (sockaddr*)&socket_address, sizeof(socket_address));
    if (success == -1)
        return errno;

    std::cout << "Connected" << std::endl;
    for (int i = 0; i < 10; ++i)
        write(server_socket, "Hello!\n", 7);
}

server.cpp

#include <iostream>
#include <string>

#include <cstring>
#include <arpa/inet.h>
#include <unistd.h>
#include <errno.h>


int main(int argc, char* argv[])
{
    std::string address = "127.0.0.1";
    int port = 38473;
    int success;

    // http://man7.org/linux/man-pages/man2/socket.2.html
    int listen_socket = socket(AF_INET, SOCK_STREAM, 0);
    if (listen_socket == -1)
        return errno;

    sockaddr_in socket_address{};
    socket_address.sin_family = AF_INET;
    socket_address.sin_port = htons(port);  // htons - Host to Network Short. Flip endianness to match machine.
    success = inet_pton(AF_INET, address.c_str(), &socket_address.sin_addr);  // Pointer (to String) to Number.
    if (success <= 0)
        return errno;

    // http://man7.org/linux/man-pages/man2/bind.2.html
    success = bind(listen_socket, (sockaddr*)&socket_address, sizeof(socket_address));
    if (success == -1)
        return errno;

    // http://man7.org/linux/man-pages/man2/listen.2.html
    success = listen(listen_socket, 10);
    if (success == -1)
        return errno;

    sockaddr_in client_address;
    socklen_t   client_address_size = sizeof(client_address);
    int client_socket = accept(listen_socket, (sockaddr*)&client_address, &client_address_size);
    if (client_socket == -1)
        return errno;

    close(listen_socket);

    size_t buffer_size = 4096;
    char buffer[buffer_size];
    ssize_t bytes_received;

    std::cout << "Connected" << std::endl;
    while (true)
    {
        memset(buffer, 0, buffer_size);
        // http://man7.org/linux/man-pages/man2/recvmsg.2.html

        if ((bytes_received = read(client_socket, buffer, buffer_size)) <= 0)
            return 0;
        std::cout << "Message: " << buffer << std::endl;
    }
}

在同一台计算机上运行两个程序时(客户端可执行文件在命令行中以127.0.0.1作为参数),它可以完美工作并发送数据。尝试将本地计算机作为服务器运行并从远程运行client.cpp时,出现错误代码111(连接被拒绝)。

在阅读了各种教程/博客/帖子后,似乎是这样的:

  1. 该端口当前未在监听。
  2. 防火墙阻止了连接。

我已经尝试通过执行以下操作来解决此问题。

端口当前未在监听。

由于远程计算机正在尝试连接到我的公共IP地址,因此我必须将端口转发到本地计算机。我登录到路由器,并添加了一个条目,将端口38473转发到我的专用IP地址。该端口是随机选择的,因为iana将其标记为未分配,并且该端口很大。由于在一台计算机上运行两个程序时可以发送数据,因此我认为端口不是问题。

Port forwarding table

防火墙阻止了连接。

我已确保在本地计算机上禁用防火墙(在“系统偏好设置”->“安全和隐私”->“防火墙”中),并在路由器中输入了一个条目,以允许来自远程计算机的消息(使用其公共IP地址)通过端口38473。

Firewall configurations

我似乎仍然无法连接。我运行了Wireshark,可以看到遥控器确实发送了SYN请求,但返回了[RST,ACK]。 RST flag似乎是防火墙终止连接的一种方法,但是考虑到我之前的上述配置,我不明白为什么它会这样做。

Wireshark output

我在做什么错了?

1 个答案:

答案 0 :(得分:1)

调整服务器代码以侦听计算机上您感兴趣的以太网卡的IP,或者:

   serv_addr.sin_family = AF_INET;
   serv_addr.sin_addr.s_addr = INADDR_ANY;
   serv_addr.sin_port = htons(portno);