C套接字:将请求转发到端口80并读取响应

时间:2011-08-15 11:42:57

标签: c sockets proxy

我有以下代码(我正在使用代码http://www.linuxhowtos.org/C_C++/socket.htm),我正在尝试将其转换为代理服务器:

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netdb.h>

void dostuff(int); /* function prototype */
void error(const char *msg)
{
    perror(msg);
    exit(1);
}

int main(int argc, char *argv[])
{
     //setup proxy:
     int sockfd, newsockfd, portno, pid;
     socklen_t clilen;
     struct sockaddr_in serv_addr, cli_addr;

     if (argc < 2) {
         fprintf(stderr,"***ERROR, no port provided\n");
         exit(1);
     }
     sockfd = socket(AF_INET, SOCK_STREAM, 0);
     if (sockfd < 0)
        error("***ERROR opening socket");
     bzero((char *) &serv_addr, sizeof(serv_addr));
     portno = atoi(argv[1]);
     serv_addr.sin_family = AF_INET;
     serv_addr.sin_addr.s_addr = INADDR_ANY;
     serv_addr.sin_port = htons(portno);
     if (bind(sockfd, (struct sockaddr *) &serv_addr,
              sizeof(serv_addr)) < 0)
              error("***ERROR on binding");
     listen(sockfd,5);
     clilen = sizeof(cli_addr);
     while (1) {
         newsockfd = accept(sockfd,
               (struct sockaddr *) &cli_addr, &clilen);
         if (newsockfd < 0)
             error("***ERROR on accept");
         pid = fork();
         if (pid < 0)
             error("***ERROR on fork");
             if (pid == 0)  {
             close(sockfd);
             dostuff(newsockfd);
             exit(0);
         }
         else close(newsockfd);
     } /* end of while */
     close(sockfd);
     return 0; /* we never get here */
}

/******** DOSTUFF() *********************
 There is a separate instance of this function 
 for each connection.  It handles all communication
 once a connnection has been established.
 *****************************************/
void dostuff (int sock)
{
   int n;
   char buffer[256];

   bzero(buffer,256);
   n = read(sock,buffer,255);
   if (n < 0){
       error("***ERROR reading from socket");
   }
   //printf("Here is the message: %s\n",buffer);




   /*
   ***Forward message to port 80 and read response here
   */





   n = write(sock,"I got your message",18);
   if (n < 0) error("***ERROR writing to socket");
}

在函数“dostuff”中,我想将'buffer'写入端口80,读取响应并将此响应写回端口20000(argv [1])。

目前,当我将浏览器的代理设置为172.16.1.218:20000时,我得到的只是“我收到了您的消息”。我想将此更改为网页的响应!

任何正确方向的指针都非常感谢。

这是我尝试过的方法(使用此代码替换多行注释“将消息转发到端口80并在此处阅读响应”):

   int sockfdi, portnoi, ni;
   struct sockaddr_in serv_addri;
   struct hostent *serveri;
   portnoi =80;

   sockfdi = socket(AF_INET, SOCK_STREAM, 0);
   if (sockfdi < 0){
        error("***ERROR opening socket");
   }
   serveri = gethostbyname("172.16.1.218");
   if (serveri == NULL){
       fprintf(stderr,"***ERROR, no such host\n");
       exit(0);
   }

   bzero((char *) &serv_addri, sizeof(serv_addri));
   serv_addri.sin_family = AF_INET;
   bcopy((char *)serveri->h_addr, (char *)&serv_addri.sin_addr.s_addr, serveri->h_length);
   serv_addri.sin_port = htons(portnoi);
   if (connect(sockfdi,(struct sockaddr *) &serv_addri,sizeof(serv_addri)) < 0){
       error("***ERROR connecting");
   }
   printf("Please enter the message: ");
   bzero(buffer,256);

但是每当我尝试通过我的webbrowser连接时,服务器回声:“*** ERROR连接:连接被拒绝”

非常感谢,

1 个答案:

答案 0 :(得分:1)

这是你要做的非常重要的任务。目前,你错过了三件事,一件简单就是两件事:

  1. 您必须打开要转接呼叫的服务器的网络连接(相当简单,请参阅socket()connect())。

  2. 然后,您将拥有一个双工连接,即两个并发的数据流,一个从客户端到转发的服务器,另一个从转发的服务器到客户端。为了应对这种并发性,您需要两个具有阻塞I / O的线程或某种非阻塞I / O(请参阅select()或AIO)。

  3. 如果转发HTTP请求而不更改其他服务器,则可能最终会在请求中使用无效的服务器名称和IP地址。然后该请求将被拒绝。因此,您需要解析HTTP标头,进行一些替换并转发修改后的HTTP请求。