编程语言
首页 > 编程语言> > VC++实现NAT穿透之NAT类型检测

VC++实现NAT穿透之NAT类型检测

作者:互联网

   VC++实现NAT穿透之NAT类型检测    

  // NatCheck.cpp : Defines the entry point for the console application.
//

#include "stdafx.h"   
#include

#define N_SuperNode 5
#define N_NatCheckRequestQueue 20
#define PORT_SERVER_NATCHECK 8888
#define PORT_SERVER_NATCHECK_1 8889
#define PORT_CLIENT_NATCHECK 7777
#define PORT_SERVER_CLIENT_INFO 4444
#define N_CheckTime 5 //最多检测次数

SOCKET sClient,sServer;//客户端和服务端的socket号
struct sockaddr_in sn1_addr,sn2_addr,sn_addr,client_addr;
struct sockaddr_in client_addr_deal;
struct in_addr in_addr_tmp,in_addr_tmp1;
FILE *fp; //客户端读取hostcache.txt文件指针
int i=0,j=0;
unsigned long int snIP[N_SuperNode];//读取的SN的IP地址列表
unsigned short int snPort[N_SuperNode];//读取的SN的Port列表
char tmpIP[15],tmpPort[5];
char ch;
char IP_return[15];
/* 本client的网络类型:
* 0 = public net
* 1 = full cone nat
* 2 = symmetric nat
* 3 = restricted cone nat
* 4 = port restricted cone nat
*/
unsigned long int myID=0,friendID=26;//本client和friend的用户ID
char myNetType=-1,friendNetType=-1;
unsigned long int myPubIp,friendPubIp,myPriIp,friendPriIp;
unsigned short int myPubPort,friendPubPort,myPriPort,friendPriPort;
//natCheck请求包结构
struct natCheckPacket{
 char version;
 char dowhat;
 unsigned short int port;
 unsigned long int ip;
 unsigned long int id;
}natCheckPacket;
typedef struct natCheckPacket NCP;
char buff[sizeof(natCheckPacket)];//发包用的缓冲区
//natCheck请求数据结构
struct natCheckRequest{
 NCP ncp;//缓存到来的请求包
 struct natCheckRequest * next;//指向下一个请求数据
 unsigned long int ip;//发送请求的client IP地址
 unsigned short int port;//发送请求的client Port
}natCheckRequest;
typedef struct natCheckRequest NCR;
struct netInfoResponsePacket{
 NCP ncp;
 unsigned long int ip_pub;
 unsigned short int port_pub;
 unsigned short int no;
}netInfoResponsePacket;
typedef struct netInfoResponsePacket NIRP;
//natCheck请求数据循环队列
NCR natCheckRequestQueue[N_NatCheckRequestQueue];
NCR * h_NatCheckRequestQueue,* t_NatCheckRequestQueue;
/*
 * 获得本地主机IP地址
 */
char * getownIP()
{
 //////////////////  
 //   获得主机名.   
 char   hostname[256];  
 int   res   =   gethostname(hostname,   sizeof(hostname));  
 if   (res   !=   0)   {  
  printf("Error:   %u/n",   WSAGetLastError());  
  return "failed";  
 }    
 ////////////////  
 //   根据主机名获取主机信息.  
 hostent*   pHostent   =   gethostbyname(hostname);  
 if   (pHostent==NULL)   {  
  printf("Error:   %u/n",   WSAGetLastError());  
  return   "failed";  
 }  
 //////////////////  
 //   解析返回的hostent信息.   
 hostent&   he   =   *pHostent;  
 sockaddr_in   sa;  
    memcpy   (   &sa.sin_addr.s_addr,   he.h_addr_list[0],he.h_length);  
 return inet_ntoa(sa.sin_addr);  
}
unsigned long int getownIP_uli()
{
 //////////////////  
 //   获得主机名.   
 char   hostname[256];  
 int   res   =   gethostname(hostname,   sizeof(hostname));  
 if   (res   !=   0)   {  
  printf("Error:   %u/n",   WSAGetLastError());  
  return -1;  
 }    
 ////////////////  
 //   根据主机名获取主机信息.  
 hostent*   pHostent   =   gethostbyname(hostname);  
 if   (pHostent==NULL)   {  
  printf("Error:   %u/n",   WSAGetLastError());  
  return   -1;  
 }  
 //////////////////  
 //   解析返回的hostent信息.   
 hostent&   he   =   *pHostent;  
 sockaddr_in   sa;  
    memcpy   (   &sa.sin_addr.s_addr,   he.h_addr_list[0],he.h_length);  
 return sa.sin_addr.S_un.S_addr;
}
/*
 * superNode接收请求线程入口函数
 */
