最近入门网络编程在看《TCP/IP网络编程》一书,对第24章的web服务器代码边看边写了一遍但是测试时发现了一些问题,所以想写笔记记录一下。
1.bug1:第一次写的时候没有初始化sklen,导致报错:accept failed。所以以后accept第三个参数都需要初始化,并且可以用errno来查看报错信息。
2. 接下来运行后会发现:收到的html网页有时全有时不全,会有其他的http请求,gdb调试报错:broken pipe,并且再次运行可执行文件时显示:bind failed。
经过查阅资料: 发现可能是客户端对应端口关闭。本来想用文章中的方法屏蔽信号,但是想到:bind failed是因为服务器端口处于time-wait状态,只有先发起fin报文的一端才会出现这种情况,和客户端先关闭的判断矛盾。并且还想知道为什么客户端会先关闭,所以想了一会。
后来突然发现:书中给的代码的content-length只有2048,而我自己传输的html大小远大于这个,所以修改完后就能收到全部网页了。并且wireshark抓包发现其他的http报文是因为:web页面是由多个对象组成的,我们有的只是html基本文件,基本文件会通过URL地址引用其他的对象因此会有不止一个http请求。
3.修改上述错误后运行会发现报错:segmentation fault。
使用gdb发现出错的线程不是传输html基本文件的线程。最后发现实在content_type函数中出错,这个函数是在打开file_name前使用并且其他的http报文格式和基本html文件的格式不一致:
而content_type函数会调用两次strtok函数将index和html分开,由于这个字符串没有"."因此第二次应该返回了NULL(没有验证)。
关于strtok函数可以看:中的第三点。
我的处理方式比较简单,将打开文件名这一操作放在handler函数中并且在content_type函数之前,这些多余的http请求处理方式与handler中其他错误处理方法一致。
还有一些bug
修改至此,可以基本实现简单的多线程web服务器,但是还是有两个问题:
- 偶尔会出现segmentation fault实在不知道怎么搞,也不想多花费时间在这里了。
- 因为对其他http请求send error,编译后第一次请求点X才能显示出基本页面:
但是服务器进程会关闭。现在还没有深入了解http,所以也没管。
总结
虽然还有一些bug,但是作为一个简单的web服务器是够了,并且接下来还需要继续学习,所以不纠结了。
并且之前的矛盾也就说的通了:负责发送数据的线程由于http的内容大小设置的太小了会使客户端端口提前关闭,接收信号后杀死进程,但是此时服务器端口还和其他http请求端口有连接,强行关闭后就会进入time-wait状态。
这也正好印证了多线程的一个缺点:一个线程出错,整个进程会被杀死,其他线程也受牵连。
现在了解的太浅,写的许多内容很模糊,希望以后看到了不是这样。