宏定义:
#define ISspace(x) isspace((int)(x))
#define SERVER_STRING "Server: jdbhttpd/0.1.0\r\n"
函数:
void accept_request(int);
void bad_request(int);
void cat(int, FILE *);
void cannot_execute(int);
void error_die(const char *);
void execute_cgi(int, const char *, const char *, const char *);
int get_line(int, char *, int);
void headers(int, const char *);
void not_found(int);
void serve_file(int, const char *);
int startup(u_short *);
void unimplemented(int);
处理流程:
main函数入手:
main函数变量:
int server_sock = -1; //服务器的套接字
u_short port = 0; //端口号
int client_sock = -1; //客户机的套接字
struct sockaddr_in client_name;//客户端的套接字信息
int client_name_len = sizeof(client_name);//客户端套接字的长度
pthread_t newthread; //线程标识符
main函数执行流程:
首先调用startup函数:server_sock = startup(&port);
由于port为0,则动态分配端口号,在startup中使用socket函数向系统注册一个socket通信,指定了IPV4和TCP协议,并使用bind函数对socket进行了绑定。如果端口号为0,则动态分配端口号。对服务器套接字建立监听,等待连接,最大连接数为5。
经过startup函数后,tinyhttpd已经建立了连接,获取了端口号,同时开始了网络的监听。此时在main函数中,将进入一个死循环,其中不断对server_sock也就是服务器套接字进行监听,如果存在连接请求,则接受连接请求并创建客户机套接字,使用该客户机套接字建立线程,线程执行函数是accept_request。然后继续监听。
关闭服务器套接字。
accept_request
请求处理最直接的函数,在此将对客户机的请求进行处理
首先调用get_line函数,这个函数用于从套接字中读取一行数据,并返回保存的字节数,详细的介绍后面再讲。
接下来进入一个循环,在这个循环中将从第一行读取客户端请求的方法并存入method数组中,也就是在读取的请求的第一行中找到第一个空格,这样就获取了客户端请求的方法,然后进行比较,如果客户端请求方法既不是“GET”方法,也不是“POST”方法,则向客户端回复unimplemented方法,提示客户端请求的方法无法执行,不被支持。如果请求的方法是post,则让cgi=1,代表要使用cgi程序。接下来跳过空白部分。跳过空白部分后就是HTTP 协议的uri,也就是请求的资源或者上传的路径,当然不论uri是什么,都是由服务器进行处理。
接下来判断请求的方法是不是get方法,如果是get方法,则让查询的query_string指针指向url,然后将query_string指向'?'或者'\0'处,'?'则代表查询,'\0'则代表结尾,如果是查询的话,则要令cgi=1,代表要执行cgi程序,将url分离为两部分,url指向资源部分,query_string指向查询部分。
接下来给path赋值htdocs+url的值,如果path的最后一个符号是'/' 或url的最后一个符号是'/',说明这是根目录,加载默认界面。
然后使用stat函数取得文件状态,如果失败的话,就说明文件不存在或者没有权限访问,向客户机回复not_found。正常的话则判断是否对文件具有可执行权限,可执行则让cgi = 1,如果cgi为0,则向客户端回送文件,为1则执行文件。
关闭客户机的socket连接。
execute_cgi
执行cgi程序
如果客户端的请求方法是get,则不断的从客户端读取头部并且舍弃,如果是post方法,就获取content-length的长度,将这个值赋给content_length变量。然后为进程创建管道并创建子进程。如果创建子进程时返回0,则说明程序处于子进程中,否则说明程序处于父进程。
将请求方法设置为环境变量,如果请求方法是get方法,将查询内容也设置到环境变量中,否则则保存请求方法的内容长度。执行参数一所代表的文件路径。
如果是在父进程中且是post方法,则接收客户机发送的数据,如果接收到子进程发送的数据,则将数据发送给客户机,等待子进程结束。
get_line
读取来自客户机的数据,读取到一行时返回数量
serve_file
向客户发送普通文件
startup
从特定端口监听web连接请求
在这个函数中,首先以IPV4和TCP协议向系统注册通信端口并绑定,成功后则等待连接,最大的连接数为5