位置: IT常识 - 正文

实现自己的http server loop_in_codes C++博客

编辑:rootadmin
实现自己的http server - loop_in_codes - C++博客实现自己的http serverWrite your own http server author : Kevin Ly 实现自己的http server

推荐整理分享实现自己的http server loop_in_codes C++博客,希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:,内容如对您有帮助,希望把文章链接给更多的朋友!

Write your own http server

author : Kevin Lynx

Why write your own?

看这个问题的人证明你知道什么是http server,世界上有很多各种规模的http server,为什么要自己实现一个?其实没什么理由。我自己问自己,感觉就是在自己娱乐自己,或者说只是练习下网络编程,或者是因为某日我看到某个库宣称自己附带一个小型的http server时,我不知道是什么东西,于是就想自己去实现一个。

What's httpd ?

httpd就是http daemon,这个是类unix系统上的名称,也就是http server。httpd遵循HTTP协议,响应HTTP客户端的request,然后返回response。 那么,什么是HTTP协议?最简单的例子,就是你的浏览器与网页服务器之间使用的应用层协议。虽然官方文档说HTTP协议可以建立在任何可靠传输的协议之上,但是就我们所见到的,HTTP还是建立在TCP之上的。 httpd最简单的response是返回静态的HTML页面。在这里我们的目标也只是一个响应静态网页的httpd而已(也许你愿意加入CGI特性)。

More details about HTTP protocol

在这里有必要讲解HTTP协议的更多细节,因为我们的httpd就是要去解析这个协议。 关于HTTP协议的详细文档,可以参看rfc2616。但事实上对于实现一个简单的响应静态网页的httpd来说,完全没必要读这么一分冗长的文档。在这里我推荐<HTTP Made Really Easy>,以下内容基本取自于本文档。

- HTTP协议结构 HTTP协议无论是请求报文(request message)还是回应报文(response message)都分为四部分: * 报文头 (initial line ) * 0个或多个header line * 空行(作为header lines的结束) * 可选body HTTP协议是基于行的协议,每一行以\r\n作为分隔符。报文头通常表明报文的类型(例如请求类型),报文头只占一行;header line 附带一些特殊信息,每一个header line占一行,其格式为name:value,即以分号作为分隔;空行也就是一个\r\n;可选body通常 包含数据,例如服务器返回的某个静态HTML文件的内容。举个例子,以下是一个很常见的请求报文,你可以截获浏览器发送的数据 包而获得:

1 GET /index.html HTTP/1.1 2 Accept-Language: zh-cn 3 Accept-Encoding: gzip, deflate 4 User-Agent: Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1; .NET CLR 2.0.50727; MAXTHON 2.0) 5 Host: localhost 6 Connection: Keep-Alive 7 我为每一行都添加了行号,第1行就是initial line,2-6行是header lines,7行是一个header line的结束符,没有显示出来。 以下是一个回应报文: 1 HTTP/1.1 200 OK 2 Server: klhttpd/0.1.0 3 Content-Type: text/html 4 Content-Length: 67 5 6 <head><head><title>index.html</title></head><body>index.html</body> 第6行就是可选的body,这里是index.html这个文件的内容。

- HTTP request method 因为我们做的事服务器端,所以我们重点对请求报文做说明。首先看initial line,该行包含几个字段,每个字段用空格分开,例 如以上的GET /index.html HTTP/1.1就可以分为三部分:GET、/index.html、HTTP/1.1。其中第一个字段GET就是所谓的request method。它表明请求类型,HTTP有很多method,例如:GET、POST、HEAD等。

就我们的目标而言,我们只需要实现对GET和HEAD做响应即可。

GET是最普遍的method,表示请求一个资源。什么是资源?诸如HTML网页、图片、声音文件等都是资源。顺便提一句,HTTP协议 中为每一个资源设置一个唯一的标识符,就是所谓的URI(更宽泛的URL)。 HEAD与GET一样,不过它不请求资源内容,而是请求资源信息,例如文件长度等信息。

实现自己的http server  loop_in_codes  C++博客

