tinyhttpd源码分析

前言

最近主要是做了一些开源项目的源码分析工作,有c项目也有python项目,想提升一下内功,今天分享一下tinyhttpd源码分析的成果。tinyhttpd是一个非常轻量型的http服务器,c代码500行左右,可以帮助我们了解http服务器运行的实质。在分析之前,我们先说一下http报文。

HTTP请求

http请求由三部分组成,分别是:起始行、消息报头、请求正文

Request Line<CRLF>

Header-Name: header-value<CRLF>

Header-Name: header-value<CRLF>

//一个或多个,均以<CRLF>结尾

<CRLF>

body//请求正文

1、起始行以一个方法符号开头,以空格分开,后面跟着请求的URI和协议的版本,格式如下:

Method Request-URI HTTP-Version CRLF

2、请求方法(所有方法全为大写)有多种,各个方法的解释如下:其中 Method表示请求方法;Request-URI是一个统一资源标识符;HTTP-Version表示请求的HTTP协议版本;CRLF表示回车和换行(除了作为结尾的CRLF外,不允许出现单独的CR或LF字符)。

  1. GET 请求获取Request-URI所标识的资源
  2. POST 在Request-URI所标识的资源后附加新的数据
  3. HEAD 请求获取由Request-URI所标识的资源的响应消息报头
  4. PUT 请求服务器存储一个资源,并用Request-URI作为其标识
  5. DELETE 请求服务器删除Request-URI所标识的资源
  6. TRACE 请求服务器回送收到的请求信息,主要用于测试或诊断
  7. CONNECT 保留将来使用
  8. OPTIONS 请求查询服务器的性能,或者查询与资源相关的选项和需求

应用举例: GET方法:在浏览器的地址栏中输入网址的方式访问网页时,浏览器采用GET方法向服务器获取资源,eg:

GET /form.html HTTP/1.1 (CRLF)

POST方法要求被请求服务器接受附在请求后面的数据,常用于提交表单。eg:

POST /reg.jsp HTTP/ (CRLF)

Accept:image/gif,image/x-xbit,... (CRLF)

...

HOST:www.guet.edu.cn (CRLF)

Content-Length:22 (CRLF)

Connection:Keep-Alive (CRLF)

Cache-Control:no-cache (CRLF)

(CRLF) //该CRLF表示消息报头已经结束,在此之前为消息报头

user=jeffrey&pwd=1234 //此行以下为提交的数据

TinyHttpd源码分析

tinyhttpd总共包含以下函数:

按照以上顺序,看一下浏览器和tinyhttpd交互的整个流程。建议源码阅读顺序: main -> startup -> accept_request -> execute_cgi 

注释版源码

注释版源码已经放到github上了,以后所有的源码分析都会上传github上。

不过这个项目并不能直接在Linux上编译运行。它本来是在solaris上实现的,貌似在socket和pthread的实现上和一般的Linux还是不一样的,需要修改一部分内容。至于如何修改大家参考这篇文章,我也将修改版上传到github上了,名称为tinyhttpd-0.1.0_for_linux,大家可以clone下来,直接make编译即可。下面演示一下如何运行tinyhttpd,编译完成的效果如下:

下面运行./httpd,并在浏览器中访问。

tinyhttpd默认cgi脚本是perl脚本,比如color.cgi,位于htdocs目录下。

下面我想用python来实现cgi脚本,添加一些页面,为了更加了解cgi程序的运行实质,不用python封装好的cgi模块,完全手工打造。首先在htdocs目录下添加一个register.html页面,html文档内容如下:

这是一个表单,action指向register.cgi,method为post。下面看一下register.cgi,其实是个python脚本。

代码的意思是从标准输入中读取post中的数据,并将显示数据输出到标准输出中,对比一下流程图,更好理解。下面看一下运行效果。

公众号的阅读代码的体验不是很好,
大家还是阅读原文,去我的博客看吧

注释后的代码位于github:https://github.com/qiyeboy/SourceAnalysis

原文发布于微信公众号 - 七夜安全博客(qiye_safe)

原文发表时间:2017-01-18

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏狂码一生

CentOS-7安装成功后配置网络连接

一、修改配置文件     vim /etc/sysconfig/network-scripts/ifcfg-ens33     将 ONBOOT=no 修改为...

3036
来自专栏magicsoar

windows下的C++ socket服务器(4)

void handleAccept(int socket_fd) { char buf[1024] = { '\0' }; string cmd...

2545
来自专栏linux驱动个人学习

预处理

预处理有很多,以下选取我实际用过和见过的: #ifdef 电脑程序语句,我们可以用它区隔一些与特定头文件、程序库和其他文件版本有关的代码。 1 #includ...

2703
来自专栏difcareer的技术笔记

JNI实现源码分析【五 结束语】正文

一开始打算用一篇文章来写的,结果写着写着,发现内容实在太多了,一篇文章显得很乱,有很多地方不知道先写哪个好,经过一段时间的构思后,决定用一个系列来写,分多个部分...

945
来自专栏Linyb极客之路

网络编程之通过cookie和session让http协议变得有状态

因为http协议是无状态的,也就是说每次连接之后就不会记住上一次连接,如果是要登录才能查看的信息,就对用户体验很不好了。这篇文章我们介绍在客户端和服务端如何使得...

1011
来自专栏WebDeveloper

item2实现ssh的免密登录

1872
来自专栏jouypub

Linux系统优化

net.ipv4.ip_local_port_range = 1024 65000

1703
来自专栏Python小屋

Python获取局域网内所有机器IP地址与网卡MAC地址

已发相关文章请参考Python获取本机所有网卡的MAC地址 本文代码使用ARP协议获取局域网内所有计算机的IP地址与MAC地址,思路是使用系统命令arp获取A...

5899
来自专栏刘望舒

Android网络编程(一)HTTP协议原理

前言 这篇文章是这个系列的开篇,作为移动开发者,开发的应用不免会对网络进行访问,虽然现在已经有很多的开源库帮助我们可以轻而易举的访问网络,但是我们仍要去了解网络...

2145
来自专栏张戈的专栏

Linux优化方法收集与整理

之前一直有博主要求整理下 VPS 主机优化方法,那么如果你是 VPS 主机(Linux),可以尝试一下了,尤其是 linux 系统的内核参数优化。 一、增加 S...

4027

扫码关注云+社区