C中简单的非并发Web服务器

时间:2013-08-29 11:08:28

标签: c sockets webserver

我一直在尝试运行我的代码一周,我不知道问题是什么。我们在编程类中获得了一个代码,允许我们创建自己的tcp客户端和/或服务器。使用该代码我们必须打开localhost:54321 [我们使用Apache作为本地服务器,如果这有助于/更好地理解正在发生的事情],并通过调整服务器的C代码,页面将显示一条消息。 / p>

这是我的代码:

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



int main(int argc, char **argv)
{
  int listenfd,connfd;
  struct sockaddr_in servaddr,cliaddr;
  socklen_t len = sizeof(cliaddr);
  char cli_ip[32];
  char mesg[1024];
  char response[1028];
  //char reply[] = "Hello World!";

  listenfd = socket(PF_INET,SOCK_STREAM,0);
  bzero(&servaddr, sizeof(servaddr));
  servaddr.sin_family = AF_INET;
  servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
  servaddr.sin_port = htons(54321);

  if ( bind( listenfd, (struct sockaddr*) &servaddr, sizeof(servaddr) ) < 0 ){
      perror(NULL);
      exit(-1);
  }

  while(1){
    //not present in udp server
    connfd = accept(listenfd,(struct sockaddr *)&cliaddr,&len);
    inet_ntop(AF_INET,(struct in_addr *) &cliaddr.sin_addr, cli_ip, sizeof(cli_ip) );
    printf("Client %s connected. \n",cli_ip);

    while(1){
      memset(mesg,0,sizeof(mesg));
      if( recvfrom(connfd,mesg,sizeof(mesg),0,(/*const*/ struct sockaddr *)&cliaddr,&len) > 0 ){
            printf("From %s port %d: %s",cli_ip,ntohs(cliaddr.sin_port),mesg);
        **sprintf(response,"HTTP/1.1 200 OK \n Content-type:text/html \n Content-length: 200 \r\n <html><body> Hello, World! </body></html>");
        sendto(connfd,response,strlen(response),0,(const struct sockaddr *)&servaddr,len);
        //break;**
      }
      else {
        printf("Client %s disconnected.\n",cli_ip);   
        break;
      }
    }

    close(connfd);
  }

  close(listenfd);

  return 0;
}

我添加的唯一行是粗体/用双星号括起来的。

我可以编译它并且我可以运行它但是当我打开localhost:54321时,我说连接但它永远不会完成加载/连接。

如果有帮助,这就是我的终端在打开localhost时显示的内容:54321。

Client 127.0.0.1 connected. 
From 127.0.0.1 port 42642: GET / HTTP/1.1
Host: localhost:54321
User-Agent: Mozilla/5.0 (X11; Ubuntu; Linux i686; rv:22.0) Gecko/20100101 Firefox/22.0
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
Accept-Language: en-US,en;q=0.5
Accept-Encoding: gzip, deflate
Connection: keep-alive

感谢您提出任何意见,建议,想法和批评。

2 个答案:

答案 0 :(得分:1)

HTTP要求行终止CRLF(\r\n)。您的代码仅使用\n,这可能会混淆一些基本的Web客户端。

更重要的是,您还需要使用额外的CRLF终止响应中的标头,并且Content-length标头中的值必须与标头后面发送的字节数相匹配。

这样的东西
const char* entity = "<html><body> Hello, World! </body></html>";
sprintf(response,"HTTP/1.1 200 OK\r\n Content-type:text/html\r\n Content-length: %d\r\n\r\n%s", strlen(entity), entity);

应该解决这个问题。

答案 1 :(得分:1)

您的标题和邮件正文之间需要一个空行。

另外,你给内容长度为200,但那不是真的,所以浏览器正在等待剩下的数据。