UDP上的NAT失败



我有此代码:

#include <cstring>
#include <cstdio>
#include <cerrno>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <string>
#include <stdlib.h>
#include <unistd.h>
#include <cstdlib>
#include <iostream>
#include <thread>
#include "STUN.h"
bool do_work = true;
using namespace std;
void receiver(int sock) {
    //boost::asio::io_service& s
    sockaddr_in from_adr;
    std::cout << "START: receivern";
    size_t blen = 2048;
    char * buf = new char[blen];
    unsigned slen = sizeof(from_adr);
    if (recvfrom(sock, (void *) buf, blen, 0, (sockaddr*) &from_adr, &slen)
            < 0) {
        std::cout << "Recv problemn";
        exit(0);
    }
    do_work = false;
    std::cout << "STOP: receivern";
}
void sender(sockaddr_in * to, int sock) {
    std::cout << "START: sendern";
    while (do_work) {
        if (sendto(sock, "OK", 3, 0, (sockaddr*) to, sizeof(sockaddr)) < 0) {
            std::cout << "SEND PROBLEMn";
            return;
        }
        sleep(1);
    }
    std::cout << "RECEIVED!!!nnn";
    exit(0);
}
int main(int argc, char* argv[]) {
    if (argc != 2) {
        std::cerr << "Usage: udp-flood-test <port>n";
        return 1;
    }
    ippd stun_adr;
    stun_adr.addr = "64.233.161.127";
    stun_adr.port = 19302;
    STUN stun(stun_adr);
    std::string eip;
    uint16_t epp;
    uint16_t lp = std::atoi(argv[1]);
    ippd sti = stun.getSTUN(lp);
    std::cout << "External address: " << sti.addr << ":" << sti.port << "n";
    std::cout << "Internal address: " << "127.0.0.1:" << lp << "n";
    std::cout << "Write external ip: ";
    std::cin >> eip;
    std::cout << "n";
    std::cout << "Write external port: ";
    std::cin >> epp;
    std::cout << "n";
    std::cout << "OK. I'll send data to " << eip << ":" << epp << "n";
    int m_sock;                     // дескриптор сокета
    sockaddr_in m_addr;             // переменная адреса интерфейса
    std::string my_addr = "0.0.0.0";             // Адрес локального интерфейса
    /* Создание сокета и присвоение значения дескриптору сокета для UDP пакетов
     * PF_INET    - IP protocol family
     * SOCK_DGRAM - Raw protocol interface */
    if ((m_sock = socket(PF_INET, SOCK_DGRAM, 0)) < 0) {
        perror("socket");
        return -1;
    }
    /* Обнуляем переменную m_addr и забиваем её нужными значениями */
    memset(&m_addr, 0, sizeof(m_addr));
    m_addr.sin_family = AF_INET;  // обязательно AF_INET!
    m_addr.sin_port = htons(lp); // 0 - выдать порт автоматом
    /* Переводим адрес в нужный нам формат */
    if (inet_aton(my_addr.c_str(), &m_addr.sin_addr) == 0) {
        perror("inet_aton");
        close(m_sock);
        return -1;
    }
    /* Биндим сокет */
    if (bind(m_sock, (struct sockaddr*) &m_addr, sizeof(m_addr)) < 0) {
        perror("bind");
        close(m_sock);
        return -1;
    }
    sockaddr_in to_addr;
    memset(&to_addr, 0, sizeof(to_addr));
    to_addr.sin_family = AF_INET;
    to_addr.sin_port = htons(epp);
    if (inet_aton(eip.c_str(), &to_addr.sin_addr) == 0) {
        perror("inet_aton");
        close(m_sock);
        return -1;
    }
    std::thread inp(receiver, m_sock);
    std::thread oup(sender, &to_addr, m_sock);
    oup.detach();
    inp.detach();
    while (1) {
        sleep(1);
    }
    return 0;
}

这是一个简单的测试程序,可以通过Stun获得外部IP和端口,并将数据从对等方发送到对等B(以及从B到A(。

因此,如果我将IP和端口设置为127.0.0.1:port或local_network_ip:port。

,程序可以正常工作。

如果我将收到的IP设置为从Stun Server对等方A将UDP数据包发送给对等B,并且B将将数据包发送到A。A和B没有任何传入的数据包。在Wireshark中,我也可以看到外向的数据包,但没有传入。

网络连接图:

a(带有主动测试程序的PC(-NAT(HOME Router( - ... -NAT(Internet提供商( - Internet -Nat(另一个Internet提供商( - ... -Nat(另一个家庭路由器( - B(另一台具有主动测试程序的PC(

也许我做错了什么?..

在哪里可以找到问题?

问题是双nat。家庭路由器(来自网络图(如果您没有配置的端口转发,则拒绝数据包。

按照以下步骤在双重NAT上工作:

  1. 选择要接收数据包的本地端口。

  2. 尝试绑定端口。(如果失败,请返回步骤1(

  3. 使用NAT-PMP或UPNP IGD配置端口转发端口。

  4. 使用Stun获得外部IP和端口

  5. 从步骤4共享数据到另一个程序实例

  6. 发送和/或接收UDP数据包

最新更新