unsigned int _stdcall ThreadFunction_sn_get(void *param)
{
 //创建套接字
 if( (sServer = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP)) < 0)
 {
  printf("获取SOCKET号失败!!!/n");
  return 0;
 }
 //绑定套接字和本机地址和端口  
 sn_addr.sin_family=AF_INET;
 sn_addr.sin_port=htons(PORT_SERVER_NATCHECK);
 sn_addr.sin_addr.s_addr =INADDR_ANY ;//inet_addr(getownIP());  //
 if(bind(sServer, (struct sockaddr*)&sn_addr, sizeof(sn_addr)) < 0 )
 {
  printf("不能将服务器地址捆绑到SOCKET号上!!!/n");
  closesocket(sServer);
  return 0;
 }
 //printf("接收..../n");
 int iLen=sizeof(client_addr);
 struct natCheckPacket * ncp=(struct natCheckPacket *)buff;
 int timeout=300; 
 while (1)
 {
  setsockopt(sServer, SOL_SOCKET, SO_RCVTIMEO, (char *)&timeout,sizeof(timeout));
  if(recvfrom(sServer,buff,sizeof(buff),0,(sockaddr*)&client_addr,&iLen)!=-1)
  {
   //判断队列是否满
   if(t_NatCheckRequestQueue->next!=h_NatCheckRequestQueue)
   {
    //先修改t
    NCR * t=t_NatCheckRequestQueue;
    t_NatCheckRequestQueue=t->next;
    //再写入
    t->ncp=*(ncp);
    t->ip=client_addr.sin_addr.S_un.S_addr;
    t->port=client_addr.sin_port;
   }
  }
  Sleep(300);
  //printf("version:%d/tdowhat:%d/tport:%d/tip:%d/n",ncp->version,ncp->dowhat,
  // ncp->port,ncp->ip);
  //printf("Client said: %s/n",buff);
  //strcpy(buff,"I am a server!!/n");
  //sendto(sServer,buff,sizeof(buff),0,(sockaddr*)&client_addr,sizeof(client_addr));
 } 
 //closesocket(sServer);
 return 0;
}
/*
 * 构造NCP包发送给SuperNode
 * 接受返回的NCP包
*/
int sendNCPtoSNandRecvNCP(unsigned long int ip_SN,unsigned short int port_SN,
    NCP ncp_send)
{
 int timeout=300; 
 sn_addr.sin_addr.S_un.S_addr=ip_SN;
 sn_addr.sin_family=AF_INET;
 sn_addr.sin_port=htons(port_SN); 
 int iLen=sizeof(sn_addr);
 NCP * ncp=(NCP *)buff;
 ncp->version=ncp_send.version;
 ncp->dowhat=ncp_send.dowhat;
 ncp->port=ncp_send.port;
 ncp->ip=ncp_send.ip; 
 ncp->id=ncp_send.id;
 setsockopt(sClient, SOL_SOCKET , SO_SNDTIMEO, (char *)&timeout,sizeof(timeout));
 sendto(sClient,buff,sizeof(buff),0,(SOCKADDR*)&sn_addr,iLen);
 in_addr_tmp.S_un.S_addr=ip_SN;
 in_addr_tmp1.S_un.S_addr=ncp->ip;
 char * ipaddr=NULL;
 char addr[15];
 in_addr inaddr;
 inaddr. s_addr=ncp->ip;
 ipaddr= inet_ntoa(inaddr);
 strcpy(addr,ipaddr);
 printf("[send][SN(%s)]/n/tversion:%d/tdowhat:%d/tport:%d/tip:%s/n",inet_ntoa(in_addr_tmp),
  ncp->version,ncp->dowhat,ncp->port,addr);
 timeout=1500;
 setsockopt(sClient, SOL_SOCKET, SO_RCVTIMEO, (char *)&timeout,sizeof(timeout));
 if(recvfrom(sClient,buff,sizeof(buff),0,(sockaddr*)&sn_addr,&iLen)<0)
  return 0;
 else
 {
  inaddr. s_addr=sn_addr.sin_addr.S_un.S_addr;
  ipaddr= inet_ntoa(inaddr);
  strcpy(addr,ipaddr); 
  in_addr_tmp.S_un.S_addr=((NCP *)buff)->ip;
  printf("[rece][sn(%s)]/n/tversion:%d/tdowhat:%d/tport:%d/tip:%s/n",addr,((NCP *)buff)->version,
   ((NCP *)buff)->dowhat,ntohs(((NCP *)buff)->port),inet_ntoa(in_addr_tmp));
  return 1;
 }
}
/*
 * 构造NCP包发送给SuperNode
 * 接受返回的NIRP包
*/
int sendNCPtoSNandRecvNIRP(unsigned long int ip_SN,unsigned short int port_SN,
    NCP ncp_send)
{
 FILE * fp_n;
 FILE * fp_c;//for compare
 //打开文件natCheckResult.txt
 if(!(fp_n=fopen("./natCheckResult.txt","a")))
 {
  printf("natCheckResult.txt file not found!/n");
  return 0;
 }
 char one_line[100],one_line_f[100];
 int timeout=300; 
 sn_addr.sin_addr.S_un.S_addr=ip_SN;
 sn_addr.sin_family=AF_INET;
 sn_addr.sin_port=htons(port_SN); 
 int iLen=sizeof(sn_addr);
 char buff_recv[sizeof(NIRP)];
 NCP * ncp=(NCP *)buff;
 NIRP * nirp=(NIRP *)buff_recv;
 ncp->version=ncp_send.version;
 ncp->dowhat=ncp_send.dowhat;
 ncp->port=ncp_send.port;
 ncp->ip=ncp_send.ip; 
 ncp->id=ncp_send.id;
 setsockopt(sClient, SOL_SOCKET , SO_SNDTIMEO, (char *)&timeout,sizeof(timeout));
 sendto(sClient,buff,sizeof(buff),0,(SOCKADDR*)&sn_addr,iLen);
 in_addr_tmp.S_un.S_addr=ip_SN;
 in_addr_tmp1.S_un.S_addr=ncp->ip;
 char * ipaddr=NULL;
 char addr[15];
 in_addr inaddr;
 inaddr. s_addr=ncp->ip;
 ipaddr= inet_ntoa(inaddr);
 strcpy(addr,ipaddr);
 printf("[send][SN(%s)]/n/tversion:%d/tdowhat:%d/tport:%d/tip:%s/n",
  inet_ntoa(in_addr_tmp),ncp->version,ncp->dowhat,ncp->port,addr);
 timeout=1500;
 setsockopt(sClient, SOL_SOCKET, SO_RCVTIMEO, (char *)&timeout,sizeof(timeout));
 int stop=0;
 while(!stop)
 {
  setsockopt(sClient, SOL_SOCKET, SO_RCVTIMEO, (char *)&timeout,sizeof(timeout));
  if(recvfrom(sClient,buff_recv,sizeof(buff_recv),0,(sockaddr*)&sn_addr,&iLen)<0)
   stop=1;
  else
  {
   sprintf(one_line,"%u %u %u %u %u %u",
    ((NIRP *)buff_recv)->ncp.id,
    ((NIRP *)buff_recv)->ncp.dowhat,
    ((NIRP *)buff_recv)->ip_pub,
    ((NIRP *)buff_recv)->port_pub,
    ((NIRP *)buff_recv)->ncp.ip,
    ((NIRP *)buff_recv)->ncp.port);
   printf("%s/n",one_line);
   friendNetType=((NIRP *)buff_recv)->ncp.dowhat;
   friendPubIp=((NIRP *)buff_recv)->ip_pub;
   friendPubPort=((NIRP *)buff_recv)->port_pub;
   friendPriIp=((NIRP *)buff_recv)->ncp.ip;
   friendPriPort=((NIRP *)buff_recv)->ncp.port;
   //打开文件natCheckResult.txt
   if(!(fp_c=fopen("./natCheckResult.txt","r")))
   {
    printf("natCheckResult.txt file not found!/n");
    return 0;
   }
   while(!feof(fp_c))
   {
    i=0;
    while((ch=fgetc(fp_c))!=10 && ch!=-1)
     one_line_f[i++]=ch;
    if(ch==-1)
     break;
    one_line_f[i]='/0';
    if(strcmp(one_line,one_line_f)==0)
    {
     fclose(fp_c);
     fclose(fp_n);
     return 1;
    }
   }
   
   fprintf(fp_n,"%s/n",one_line);
   fflush(fp_n);
  }
 }
 fclose(fp_c);
 fclose(fp_n);
 return 1;
}
void longIP2string(unsigned long int ip,char* addr)
{
 in_addr inaddr;
 char * ipaddr=NULL;
 inaddr. s_addr=sn_addr.sin_addr.S_un.S_addr;
 ipaddr= inet_ntoa(inaddr);
 strcpy(addr,ipaddr);
}
/*
 * 从client端的hostcache.txt文件获取一个SN
 *  写入ip_SN,port_SN
 *  如果文件已经到结尾,或者文件语法错误,则返回0
 */
