基于C语言 HTTP 服务器客户端的实验

基于 HTTP 服务器/客户端的实验

实验内容

1、使用 C 语言分别实现最简单的 HTTP 服务器和 HTTP 客户端

  • 服务器监听 80 端口,收到 HTTP 请求,解析请求内容,回复 HTTP 应答
  • 对于本地存在的文件,返回 HTTP 200 OK 和相应文件
  • 对于本地不存在的文件,返回 HTTP 404 File Not Found

2、服务器、客户端需要支持 HTTP Get 方法

3、服务器使用多线程支持多路并发

设计思路

1、已有代码框架已经能实现 socket 下的 echo,而实现 HTTP client 和 server 首先要做的是实现 HTTP GET 请求与相应。

2、在查询 HTTP GET 请求格式后,了解到 GET 请求报文格式如下图:

因此,本设计在 http-client.c 文件里添加了字符串 request_head 来生成请求报文,并在程序运行时通过 scanf 获得用户输入的请求文件路径。在加上请求头部后,Client 将请求发送给 Server

3、Server 在收到 HTTP GET 请求格式后,了解到响应报文格式如下图:

因此,本设计在 http-server.c 文件里添加了 int msg_handler(char * msg, char * path) 函数来解析请求报文,并提取出请求文件的地址。

4、Server 在打开目标文件后,读取目标文件的内容,并将内容传递至 return_messagereturn_message 此外还会附上必要的响应头部属性最后 Server 会将 return_message 发送给 Client

5、Server 为实现多线程发送文件的功能,本设计增加了 支持 pthread 相关函数。通过 pthread_detach 请求处理线程来实现多路并发。相关代码如下:

复制代码
while(1) {
    pthread_t thread;
    pthread_create(&thread, NULL, handle_request, &s);
    pthread_detach(thread);
}

测试项目

1、所实现的 HTTP 服务器、客户端可以完成上述功能

使用客户端连续多次获取文件以及请求不存在的文件

本客户端通过与 SimpleHTTPServer 建立连接来测试客户端连续多次获得文件的能力,如下图:

左侧 Xterm 界面显示了 Client 接收文件的反馈。Client 在控制台打印输出了请求的内容和相应的内容。

左侧 Xterm 里,Client 分别请求了 1.txt2.txt 以及 3.txt

1.txt 里的内容为 crz is great 12.txt 不存在,3.txt 里的内容为 crz is great。 左边输出内容代表三个请求均反馈正确,且正常收取了 1.txt3.txt 两个文件。接收的文件名在原文件名前增加 recv_ ,如下图:

同时启动多个客户端进程分别获取文件

通过 pthread 支持多线程多路并发,本设计支持同时启动多个客户端进程分别获取文件,具体的测试实现是借助 wget 10.0.0.1/1.txt & wget 10.0.0.1/3.txt 实现两个 wget 指令同时请求,测试结果如下:

由图可知, 1.txt3.txt 均被顺利接收。

2、使用"python -m SimpleHTTPServer 80"和 wget 分别替代服务器和客户端,对端可以正常工作

上述测试方式即可验证每一端均可正常运行。另外为了测试本设计中的 ClientServer 之间可以正常通行,增加了下述测试:

即多个 ClientServer 发送请求。结果如下:

遇到的问题

在实验过程中,通过抓包发现,ClientSimpleHTTPServer 发送请求信号后, 如果多次发送请求,SimpleHTTPServer 在第一次正常返回响应后,TCP 会发出 RST 信号,导致 Client 如果不重新建立连接就会发生异常。

因此本设计中,如果 ClientSimpleHTTPServer 连续发包的话,需要在每次发包后重连。实现代码及其注释如下:

复制代码
    /* The following part used depends on whether uses SimpleHttpServer. 
    If you want to use SimpleHttpServer, you should enable this part
    because SimpleHttpServer will send TCP RST signal after once connection.
    */ 

    /* 
    close(sock);
    sock = socket(AF_INET, SOCK_STREAM, 0);
    connect(sock, (struct sockaddr *)&server, sizeof(server));
    */

参考文献及网站

1\] [https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers "https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers") \[2\] [https://www.cnblogs.com/an-wen/p/11180076.html](https://www.cnblogs.com/an-wen/p/11180076.html "https://www.cnblogs.com/an-wen/p/11180076.html")

相关推荐
津津有味道14 小时前
易语言TCP服务端接收刷卡数据并向客户端读卡器发送指令
服务器·网络协议·tcp·易语言
Fᴏʀ ʏ꯭ᴏ꯭ᴜ꯭.15 小时前
Keepalived VIP迁移邮件告警配置指南
运维·服务器·笔记
Genie cloud15 小时前
1Panel SSL证书申请完整教程
服务器·网络协议·云计算·ssl
Tingjct15 小时前
【初阶数据结构-二叉树】
c语言·开发语言·数据结构·算法
!chen16 小时前
linux服务器静默安装Oracle26ai
linux·运维·服务器
莫大33016 小时前
2核2G云服务器PHP8.5+MySQL9.0+Nginx(LNMP)安装WordPress网站详细教程
运维·服务器·nginx
飞机和胖和黄16 小时前
考研之王道C语言第三周
c语言·数据结构·考研
醉颜凉16 小时前
【LeetCode】打家劫舍III
c语言·算法·leetcode·树 深度优先搜索·动态规划 二叉树
一匹电信狗16 小时前
【LeetCode_21】合并两个有序链表
c语言·开发语言·数据结构·c++·算法·leetcode·stl
2501_9277730717 小时前
imx6驱动
linux·运维·服务器