一个几百行代码实现的http服务器tinyhttpd
- /* 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>
- #include <stdint.h>
- #define ISspace(x) isspace((int)(x))
- #define SERVER_STRING "Server: jdbhttpd/0.1.0\r\n"
- #define STDIN 0
- #define STDOUT 1
- #define STDERR 2
- void accept_request(void *);
- 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(void *arg)
- {
- int client = (intptr_t)arg;
- char buf[];
- size_t numchars;
- char method[];
- char url[];
- char path[];
- size_t i, j;
- struct stat st;
- int cgi = ; /* becomes true if server decides this is a CGI
- * program */
- char *query_string = NULL;
- numchars = get_line(client, buf, sizeof(buf));
- i = ; j = ;
- while (!ISspace(buf[i]) && (i < sizeof(method) - ))
- {
- method[i] = buf[i];
- i++;
- }
- j=i;
- method[i] = '\0';
- if (strcasecmp(method, "GET") && strcasecmp(method, "POST"))
- {
- unimplemented(client);
- return;
- }
- if (strcasecmp(method, "POST") == )
- cgi = ;
- i = ;
- while (ISspace(buf[j]) && (j < numchars))
- j++;
- while (!ISspace(buf[j]) && (i < sizeof(url) - ) && (j < numchars))
- {
- url[i] = buf[j];
- i++; j++;
- }
- url[i] = '\0';
- if (strcasecmp(method, "GET") == )
- {
- query_string = url;
- while ((*query_string != '?') && (*query_string != '\0'))
- query_string++;
- if (*query_string == '?')
- {
- cgi = ;
- *query_string = '\0';
- query_string++;
- }
- }
- sprintf(path, "htdocs%s", url);
- if (path[strlen(path) - ] == '/')
- strcat(path, "index.html");
- if (stat(path, &st) == -) {
- while ((numchars > ) && strcmp("\n", buf)) /* read & discard headers */
- numchars = get_line(client, buf, sizeof(buf));
- not_found(client);
- }
- else
- {
- if ((st.st_mode & S_IFMT) == S_IFDIR)
- strcat(path, "/index.html");
- if ((st.st_mode & S_IXUSR) ||
- (st.st_mode & S_IXGRP) ||
- (st.st_mode & S_IXOTH) )
- cgi = ;
- if (!cgi)
- serve_file(client, path);
- else
- execute_cgi(client, path, method, query_string);
- }
- close(client);
- }
- /**********************************************************************/
- /* Inform the client that a request it has made has a problem.
- * Parameters: client socket */
- /**********************************************************************/
- void bad_request(int client)
- {
- char buf[];
- sprintf(buf, "HTTP/1.0 400 BAD REQUEST\r\n");
- send(client, buf, sizeof(buf), );
- sprintf(buf, "Content-type: text/html\r\n");
- send(client, buf, sizeof(buf), );
- sprintf(buf, "\r\n");
- send(client, buf, sizeof(buf), );
- sprintf(buf, "<P>Your browser sent a bad request, ");
- send(client, buf, sizeof(buf), );
- sprintf(buf, "such as a POST without a Content-Length.\r\n");
- send(client, buf, sizeof(buf), );
- }
- /**********************************************************************/
- /* Put the entire contents of a file out on a socket. This function
- * is named after the UNIX "cat" command, because it might have been
- * easier just to do something like pipe, fork, and exec("cat").
- * Parameters: the client socket descriptor
- * FILE pointer for the file to cat */
- /**********************************************************************/
- void cat(int client, FILE *resource)
- {
- char buf[];
- fgets(buf, sizeof(buf), resource);
- while (!feof(resource))
- {
- send(client, buf, strlen(buf), );
- fgets(buf, sizeof(buf), resource);
- }
- }
- /**********************************************************************/
- /* Inform the client that a CGI script could not be executed.
- * Parameter: the client socket descriptor. */
- /**********************************************************************/
- void cannot_execute(int client)
- {
- char buf[];
- sprintf(buf, "HTTP/1.0 500 Internal Server Error\r\n");
- send(client, buf, strlen(buf), );
- sprintf(buf, "Content-type: text/html\r\n");
- send(client, buf, strlen(buf), );
- sprintf(buf, "\r\n");
- send(client, buf, strlen(buf), );
- sprintf(buf, "<P>Error prohibited CGI execution.\r\n");
- send(client, buf, strlen(buf), );
- }
- /**********************************************************************/
- /* Print out an error message with perror() (for system errors; based
- * on value of errno, which indicates system call errors) and exit the
- * program indicating an error. */
- /**********************************************************************/
- void error_die(const char *sc)
- {
- perror(sc);
- exit();
- }
- /**********************************************************************/
- /* Execute a CGI script. Will need to set environment variables as
- * appropriate.
- * Parameters: client socket descriptor
- * path to the CGI script */
- /**********************************************************************/
- void execute_cgi(int client, const char *path,
- const char *method, const char *query_string)
- {
- char buf[];
- int cgi_output[];
- int cgi_input[];
- pid_t pid;
- int status;
- int i;
- char c;
- int numchars = ;
- int content_length = -;
- buf[] = 'A'; buf[] = '\0';
- if (strcasecmp(method, "GET") == )
- while ((numchars > ) && strcmp("\n", buf)) /* read & discard headers */
- numchars = get_line(client, buf, sizeof(buf));
- else if (strcasecmp(method, "POST") == ) /*POST*/
- {
- numchars = get_line(client, buf, sizeof(buf));
- while ((numchars > ) && strcmp("\n", buf))
- {
- buf[] = '\0';
- if (strcasecmp(buf, "Content-Length:") == )
- content_length = atoi(&(buf[]));
- numchars = get_line(client, buf, sizeof(buf));
- }
- if (content_length == -) {
- bad_request(client);
- return;
- }
- }
- else/*HEAD or other*/
- {
- }
- if (pipe(cgi_output) < ) {
- cannot_execute(client);
- return;
- }
- if (pipe(cgi_input) < ) {
- cannot_execute(client);
- return;
- }
- if ( (pid = fork()) < ) {
- cannot_execute(client);
- return;
- }
- sprintf(buf, "HTTP/1.0 200 OK\r\n");
- send(client, buf, strlen(buf), );
- if (pid == ) /* child: CGI script */
- {
- char meth_env[];
- char query_env[];
- char length_env[];
- dup2(cgi_output[], STDOUT);
- dup2(cgi_input[], STDIN);
- close(cgi_output[]);
- close(cgi_input[]);
- sprintf(meth_env, "REQUEST_METHOD=%s", method);
- putenv(meth_env);
- if (strcasecmp(method, "GET") == ) {
- sprintf(query_env, "QUERY_STRING=%s", query_string);
- putenv(query_env);
- }
- else { /* POST */
- sprintf(length_env, "CONTENT_LENGTH=%d", content_length);
- putenv(length_env);
- }
- execl(path, NULL);
- exit();
- } else { /* parent */
- close(cgi_output[]);
- close(cgi_input[]);
- if (strcasecmp(method, "POST") == )
- for (i = ; i < content_length; i++) {
- recv(client, &c, , );
- write(cgi_input[], &c, );
- }
- while (read(cgi_output[], &c, ) > )
- send(client, &c, , );
- close(cgi_output[]);
- close(cgi_input[]);
- waitpid(pid, &status, );
- }
- }
- /**********************************************************************/
- /* Get a line from a socket, whether the line ends in a newline,
- * carriage return, or a CRLF combination. Terminates the string read
- * with a null character. If no newline indicator is found before the
- * end of the buffer, the string is terminated with a null. If any of
- * the above three line terminators is read, the last character of the
- * string will be a linefeed and the string will be terminated with a
- * null character.
- * Parameters: the socket descriptor
- * the buffer to save the data in
- * the size of the buffer
- * Returns: the number of bytes stored (excluding null) */
- /**********************************************************************/
- int get_line(int sock, char *buf, int size)
- {
- int i = ;
- char c = '\0';
- int n;
- while ((i < size - ) && (c != '\n'))
- {
- n = recv(sock, &c, , );
- /* DEBUG printf("%02X\n", c); */
- if (n > )
- {
- if (c == '\r')
- {
- n = recv(sock, &c, , MSG_PEEK);
- /* DEBUG printf("%02X\n", c); */
- if ((n > ) && (c == '\n'))
- recv(sock, &c, , );
- else
- c = '\n';
- }
- buf[i] = c;
- i++;
- }
- else
- c = '\n';
- }
- buf[i] = '\0';
- return(i);
- }
- /**********************************************************************/
- /* Return the informational HTTP headers about a file. */
- /* Parameters: the socket to print the headers on
- * the name of the file */
- /**********************************************************************/
- void headers(int client, const char *filename)
- {
- char buf[];
- (void)filename; /* could use filename to determine file type */
- strcpy(buf, "HTTP/1.0 200 OK\r\n");
- send(client, buf, strlen(buf), );
- strcpy(buf, SERVER_STRING);
- send(client, buf, strlen(buf), );
- sprintf(buf, "Content-Type: text/html\r\n");
- send(client, buf, strlen(buf), );
- strcpy(buf, "\r\n");
- send(client, buf, strlen(buf), );
- }
- /**********************************************************************/
- /* Give a client a 404 not found status message. */
- /**********************************************************************/
- void not_found(int client)
- {
- char buf[];
- sprintf(buf, "HTTP/1.0 404 NOT FOUND\r\n");
- send(client, buf, strlen(buf), );
- sprintf(buf, SERVER_STRING);
- send(client, buf, strlen(buf), );
- sprintf(buf, "Content-Type: text/html\r\n");
- send(client, buf, strlen(buf), );
- sprintf(buf, "\r\n");
- send(client, buf, strlen(buf), );
- sprintf(buf, "<HTML><TITLE>Not Found</TITLE>\r\n");
- send(client, buf, strlen(buf), );
- sprintf(buf, "<BODY><P>The server could not fulfill\r\n");
- send(client, buf, strlen(buf), );
- sprintf(buf, "your request because the resource specified\r\n");
- send(client, buf, strlen(buf), );
- sprintf(buf, "is unavailable or nonexistent.\r\n");
- send(client, buf, strlen(buf), );
- sprintf(buf, "</BODY></HTML>\r\n");
- send(client, buf, strlen(buf), );
- }
- /**********************************************************************/
- /* Send a regular file to the client. Use headers, and report
- * errors to client if they occur.
- * Parameters: a pointer to a file structure produced from the socket
- * file descriptor
- * the name of the file to serve */
- /**********************************************************************/
- void serve_file(int client, const char *filename)
- {
- FILE *resource = NULL;
- int numchars = ;
- char buf[];
- buf[] = 'A'; buf[] = '\0';
- while ((numchars > ) && strcmp("\n", buf)) /* read & discard headers */
- numchars = get_line(client, buf, sizeof(buf));
- resource = fopen(filename, "r");
- if (resource == NULL)
- not_found(client);
- else
- {
- headers(client, filename);
- cat(client, resource);
- }
- fclose(resource);
- }
- /**********************************************************************/
- /* This function starts the process of listening for web connections
- * on a specified port. If the port is 0, then dynamically allocate a
- * port and modify the original port variable to reflect the actual
- * port.
- * Parameters: pointer to variable containing the port to connect on
- * Returns: the socket */
- /**********************************************************************/
- int startup(u_short *port)
- {
- int httpd = ;
- int on = ;
- struct sockaddr_in name;
- httpd = socket(PF_INET, SOCK_STREAM, );
- if (httpd == -)
- error_die("socket");
- memset(&name, , sizeof(name));
- name.sin_family = AF_INET;
- name.sin_port = htons(*port);
- name.sin_addr.s_addr = htonl(INADDR_ANY);
- if ((setsockopt(httpd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on))) < )
- {
- error_die("setsockopt failed");
- }
- if (bind(httpd, (struct sockaddr *)&name, sizeof(name)) < )
- error_die("bind");
- if (*port == ) /* if dynamically allocating a port */
- {
- socklen_t namelen = sizeof(name);
- if (getsockname(httpd, (struct sockaddr *)&name, &namelen) == -)
- error_die("getsockname");
- *port = ntohs(name.sin_port);
- }
- if (listen(httpd, ) < )
- error_die("listen");
- return(httpd);
- }
- /**********************************************************************/
- /* Inform the client that the requested web method has not been
- * implemented.
- * Parameter: the client socket */
- /**********************************************************************/
- void unimplemented(int client)
- {
- char buf[];
- sprintf(buf, "HTTP/1.0 501 Method Not Implemented\r\n");
- send(client, buf, strlen(buf), );
- sprintf(buf, SERVER_STRING);
- send(client, buf, strlen(buf), );
- sprintf(buf, "Content-Type: text/html\r\n");
- send(client, buf, strlen(buf), );
- sprintf(buf, "\r\n");
- send(client, buf, strlen(buf), );
- sprintf(buf, "<HTML><HEAD><TITLE>Method Not Implemented\r\n");
- send(client, buf, strlen(buf), );
- sprintf(buf, "</TITLE></HEAD>\r\n");
- send(client, buf, strlen(buf), );
- sprintf(buf, "<BODY><P>HTTP request method not supported.\r\n");
- send(client, buf, strlen(buf), );
- sprintf(buf, "</BODY></HTML>\r\n");
- send(client, buf, strlen(buf), );
- }
- /**********************************************************************/
- int main(void)
- {
- int server_sock = -;
- u_short port = ;
- int client_sock = -;
- struct sockaddr_in client_name;
- socklen_t client_name_len = sizeof(client_name);
- pthread_t newthread;
- server_sock = startup(&port);
- printf("httpd running on port %d\n", port);
- while ()
- {
- client_sock = accept(server_sock,
- (struct sockaddr *)&client_name,
- &client_name_len);
- if (client_sock == -)
- error_die("accept");
- /* accept_request(&client_sock); */
- if (pthread_create(&newthread , NULL, (void *)accept_request, (void *)(intptr_t)client_sock) != )
- perror("pthread_create");
- }
- close(server_sock);
- return();
- }
其实就是建立tcp连接,通过对数据包的解析是否有http的头字段来判断是不是http的,wireshark就是这样
下面是一个别人总结的图
这个我自己写了一下但是没实现到cgi那里,贴一个别人用windows实现的地址
https://blog.csdn.net/magictong/article/details/53201038
一个几百行代码实现的http服务器tinyhttpd的更多相关文章
- 一个只有99行代码的JS流程框架(二)
欢迎大家关注腾讯云技术社区-博客园官方主页,我们将持续在博客园为大家推荐技术精品文章哦~ 张镇圳,腾讯Web前端高级工程师,对内部系统前端建设有多年经验,喜欢钻研捣鼓各种前端组件和框架. 导语 前面写 ...
- (转)如何基于FFMPEG和SDL写一个少于1000行代码的视频播放器
原文地址:http://www.dranger.com/ffmpeg/ FFMPEG是一个很好的库,可以用来创建视频应用或者生成特定的工具.FFMPEG几乎为你把所有的繁重工作都做了,比如解码.编码. ...
- JELLY技术周刊 Vol.24 -- 技术周刊 · 实现 Recoil 只需百行代码?
蒲公英 · JELLY技术周刊 Vol.24 理解一个轮子最好的方法就是仿造一个轮子,很多框架都因此应运而生,比如面向 JS 开发者的 AI 工具 Danfo.js:参考 qiankun 的微前端框架 ...
- 一个只有99行代码的JS流程框架
张镇圳,腾讯Web前端高级工程师,对内部系统前端建设有多年经验,喜欢钻研捣鼓各种前端组件和框架. 最近一直在想一个问题,如何能让js代码写起来更语义化和更具有可读性. 上周末的时候突发奇想,当代码在运 ...
- 几百行代码实现一个 JSON 解析器
前言 之前在写 gscript时我就在想有没有利用编译原理实现一个更实际工具?毕竟真写一个语言的难度不低,并且也很难真的应用起来. 一次无意间看到有人提起 JSON 解析器,这类工具充斥着我们的日常开 ...
- 继续node爬虫 — 百行代码自制自动AC机器人日解千题攻占HDOJ
前言 不说话,先猛戳 Ranklist 看我排名. 这是用 node 自动刷题大概半天的 "战绩",本文就来为大家简单讲解下如何用 node 做一个 "自动AC机&quo ...
- 几百行代码写个Mybatis,原理搞的透透的!
作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获! 一.前言 Mybatis 最核心的原理也是它最便于使用的体现,为什么这说? 因为我们在使用 M ...
- IOS 作业项目(1) 关灯游戏 (百行代码搞定)
1,准备工作,既然要开关灯,就需要确定灯的灯的颜色状态 首先想到的是扩展UIColor
- Redux百行代码千行文档
接触Redux不过短短半年,从开始看官方文档的一头雾水,到渐渐已经理解了Redux到底是在做什么,但是绝大数场景下Redux都是配合React一同使用的,因而会引入了React-Redux库,但是正是 ...
随机推荐
- Hibernate 集合映射
Set映射: <?xml version="1.0" encoding="utf-8"?> <!DOCTYPE hibernate-mappi ...
- Golang教程:包
什么是包?为什么使用包? 到目前为止我们见到的 Go 程序都只有一个文件,文件中包含了一个main函数和几个其他函数.在实际中这种将所有代码都放在一个文件里的组织方式是不可行的.这样的组织方式使得代码 ...
- ElasticSearch基础入门
1.query查询表达式 Elasticsearch 提供一个丰富灵活的查询语言叫做 查询表达式 , 查询表达式(Query DSL)是一种非常灵活又富有表现力的 查询语言,它支持构建更加复杂和健壮的 ...
- [转]如何在 .Net Framework 4.0 项目上使用 OData?
本文转自:http://www.cnblogs.com/fiozhao/p/3536469.html 最新的 Microsoft ASP.NET Web API 2.1 OData 5.1.0 已只能 ...
- WPF简单的数据库查询
做一个简单WPF连接数据库的 控件类型和名称:DataGrid:dataGrid Button1 :Button1 Button: Button2 ...
- Eigen库矩阵运算使用方法
Eigen库矩阵运算使用方法 Eigen这个类库,存的东西好多的,来看一下主要的几个头文件吧: ——Core 有关矩阵和数组的类,有基本的线性代数(包含 三角形 和 自伴乘积 相关),还有相应对数组的 ...
- 文档类型DTD,DOCTYPE和浏览器模式
出处:http://blog.csdn.net/freshlover/article/details/11616563 浏览器从服务端获取网页后会根据文档的DOCTYPE定义显示网页,如果文档正确定义 ...
- shiro权限控制入门
一:权限控制两种主要方式 粗粒度 URL 级别权限控制和细粒度方法级别权限控制 1.粗粒度 URL 级别权限控制 可以基于 Filter 实现在数据库中存放 用户.权限.访问 URL 对应关系, 当前 ...
- IntelliJ IDEA+Mysql connecter/j JDBC驱动连接
在IntelliJ IDEA中用connecter/j jdbc驱动连接MYSQL 以下是解决过程,待整合...有点懒,有空再改 官方文档:https://www.cnblogs.com/cn-chy ...
- 基于bootstrap的内容折叠功能
加入js及css支持: <link rel="stylesheet" href="css/bootstrap.min.css"/> <scri ...