- More detail 继续说说initial line后面的内容: 对应于GET和HEAD两个method,紧接着的字段就是资源名,其实从这里可以看出,也就是文件名(相对于你服务器的资源目录),例 如这里的/index.html;最后一个字段表明HTTP协议版本号。目前我们只需要支持HTTP1.1和1.0,没有多大的技术差别。

然后是header line。我们并不需要关注每一个header line。我只罗列有用的header line : - Host : 对于HTTP1.1而言,请求报文中必须包含此header,如果没有包含,服务器需要返回bad request错误信息。 - Date : 用于回应报文,用于客户端缓存数据用。 - Content-Type : 用于回应报文,表示回应资源的文件类型,以MIME形式给出。什么是MIME?它们都有自己的格式,例如: text/html, image/jpg, image/gif等。 - Content-Length : 用于回应报文,表示回应资源的文件长度。

body域很简单,你只需要将一个文件全部读入内存,然后附加到回应报文段后发送即可,即使是二进制数据。

- 回应报文 之前提到的一个回应报文例子很典型,我们以其为例讲解。首先是initial line,第一个字段表明HTTP协议版本,可以直接以请求 报文为准(即请求报文版本是多少这里就是多少);第二个字段是一个status code,也就是回应状态,相当于请求结果,请求结果 被HTTP官方事先定义,例如200表示成功、404表示资源不存在等;最后一个字段为status code的可读字符串,你随便给吧。

回应报文中最好跟上Content-Type、Content-Length等header。

具体实现 正式写代码之前我希望你能明白HTTP协议的这种请求/回应模式,即客户端发出一个请求,然后服务器端回应该请求。然后继续这个过程(HTTP1.1是长连接模式,而HTTP1.0是短连接,当服务器端返回第一个请求时,连接就断开了)。 这里,我们无论客户端,例如浏览器,发出什么样的请求,请求什么资源,我们都回应相同的数据:

/**//*阻塞地接受一个客户端连接*/SOCKETcon=accept(s,0,0);/**//*recvrequest*/charrequest[1024]={0};ret=recv(con,request,sizeof(request),0);printf(request);/**//*whateverwerecv,wesend200response*/{charcontent[]="<head><head><title>index.html</title></head><body>index.html</body>";charresponse[512];sprintf(response,"HTTP/1.1200OK\r\nContent-Type:text/html\r\nContent-Length:%d\r\n\r\n%s",strlen(content),content);ret=send(con,response,strlen(response),0);}closesocket(con);

程序以最简单的阻塞模式运行,我们可以将重点放在协议的分析上。运行程序,在浏览器里输入http://localhost:8080/index.html,然后就可以看到浏览器正常显示content中描述的HTML文件。假设程序在8080端口监听。

现在你基本上明白了整个工作过程,我们可以把代码写得更全面一点,例如根据GET的URI来载入对应的文件然后回应给客户端。其实这个很简单,只需要从initial line里解析出(很一般的字符串解析)URI字段,然后载入对应的文件即可。例如以下函数:

voidhttp_response(SOCKETcon,constchar*request){/**//*getthemethod*/char*token=strtok(request,"");char*uri=strtok(0,"");charfile[64];sprintf(file,".%s",uri);{/**//*loadthefilecontent*/FILE*fp=fopen(file,"rb");if(fp==0){/**//*response404statuscode*/charresponse[]="HTTP/1.1404NOTFOUND\r\n\r\n";send(con,response,strlen(response),0);}else{/**//*responsetheresource*//**//*first,loadthefile*/intfile_size;char*content;charresponse[1024];fseek(fp,0,SEEK_END);file_size=ftell(fp);fseek(fp,0,SEEK_SET);content=(char*)malloc(file_size+1);fread(content,file_size,1,fp);content[file_size]=0;sprintf(response,"HTTP/1.1200OK\r\nContent-Type:text/html\r\nContent-Length:%d\r\n\r\n%s",file_size,content);send(con,response,strlen(response),0);free(content);}}}

其他 要将这个简易的httpd做完善,我们还需要注意很多细节。包括:对不支持的method返回501错误;对于HTTP1.1要求有Host这个header;为了支持客户端cache,需要添加Date header;支持HEAD请求等。

