我有此代码:
#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(
-
使用NAT-PMP或UPNP IGD配置端口转发端口。
-
使用Stun获得外部IP和端口
-
从步骤4共享数据到另一个程序实例
-
发送和/或接收UDP数据包