int getSNfromHostCache(FILE * fp,unsigned long int &ip_SN,unsigned short int &port_SN)
{
 if(feof(fp))
  return 0;
 i=0;
 while((ch=fgetc(fp))!=' ' && ch!=10 && ch!=-1)
  tmpIP[i++]=ch;
 if(ch==10 || ch==-1)
 {
  printf("hostcache.txt file error!/n");
  return 0;
 }
 tmpIP[i]='/0';
 i=0;
 while((ch=fgetc(fp))!=10 && ch!=-1)
  tmpPort[i++]=ch;
 tmpPort[i]='/0';
 ip_SN=inet_addr(tmpIP);
 port_SN=(unsigned short int)atoi(tmpPort);
 return 1;
}
/*
 * client最先知道检测结果
 *  client需要向自己缓存的SN节点汇报自己的网络类型信息
 *  这里依然使用了NCP包
void broadcastNatCheckResult(char version,
        char dowhat,unsigned long int ip,unsigned short int port,
        unsigned long int id)
*/
void broadcastNatCheckResult(NCP ncp_send)
{
 FILE * fp_b=fp;
 fseek(fp_b,0L,0);
 while (!feof(fp_b)) {
  i=0;
  while((ch=fgetc(fp_b))!=' ' && ch!=10 && ch!=-1)
   tmpIP[i++]=ch;
  if(ch==10 || ch==-1)
  {
   printf("hostcache.txt file error!/n");
   return;
  }
  tmpIP[i]='/0';
  i=0;
  while((ch=fgetc(fp_b))!=10 && ch!=-1)
   tmpPort[i++]=ch;
  tmpPort[i]='/0';
  int timeout=300;
  sn_addr.sin_addr.S_un.S_addr=inet_addr(tmpIP);
  sn_addr.sin_family=AF_INET;
  sn_addr.sin_port=htons((unsigned short int)atoi(tmpPort));
  int iLen=sizeof(sn_addr);
  NCP * ncp=(NCP *)buff;
  ncp->version=ncp_send.version;
  ncp->dowhat=ncp_send.dowhat;
  ncp->port=ncp_send.port;
  ncp->ip=ncp_send.ip; 
  ncp->id=ncp_send.id;
  setsockopt(sClient, SOL_SOCKET , SO_SNDTIMEO, (char *)&timeout,sizeof(timeout));
  sendto(sClient,buff,sizeof(buff),0,(SOCKADDR*)&sn_addr,iLen);
 }
 fclose(fp_b);
 return;
}
/*
 * SN收到client汇报的结果,
 * 向NatCheckResult.txt 写入id 网络类型 外网ip port 内网ip port 
 */
