HTTP服务器的核心流程如下:
//http_server.c
//该文件包含了整个http服务器的实现
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<unistd.h>
#include<sys/stat.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<fcntl.h>
#include<arpa/inet.h>
#include<pthread.h>
#include<sys/wait.h>
typedef struct sockaddr sockaddr;
typedef struct sockaddr_in sockaddr_in;
#define SIZE (1024*4)
typedef struct HttpRequest{
char first_line[SIZE];
char* method;
char* url;
char* url_path;
char* query_string;
int content_length;
}HttpRequest;
//从socket中读取一行数据,HTTP请求中的换行符究竟是什么
//\n,\r,\r\n都可以进行处理
//核心思路,把未知问题转化成已知问题,把\r,\r\n往\n上转换
int ReadLine(int sock,char output[],int max_size)
{
//1.一个字符一个字符从socket中读取数据
char c='\0';
ssize_t i=0;//记录了Output缓冲区当前已经写了多少个字符
while(i<max_size)
{
ssize_t read_size=recv(sock,&c,1,0);
if(read_size<=0)
{
//由于此处希望能读到完整的行,如果还没读到换行,就读到了EOF,此时认为出错
return -1;
}
//2.判定当前字符是不是\r
//3.如果是,尝试读取下一个字符
if(c=='\r')
{
// a)如果下一个字符是\n
// 偷看牌
recv(sock,&c,1,MSG_PEEK);
if(c=='\n')
{
//真正的摸牌
recv(sock,&c,1,0);
}
else
{
// b)不是\n,把'\r'转换为'\n'
c='\n';
// 针对这两种情况,都把当前字符转化为\n
}
}
//此时无论分隔符是什么,c=='\n'
// 4.如果当前字符是\n直接结束函数,(这一行已经读完了)
output[i++]=c;
if(c=='\n')
{
break;
}
}
// 5.如果当前字符是一个普通字符,直接追加到输出结果中
output[i]='\0';
return i;
}
//解析首行,获取到其中的method和url
//首行格式刑如
//暂时不考虑带域名的情况的url
//:Get http://www.baidu.com/index.html?a=10&b=20 HTTP/1.1
int Split(char input[],const char* split_char,char* output[])
{
char *tmp=NULL;
char* p=strtok_r(input,split_char,&tmp);//避免线程不安全
int output_index=0;
while(p!=NULL)
{
output[output_index++]=p;
p=strtok_r(NULL,split_char,&tmp);
}
return output_index;
}
int ParseFirstLine(char first_line[],char **p_method,char** p_url)
{
char* tok[10]={0};
//使用Split函数对字符串进行切分,n表示切分结果有几个部分
int n=Split(first_line," ",tok);
if(n!=3)
{
printf("Split failed!n=%d\n",n);
return -1;
}
//此处还可进行更复杂的校验
*p_method=tok[0];
*p_url=tok[1];
return 0;
}
//url 形如:/index.html?a=10&b=20
// http://www.baidu.com/index.html?a=10&b=20(暂时不考虑)
int ParseUrl(char url[],char** p_url_path,char** p_query_string)
{
*p_url_path=url;
//查找问号所在的位置
char* p=ur