相关下载中我提供了一个完整的httpd library,纯C的代码,在其上加上一层资源载入即可实现一个简单的httpd。在这里我将对代码做简要的说明: evbuffer.h/buffer.c : 取自libevent的buffer,用于缓存数据; klhttp-internal.h/klhttp-internal.c :主要用于处理/解析HTTP请求,以及创建回应报文; klhttp-netbase.h/klhttp-netbase.c :对socket api的一个简要封装,使用select模型; klhttp.h/klhttp.c :库的最上层,应用层主要与该层交互,这一层主要集合internal和netbase。 test_klhttp.c :一个测试例子。

相关下载: klhttpd 文中相关代码

参考资料:

http://www.w3.org/Protocols/rfc2616/rfc2616.htmlhttp://jmarshall.com/easy/http/http://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html

本文链接地址:https://www.jiuchutong.com/zhishi/304100.html 转载请保留说明!

上一篇:【HDLBits刷题笔记】13 Finite State Machines(刷题笔贴吧)

下一篇:HTTP Tunneling (HTTP Proxy Socket Client)

  • 增值税专用发票抵扣期限
  • 异地提供建筑服务
  • 现金流量表财务费用怎么填
  • 实验用品包括实验仪器吗
  • 分公司注销需要总公司法人实名确认吗
  • 资产处置损益是营业外收入吗
  • 销货清单和送货单的区别
  • 建筑企业预收款开票
  • 设备售后回租需交税吗
  • 小规模纳税人每个月需要报什么税
  • 电梯提供安装的增值税税率最新的
  • 费用报销凭证是出纳的吗
  • 备用金可以银行贷款吗
  • 收到银行存款怎么记账
  • 商业折扣如何开票
  • 建筑业附加税税率是多少
  • 月末计提税金数据从哪得来的
  • 到底如何理解参数方程
  • 应发工资包含
  • 造价咨询暂定级能接业务吗
  • 华为手机屏幕有个圆点怎么取消
  • 本期已缴税额不能大于本期应纳税额
  • 电冰箱一天用多少电费正常
  • 原始凭证的分类依据有哪些?分别能分为哪些类型?
  • 别人从我公司走账怎么做账务处理
  • 返还土地出让金如何追缴的
  • 生产型企业出口不是自己生产的东西
  • php数组求最大值
  • 大雾山国家公园在哪里
  • PHP+Jquery与ajax相结合实现下拉淡出瀑布流效果【无需插件】
  • phpwechat
  • 蓝桥杯第十四届模拟赛第三期
  • 滑模控制理论与应用研究pdf
  • 计算机视觉算法
  • kill命令机制
  • 建筑行业预交增值税怎么结转
  • 不同情况不同处理的句子
  • 一般公司是什么企业类型
  • 客户的赔偿金会计分录
  • mysql复制表语句
  • 投资款不打入公司账户可以吗
  • 工程材料发票备注要求
  • 砂石可以开专票吗
  • 委托加工应税消费品是指委托方提供原料和主要材料
  • 一般纳税人报税流程详细操作
  • 风险纳税人认定条件2020
  • 可供出售金融资产公允价值变动
  • 施工审图费计入什么科目
  • 房地产开发公司是做什么的
  • 分公司和总公司在一个地方
  • 债权人和债务人的区别
  • 预付账款业务
  • 企业商品销售收入确认条件
  • 技术服务发票怎么做成本
  • 购入一辆运输汽车,价值5.6万元,款项尚未支付
  • 日记账的设置和登记
  • mysql 报错
  • win7怎么清除记录
  • vista桌面为什么不能登陆
  • ubuntu服务
  • win7盘符不见了
  • image driver
  • mac安全设置在哪里
  • windows7右下角网络图标不见了怎么办
  • 微软推送win11
  • linux双网卡双ip配置
  • Win10 mobile 10586.107升级后会出现哪些bug?
  • Javascript事件实例详解
  • python怎么写链表
  • linux中使用最多的命令
  • 深入理解计算机系统
  • python生成随机数据
  • jquery去除class
  • android draglistview中拖动异常显示,无法移动到原先位置
  • Android的事件处理机制是一种______机制
  • 南京税务局几点上班几点下班
  • 车辆购置税如何入账
  • 省税务局属于什么级别
  • 收心归位经典语录短句
  • 医生兼职取酬违反什么纪律
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设