int storeNatCheckResult(unsigned long int id,char type,
      unsigned long int pub_ip,unsigned short int pub_port,
      unsigned long int pri_ip,unsigned short int pri_port)
{
 FILE * fp_n;
 char one_line[100],one_line_f[100];
 sprintf(one_line,"%u %u %u %u %u %u",id,type,pub_ip,pub_port,pri_ip,pri_port);
 //打开文件natCheckResult.txt
 if(!(fp_n=fopen("./natCheckResult.txt","rw")))
 {
  printf("natCheckResult.txt file not found!/n");
  return 0;
 }
 while(!feof(fp_n))
 {
  i=0;
  while((ch=fgetc(fp_n))!=10 && ch!=-1)
   one_line_f[i++]=ch;
  if(ch==-1)
  {
   break;
  }
  one_line_f[i]='/0';
  if(strcmp(one_line,one_line_f)==0)
   return 0;
 }
 fclose(fp_n);
 if(!(fp_n=fopen("./natCheckResult.txt","a")))
 {
  printf("natCheckResult.txt file not found!/n");
  return 0;
 } 
 fprintf(fp_n,"%u %u %u %u %u %u/n",id,type,pub_ip,pub_port,pri_ip,pri_port);
 printf("%u %u %u %u %u %u/n",id,type,pub_ip,pub_port,pri_ip,pri_port);
 fflush(fp_n);
 fclose(fp_n);
 return 1;
}
/*
 * SN收到client的查询请求,
 * 从NatCheckResult.txt 中查找id对应的 网络类型 外网ip port 内网ip port 
 */
int searchNatCheckResult(NCP ncp,SOCKET sServer,sockaddr_in addr)
{
 FILE * fp_n;
 char id[32];
 //打开文件natCheckResult.txt
 if(!(fp_n=fopen("./natCheckResult.txt","r")))
 {
  printf("natCheckResult.txt file not found!/n");
  return 0;
 }
 int j=0;
 while(!feof(fp_n))
 {
  i=0;
  while((ch=fgetc(fp_n))!=' ' && ch!=10 && ch!=-1)
   id[i++]=ch;
  if(ch==10)
  {
   printf("natCheckResult.txt file error!/n");
   return 0;
  }
  if(ch==-1)
  {
   break;
  }
  id[i]='/0';
  if(ncp.id!=(unsigned long int)atoi(id))
  {
   while(fgetc(fp_n)!=10);
   continue;
  }
  unsigned long int ip_pub;
  unsigned short int port_pub;
  fscanf(fp_n,"%d %d %d %d %d/n",&(ncp.dowhat),&ip_pub,&port_pub,&(ncp.ip),&(ncp.port));
  char buff_send[sizeof(NIRP)];
  NIRP * nirp = (NIRP *) buff_send;
  nirp->ncp=ncp;
  nirp->ip_pub=ip_pub;
  nirp->port_pub=port_pub;
  nirp->no=j++;  
  sendto(sServer,buff_send,sizeof(buff_send),0,(sockaddr*)&addr,sizeof(addr));
 }
 fclose(fp_n);
 return 1;
}
/*
 * main函数
 */
