<form id="hz9zz"></form>
  • <form id="hz9zz"></form>

      <nobr id="hz9zz"></nobr>

      <form id="hz9zz"></form>

    1. 明輝手游網中心:是一個免費提供流行視頻軟件教程、在線學習分享的學習平臺!

      穿透代理服務器編程

      [摘要]田進恩 allfresh@263.net 關鍵詞:代理服務器、Socks4、Socks5、Http代理 在網絡程序設計過程中,我們經常要與各種類型的代理服務器打交道,比如在企業內部網通過代理去訪問Internet網上的服務器等等,一般代理服務器支持幾種常見的代理協議標準,如Sock...
      田進恩 allfresh@263.net


          關鍵詞:代理服務器、Socks4、Socks5、Http代理  

          在網絡程序設計過程中,我們經常要與各種類型的代理服務器打交道,比如在企業內部網通過代理去訪問Internet網上的服務器等等,一般代理服務器支持幾種常見的代理協議標準,如Socks4,Socks5,Http代理,其中Socks5需要用戶驗證,代理相對復雜。我在查閱RFC文檔和相關資料后,特總結一些TCP協議穿透代理服務器的程序片斷,希望對大家有所幫助。

      //使用到的結構
      struct sock4req1
      {
      char VN;
      char CD;
      unsigned short Port;
      unsigned long IPAddr;
      char other[1];
      };

      struct sock4ans1
      {
      char VN;
      char CD;
      };

      struct sock5req1
      {
      char Ver;
      char nMethods;
      char Methods[255];
      };

      struct sock5ans1
      {
      char Ver;
      char Method;
      };

      struct sock5req2
      {
      char Ver;
      char Cmd;
      char Rsv;
      char Atyp;
      char other[1];
      };

      struct sock5ans2
      {
      char Ver;
      char Rep;
      char Rsv;
      char Atyp;
      char other[1];
      };

      struct authreq
      {
      char Ver;
      char Ulen;
      char Name[255];
      char PLen;
      char Pass[255];
      };

      struct authans
      {
      char Ver;
      char Status;
      };

      //通過Socks4方式代理
      if( !ClientSock.Connect( g_ProxyInfo.m_strProxyIP,g_ProxyInfo.m_nProxyPort) )
      {
      m_sError = _T("不能連接到代理服務器!");
      ClientSock.Close();
      return FALSE;
      }
      char buff[100];
      memset(buff,0,100);
      struct sock4req1 *m_proxyreq;
      m_proxyreq = (struct sock4req1 *)buff;
      m_proxyreq->VN = 4;
      m_proxyreq->CD = 1;
      m_proxyreq->Port = ntohs(GetPort());
      m_proxyreq->IPAddr = inet_addr(GetServerHostName());
      ClientSock.Send(buff,9);
      struct sock4ans1 *m_proxyans;
      m_proxyans = (struct sock4ans1 *)buff;
      memset(buff,0,100);
      ClientSock.Receive(buff,100);
      if(m_proxyans->VN != 0 m_proxyans->CD != 90)
      {
      m_sError = _T("通過代理連接主站不成功!");
      ClientSock.Close();
      return FALSE;
      }


        

      //通過Socks5方式代理
      if( !ClientSock.Connect( g_ProxyInfo.m_strProxyIP,g_ProxyInfo.m_nProxyPort) )
      {
      m_sError = _T("不能連接到代理服務器!");
      ClientSock.Close();
      return FALSE;
      }
      char buff[600];
      struct sock5req1 *m_proxyreq1;
      m_proxyreq1 = (struct sock5req1 *)buff;
      m_proxyreq1->Ver = 5;
      m_proxyreq1->nMethods = 2;
      m_proxyreq1->Methods[0] = 0;
      m_proxyreq1->Methods[1] = 2;
      ClientSock.Send(buff,4);
      struct sock5ans1 *m_proxyans1;
      m_proxyans1 = (struct sock5ans1 *)buff;
      memset(buff,0,600);
      ClientSock.Receive(buff,600);
      if(m_proxyans1->Ver != 5 (m_proxyans1->Method!=0 && m_proxyans1->Method!=2))
      {
      m_sError = _T("通過代理連接主站不成功!");
      ClientSock.Close();
      return FALSE;
      }
      if(m_proxyans1->Method == 2)
      {
      int nUserLen = strlen(g_ProxyInfo.m_strProxyUser);
      int nPassLen = strlen(g_ProxyInfo.m_strProxyPass);
      struct authreq *m_authreq;
      m_authreq = (struct authreq *)buff;
      m_authreq->Ver = 1;
      m_authreq->Ulen = nUserLen;
      strcpy(m_authreq->Name,g_ProxyInfo.m_strProxyUser);
      m_authreq->PLen = nPassLen;
      strcpy(m_authreq->Pass,g_ProxyInfo.m_strProxyPass);
      ClientSock.Send(buff,513);
      struct authans *m_authans;
      m_authans = (struct authans *)buff;
      memset(buff,0,600);
      ClientSock.Receive(buff,600);
      if(m_authans->Ver != 1 m_authans->Status != 0)
      {
      m_sError = _T("代理服務器用戶驗證不成功!");
      ClientSock.Close();
      return FALSE;
      }
      }
      struct sock5req2 *m_proxyreq2;
      m_proxyreq2 = (struct sock5req2 *)buff;
      m_proxyreq2->Ver = 5;
      m_proxyreq2->Cmd = 1;
      m_proxyreq2->Rsv = 0;
      m_proxyreq2->Atyp = 1;
      unsigned long tmpLong = inet_addr(GetServerHostName());
      unsigned short port = ntohs(GetPort());
      memcpy(m_proxyreq2->other,&tmpLong,4);
      memcpy(m_proxyreq2->other+4,&port,2);
      ClientSock.Send(buff,sizeof(struct sock5req2)+5);
      struct sock5ans2 *m_proxyans2;
      memset(buff,0,600);
      m_proxyans2 = (struct sock5ans2 *)buff;
      ClientSock.Receive(buff,600);
      if(m_proxyans2->Ver != 5 m_proxyans2->Rep != 0)
      {
      m_sError = _T("通過代理連接主站不成功!");
      ClientSock.Close();
      return FALSE;
      }


        

      //通過HTTP方式代理
      if( !ClientSock.Connect( g_ProxyInfo.m_strProxyIP,g_ProxyInfo.m_nProxyPort) )
      {
      m_sError = _T("不能連接到代理服務器!");
      ClientSock.Close();
      return FALSE;
      }
      char buff[600];
      sprintf( buff, "%s%s:%d%s","CONNECT ",GetServerHostName(),GetPort()," HTTP/1.1\r\nUser-Agent: MyApp/0.1\r\n\r\n");
      ClientSock.Send(buff,strlen(buff)); //發送請求
      memset(buff,0,600);
      ClientSock.Receive(buff,600);
      if(strstr(buff, "HTTP/1.0 200 Connection established") == NULL) //連接不成功
      {
      m_sError = _T("通過代理連接主站不成功!");
      ClientSock.Close();
      return FALSE;
      }


          我們一般先與代理服務器連通,然后向代理服務器發送代理驗證的用戶名和密碼(如果需要,如Socks5代理),驗證成功后,再向代理服務器發送需要連接的目的地址和端口。以上代碼僅用于TCP連接,如果在內部網偵聽或通過UDP協議發送信息,可查閱RFC1829等文檔資料。


      日韩精品一区二区三区高清