文章目录
- 1. 运行项目
- 2. 函数功能分析
- 3. 函数关系图
- 4. 注释版代码
- 5. 参考
tinythhpd是一个超精简的web服务器,作者是1999年写的,到现在仍然能正常运行。 这个web服务器涉及到进程、线程、管道、socket等内容,源码只有五六百行,非常适合初学者阅读。
运行项目
我直接运行make httpd的时候报错说找不到-lsocket,从网上搜到说,这个库会自动链接上,并且也不叫这个名字了。 于是删掉-lsocket,但是又提示未定义的引用pthread_create,原来-lpthread的参数要放到最后边才行。 然后最终运行以下命令,就生成httpd了。
代码语言:javascript复制gcc -o httpd httpd.c -lpthread
因为我本机的per在/usr/bin目录下,所以改一下perl脚本中的第一行。 然后执行:
代码语言:javascript复制./httpd &
然后在浏览器地址栏输入下面地址测试web服务器。
代码语言:javascript复制127.0.0.1:<端口号>
函数功能分析
代码语言:javascript复制void accept_request(int);//每次创建一个线程,执行这个函数,用来处理一次http请求。
void bad_request(int);//当客户端发来的请求格式错误时(比如http协议的某些字段印错了),会向客户端发送400信息。
void cat(int, FILE *);//把某个文件的内容全部写入到与客户端的那个连接中。
void cannot_execute(int)//;当服务器身程序执行错误时(比如服务器自己在建立管道,或者fork进程时发生错误),会调用此函数,会向客户端发送500的信息。
void error_die(const char *);//打印错误信息的函数
void execute_cgi(int, const char *, const char *, const char *);//如果是post请求,这个函数用来执行cgi脚本。
int get_line(int, char *, int);//从连接中读出一行数据。
void headers(int, const char *);//将http的开头的字段写入连接。
void not_found(int);//如果客户端请求的地址不存在,调用此函数,并给客户端返回404.
void serve_file(int, const char *);//负责给返回客户端所请求的文件,通过调用headers(),cat(),not_found,gen_line()几个函数来实现。
int startup(u_short *);//负责程序刚开始建立socket、绑定ip地址、监听连接等工作。
void unimplemented(int);//如果客户端发过来的不死get或post请求,会调用此函数。
程序阅读顺序:可以先大概看看这些函数的功能,再从主函数顺着看一下主体,主函数中先调用startup创建socket连接。 然后监听到连接请求之后再创建新线程执行accept_request来处理请求。
函数关系图
注释版代码
代码语言:javascript复制/* J. David's webserver */
/* This is a simple webserver.
* Created November 1999 by J. David Blackstone.
* CSE 4344 (Network concepts), Prof. Zeigler
* University of Texas at Arlington
*/
/* This program compiles for Sparc Solaris 2.6.
* To compile for Linux:
* 1) Comment out the #include <pthread.h> line.
* 2) Comment out the line that defines the variable newthread.
* 3) Comment out the two lines that run pthread_create().
* 4) Uncomment the line that runs accept_request().
* 5) Remove -lsocket from the Makefile.
*/
#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <ctype.h>
#include <strings.h>
#include <string.h>
#include <sys/stat.h>
#include <pthread.h>
#include <sys/wait.h>
#include <stdlib.h>
#define ISspace(x) isspace((int)(x))
#define SERVER_STRING "Server: jdbhttpd/0.1.0rn"
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);
/**********************************************************************/
/* A request has caused a call to accept() on the server port to
* return. Process the request appropriately.
* Parameters: the socket connected to the client */
/**********************************************************************/
void accept_request(int client)
{
char buf[1024];
int numchars;
char method[255];
char url[255];
char path[512];
size_t i, j;
struct stat st;
int cgi = 0; /* becomes true if server decides this is a CGI
* program */
char *query_string = NULL;
//从连接中读出一行数据,numchars存放所读到的数据长度。
numchars = get_line(client, buf, sizeof(buf));
i = 0; j = 0;
//将buf中的http“请求方式”字段复制到method中,遇到空格停止。
while (!ISspace(buf[j]) && (i < sizeof(method) - 1))
{
method[i] = buf[j];
i ; j ;
}
method[i] = '