int main(int argc, char* argv[])
{  
 if(argc < 2)
 {
  printf("usage: %s -c or -s /n", argv[0]);
  printf("-c: for clients/n-s: for superNode/n");
  return 0;
 }
 //////////////////
 //   初始化winsock环境.
 WSADATA wsadata;
 WORD wVersion=MAKEWORD(2,0);  //winsock 2.0
 if(WSAStartup(wVersion,&wsadata)!=0)
 {
  printf("winsocket initalize failed!/n");
  WSACleanup();
  return 0;
 } 
 ///////////////////////////////////////////////////////////////////
 //客户端程序入口
 ////////////////////////////////////////////////////////////////////
 if(strcmp(argv[1],"-c")==0)
 { 
  if(argc>=3)
   myID=atoi(argv[2]);
  NCP ncp_send;
  ncp_send.version=0;
  ncp_send.id=myID;
  //////////////////
  //  读取hostcache文件中的前两个supernode
  if(!(fp=fopen("./hostcache.txt","r")))
  {
   printf("hostcache.txt file not found!/n");
   return 0;
  }
  if( (sClient=socket(AF_INET,SOCK_DGRAM,IPPROTO_UDP)) < 0)
  {
   printf("获取SOCKET号失败!!!/n");
   return 0;
  } 
  myPriIp=inet_addr(getownIP());
  myPriPort=htons(PORT_CLIENT_NATCHECK);
  client_addr.sin_addr.S_un.S_addr=INADDR_ANY; /* use my IP address */
  client_addr.sin_family=AF_INET;
  client_addr.sin_port=htons(PORT_CLIENT_NATCHECK);
  printf("[info]client ip=%s/tport:%d/n",getownIP(),ntohs(myPriPort));
  //bind(int sockfd, const struct sockaddr *my_addr, int addrlen);
  if(bind(sClient,(struct sockaddr*)&client_addr,sizeof(client_addr))<0)
  {
   printf("不能捆绑到SOCKET号上!!!/n");
   closesocket(sClient);
   return 0;
  }
  /*/////////////////////////////////////////////////////////////////////////////////////
  // NAT检测开始
  // 检测流程:
   1.向超级节点SN1发送0号包,等待一段超时时间,如果没有返回,换下一个超级节点。
     否则go 2,如果hostcache中的节点都失效了,那么开始向中心节点发请求,来得到一批
     新的superNode节点列表;
   2.判断超级节点返回的7号包中携带的(ip,port)和自己发包时用的(ip,port)是否相同,
     如果相同的话,说明本地是公网IP;如果不同的话,说明在NAT后;
   3.向SN1发送1号包,并携带一个自己列表中的另一个superNode,SN2的(ip,port),让SN1
     发8号包给SN2,告诉本client的外网(ip,port),让SN2给本client发6号包;另外SN1还可
     以在自己维护的临近的SN列表中选择3个也给client发送6号包。
   4.如果client在一段超时时间内,没有收到任何包,那么在将3中的过程进行一次,总共最多
     可以进行N_CheckTime(5)次。其中只要有一次收到包,就go 5。
   5.如果收到了SN2、SN3、SN4中的任何一个,检测就停止,结果是full cone Nat。如果没有
     收到任何包,检测继续,go 6
   6.client从自己的hostcache中选择一个SN2,向其发送2号包,等待一段超时时间,如果没有
     收到5号包的话,继续从hostcache中另选择一个SN3,如果hostcache中的所有SN都不给回应
     5号包的话,从中心节点获取新的SN后,继续发2号包。收到第一个5号包,开始比较,如果和
     SN1的返回结果(ip,port)中如果有任何一对不一致,则说明是Symmetric NAT,检测停止。
     否则是Restricted Cone NAT,同时继续检测,go 7
   7.client向SN1发送3号包,让SN1换用另一个端口发4号包,重复若干次,如果client能收到说明
     是Restricted Cone NAT,否则是Port Restricted Cone NAT。检测结束。
  *///////////////////////////////////////////////////////////////////////////////////////
  //读取第一个SN,SN1的(IP,Port),并给其发0号包
  if(getSNfromHostCache(fp,snIP[0],snPort[0])==0)
  {
   printf("[error]there is no SN in HostCache.txt!/n");
   return 0;
  }
  /*
  int sendNCPtoSNandRecvNCP(unsigned long int ip_SN,
   unsigned short int port_SN,
   char version,
   char dowhat,
   unsigned long int ip,
   unsigned short int port)
  */
  ncp_send.dowhat=0;
  ncp_send.ip=0;
  ncp_send.port=0; 
  while(sendNCPtoSNandRecvNCP(snIP[0],snPort[0],ncp_send)==0)
  {
   if(getSNfromHostCache(fp,snIP[0],snPort[0])==0)
   {
    printf("[error]there is no more SN in HostCache.txt!/n");
    //此时应该从中心节点获取更新的SN列表
    //......
    printf("[error]check failed temporarily");
    return 0;
   }
  }
  long ft=ftell(fp);
  in_addr_tmp.S_un.S_addr=snIP[0];
  printf("[info]SN1 ip=%s/tport=%d/n",inet_ntoa(in_addr_tmp),snPort[0]);
  //对比返回的(ip,port) ip_return port_return
  unsigned long int ip_return=((NCP *)buff)->ip;
  unsigned short int port_return=((NCP *)buff)->port;
  if(myPriIp==ip_return && myPriPort==port_return)
  {
   printf("[Reslut]public net/n");
   myNetType=0;
   ncp_send.dowhat=9;
   ncp_send.ip=myPubIp=myPriIp;
   ncp_send.port=myPubPort=myPriPort;
   broadcastNatCheckResult(ncp_send);
   return 0;
  }
  //再读取一个SN,SN2的(IP,Port)
  int checkTime=N_CheckTime;
  while(checkTime>0 && !feof(fp))
  {
   if(getSNfromHostCache(fp,snIP[1],snPort[1])==0)
   {
    printf("[error]there is no more SN in HostCache.txt!/n");
    break;
   } 
   ncp_send.dowhat=1;
   ncp_send.ip=snIP[1];
   ncp_send.port=snPort[1];
   if(sendNCPtoSNandRecvNCP(snIP[0],snPort[0],ncp_send)==1)
   { 
    printf("[Reslut]full cone nat/n");
    myNetType=1;
    ncp_send.dowhat=10;
    ncp_send.ip=myPriIp;
    ncp_send.port=myPriPort;
    myPubIp=ip_return;
    myPubPort=port_return;
    broadcastNatCheckResult(ncp_send);
    return 0;
   }  
   checkTime--;
  }
  printf("[Reslut]not full cone nat, checking is continued./n");
  in_addr_tmp.S_un.S_addr=snIP[1];
  printf("[info]SN2 ip=%s/tport=%d/n",inet_ntoa(in_addr_tmp),snPort[1]);  
  //在hostcache中的SN1下再找一个SN3,给他发送2号包,让他返回client的ip和port
  fseek(fp,ft,0);
  if(getSNfromHostCache(fp,snIP[1],snPort[1])==0)
  {
   printf("[error]there is no SN in HostCache.txt!/n");
   return 0;
  } 
  ncp_send.dowhat=2;
  ncp_send.ip=0;
  ncp_send.port=0; 
  while(sendNCPtoSNandRecvNCP(snIP[1],snPort[1],ncp_send)==0)
  {
   if(getSNfromHostCache(fp,snIP[1],snPort[1])==0)
   {
    printf("[error]there is no more SN in HostCache.txt!/n");
    //此时应该从中心节点获取更新的SN列表
    //......
    printf("[error]check failed temporarily");
    return 0;
   }
  }
  in_addr_tmp.S_un.S_addr=snIP[1];
  printf("[info]SN3 ip=%s/tport=%d/n",inet_ntoa(in_addr_tmp),snPort[1]);
  //对比返回的(ip,port) ip_return port_return
  unsigned long int ip_return1=((NCP *)buff)->ip;
  unsigned short int port_return1=((NCP *)buff)->port;
  if(ip_return1!=ip_return || port_return1!=port_return)
  {
   printf("[Reslut]symmetric nat/n");
   myNetType=2;
   ncp_send.dowhat=11;
   ncp_send.ip=myPriIp;
   ncp_send.port=myPriPort;
   broadcastNatCheckResult(ncp_send);
   return 0;
  } 
  ncp_send.dowhat=3;
  ncp_send.ip=0;
  ncp_send.port=0; 
  if(sendNCPtoSNandRecvNCP(snIP[0],snPort[0],ncp_send)==1)
  {
   printf("[Reslut]Restricted Cone nat/n");
   myNetType=3;
   ncp_send.dowhat=12;
   ncp_send.ip=myPriIp;
   ncp_send.port=myPriPort;
   myPubIp=ip_return;
   myPubPort=port_return;
   broadcastNatCheckResult(ncp_send);
   return 0;
  }
  printf("[Reslut]Port Restricted Cone nat/n");
  myNetType=4;
  ncp_send.dowhat=13;
  ncp_send.ip=myPriIp;
  ncp_send.port=myPriPort;
  myPubIp=ip_return;
  myPubPort=port_return;
  broadcastNatCheckResult(ncp_send);
  //以上检测网络类型结束,下面开始与某好友通讯
  printf("do you want to talk with someone? [Y]yes [N]no/n");
  scanf("%c",&ch);
  if(ch!='Y' && ch!='y')
   return 0;
  printf("which friend would you want to talk with? Input his id:/n");
  scanf("%d",&friendID);
  //从SN1获取friend网络类型列表
  ncp_send.dowhat=14;
  ncp_send.ip=0;
  ncp_send.port=0;
  ncp_send.id=friendID;
  if(sendNCPtoSNandRecvNIRP(snIP[0],snPort[0],ncp_send)==0)
  { 
   printf("[error]查询好友网络信息失败!/n");
   return 0;
  }
  switch(friendNetType)
  {
  case 9:
   break;
  case 10:
   break;
  case 11:
   break;
  case 12:
   break;
  case 13:
   break;
  default:
   break;
  }
 }
 ////////////////////////////////////////////////////////////////////
 //服务端程序入口
 /////////////////////////////////////////////////////////////////////
 else if(strcmp(argv[1],"-s")==0)
 {
  //创建另一个套接字
  SOCKET sServer1;
  if( (sServer1 = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP)) < 0)
  {
   printf("获取SOCKET号失败!!!/n");
   return 0;
  }
  //绑定套接字和本机地址和端口  
  sn1_addr.sin_family=AF_INET;
  sn1_addr.sin_port=htons(PORT_SERVER_NATCHECK_1);
  sn1_addr.sin_addr.s_addr =INADDR_ANY ;//inet_addr(getownIP());  //
  if(bind(sServer1, (struct sockaddr*)&sn1_addr, sizeof(sn1_addr)) < 0 )
  {
   printf("不能将服务器地址捆绑到SOCKET号上!!!/n");
   closesocket(sServer1);
   return 0;
  }
  //////////////////
  //  循环队列natCheckRequestQueue初始化
  for(i=0;i<N_NatCheckRequestQueue-1;i++)
  {
   natCheckRequestQueue[i].next= & natCheckRequestQueue[i+1];
  }
  natCheckRequestQueue[N_NatCheckRequestQueue-1].next
   = & natCheckRequestQueue[0];
  h_NatCheckRequestQueue = & natCheckRequestQueue[0];
  t_NatCheckRequestQueue = & natCheckRequestQueue[0];
  //////////////////
  //  启动接受请求的服务线程
  //  入口函数为:ThreadFunction_sn_get
  //  判断队列是否满,不满的条件下,先修改t,再写进t指向的单元
  //HANDLE thread_sn_get = (HANDLE)_beginthread(ThreadFunction_sn_get,0,NULL);
  HANDLE thread_sn_get = (HANDLE)_beginthreadex(0,0,&ThreadFunction_sn_get,0,0,NULL);
  //////////////////
  //  主服务线程
  //  用来从循环队列natCheckRequestQueue的头指针得到请求数据来处理
  //  判断队列是否空,非空的条件下,先读出h指向的单元,再修改h
  while(1)
  {
   //判断队列是否空
   if(t_NatCheckRequestQueue!=h_NatCheckRequestQueue)
   {
    //先读出
    NCP * ncp_deal=(NCP *)buff;
    *(ncp_deal)=h_NatCheckRequestQueue->ncp;
    unsigned long int ip_deal=h_NatCheckRequestQueue->ip;
    unsigned short int port_deal=h_NatCheckRequestQueue->port;
    //再修改h
    h_NatCheckRequestQueue=h_NatCheckRequestQueue->next;
    //处理这个来自(ip,port)的请求 
    switch(ncp_deal->dowhat)
    {
    /* dowhat字段: 
    *   0 请求用户返回Ip:port   7 单独的响应包
    *   1 客户请求提供其他主机检测  6 其他主机响应包
    *   2 同时请求两主机发回响应  5 针对2的回应包
    *   3 请求使用另一端口发回响应  4 使用另一端口响应
    *   8 SN请求其他主机检测
    -----------------------------------------------------------
    *  9  汇报检测结果给SN public net
    *  10 汇报检测结果给SN full cone nat
    *  11 汇报检测结果给SN symmetric nat
    *  12 汇报检测结果给SN restricted cone nat
    *  13 汇报检测结果给SN port restricted cone nat
    -----------------------------------------------------------
    */
    case 0:
     ncp_deal->dowhat=7;
     ncp_deal->ip=ip_deal;
     ncp_deal->port=port_deal;
     client_addr_deal.sin_family=AF_INET;
     client_addr_deal.sin_addr.S_un.S_addr=ip_deal;
     client_addr_deal.sin_port=port_deal;
     printf("[deal][0]send [7] to:%d.%d.%d.%d:%d/n",
      client_addr_deal.sin_addr.S_un.S_un_b.s_b1,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b2,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b3,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b4,
      htons(client_addr_deal.sin_port));
     sendto(sServer,buff,sizeof(buff),0,(sockaddr*)&client_addr_deal,sizeof(client_addr_deal));
     break;
    case 1:
     client_addr_deal.sin_family=AF_INET;
     client_addr_deal.sin_addr.S_un.S_addr=ncp_deal->ip;
     client_addr_deal.sin_port=htons(ncp_deal->port);
     ncp_deal->dowhat=8;
     ncp_deal->ip=ip_deal;
     ncp_deal->port=port_deal;
     printf("[deal][1]send [8] to:%d.%d.%d.%d:%d/n",
      client_addr_deal.sin_addr.S_un.S_un_b.s_b1,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b2,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b3,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b4,
      htons(client_addr_deal.sin_port));     
     sendto(sServer,buff,sizeof(buff),0,(sockaddr*)&client_addr_deal,sizeof(client_addr_deal));
     break;
    case 2:
     client_addr_deal.sin_family=AF_INET;
     client_addr_deal.sin_addr.S_un.S_addr=ip_deal;
     client_addr_deal.sin_port=port_deal;
     ncp_deal->dowhat=5;
     ncp_deal->ip=ip_deal;
     ncp_deal->port=port_deal;
     printf("[deal][2]send [5] to:%d.%d.%d.%d:%d/n",
      client_addr_deal.sin_addr.S_un.S_un_b.s_b1,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b2,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b3,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b4,
      htons(client_addr_deal.sin_port));     
     sendto(sServer,buff,sizeof(buff),0,(sockaddr*)&client_addr_deal,sizeof(client_addr_deal));
     break;
    case 3:
     client_addr_deal.sin_family=AF_INET;
     client_addr_deal.sin_addr.S_un.S_addr=ip_deal;
     client_addr_deal.sin_port=port_deal;
     ncp_deal->dowhat=4;
     ncp_deal->ip=ip_deal;
     ncp_deal->port=port_deal;
     printf("[deal][3]send [4] to:%d.%d.%d.%d:%d/n",
      client_addr_deal.sin_addr.S_un.S_un_b.s_b1,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b2,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b3,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b4,
      htons(client_addr_deal.sin_port));     
     sendto(sServer1,buff,sizeof(buff),0,(sockaddr*)&client_addr_deal,sizeof(client_addr_deal));
     break;
    case 8:
     ncp_deal->dowhat=6;
     client_addr_deal.sin_family=AF_INET;
     client_addr_deal.sin_addr.S_un.S_addr=ncp_deal->ip;
     client_addr_deal.sin_port=htons(ncp_deal->port);
     ncp_deal->ip=getownIP_uli();
     ncp_deal->port=htons(PORT_SERVER_NATCHECK);
     printf("[deal][8] send[6] to:%d.%d.%d.%d:%d/n",
      client_addr_deal.sin_addr.S_un.S_un_b.s_b1,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b2,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b3,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b4,
      htons(client_addr_deal.sin_port));
     sendto(sServer,buff,sizeof(buff),0,(sockaddr*)&client_addr_deal,sizeof(client_addr_deal));
     break;
    case 9:
     //写入id 网络类型 外网ip port 内网ip port 
     storeNatCheckResult(ncp_deal->id,
      ncp_deal->dowhat,ip_deal,port_deal,
      ncp_deal->ip,ncp_deal->port);
     printf("[deal][9] writing file/n");
     break;
    case 10:
     //写入id 网络类型 外网ip port 内网ip port
     storeNatCheckResult(ncp_deal->id,
      ncp_deal->dowhat,ip_deal,port_deal,
      ncp_deal->ip,ncp_deal->port);
     printf("[deal][10] writing file/n");
     break;
    case 11:
     //写入id 网络类型 外网ip port 内网ip port
     storeNatCheckResult(ncp_deal->id,
      ncp_deal->dowhat,ip_deal,port_deal,
      ncp_deal->ip,ncp_deal->port);
     printf("[deal][11] writing file/n");
     break;
    case 12:
     //写入id 网络类型 外网ip port 内网ip port
     storeNatCheckResult(ncp_deal->id,
      ncp_deal->dowhat,ip_deal,port_deal,
      ncp_deal->ip,ncp_deal->port);
     printf("[deal][12] writing file/n");
     break;
    case 13:
     //写入id 网络类型 外网ip port 内网ip port 
     storeNatCheckResult(ncp_deal->id,
      ncp_deal->dowhat,ip_deal,port_deal,
      ncp_deal->ip,ncp_deal->port);
     printf("[deal][13] writing file/n"); 
     break;
    case 14:
     //处理client发来的查询请求
     //查询id对应用户的网络信息 
     client_addr_deal.sin_family=AF_INET;
     client_addr_deal.sin_addr.S_un.S_addr=ip_deal;
     client_addr_deal.sin_port=port_deal;
     printf("[deal][14]send [15] to:%d.%d.%d.%d:%d/n",
      client_addr_deal.sin_addr.S_un.S_un_b.s_b1,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b2,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b3,
      client_addr_deal.sin_addr.S_un.S_un_b.s_b4,
      htons(client_addr_deal.sin_port));
     searchNatCheckResult(*ncp_deal,sServer,client_addr_deal);
     break;
    default:
     break;
    }
   }
   Sleep(300);
  }  
 } 
 //////////////////  
 //   终止   Windows   sockets   API  
 WSACleanup(); 
 return 0;
}


标签:addr,deal,ncp,ip,C++,穿透,client,NAT,port
来源: https://blog.51cto.com/u_9634496/2732207