简介
- 理解http报文格式
- socket+epoll+多线程框架
- 请求和响应类封装
- 正则表达式解析头部
- 请求文件读取
- php-cgi解析php文件
理解http报文格式
socket+epoll+多线程框架
请求和响应类封装
为了方便操作以及功能模块的拆分,这里我们将请求内容和响应内容进行封装。这里我们只列出头文件,封装类中我们仅仅做了关系字段的提取。
Request.h
请求报文进行封装,我们将请求的内容放到body中,然后调用parse进行解析,解析后的数据分别存放到定义的成员变量中。
#ifndef REQUEST_H
#define REQUEST_H
#include
using namespace std;
#include
#include
Reponse.h
对响应报文进行封装,重要是响应状态吗、协议、Content-Length等。通过getData对头部进行拼装。
#ifndef RESPONSE_H
#define RESPONSE_H
#include
using namespace std;
#include
正则表达式解析头部
由于我的gcc版本为4.4.7,无法使用c++11 标准正则表达式。这里就是用了boost的Regex库。使用方法和代码一致。关于正则表达式的知识大家可在正则表达式基础上学习。
前面我们学习到http请求数据格式为 起始行+首部+请求体。所以我们直接匹配第一行数据。然后将方法、路径匹配出来。
boost::regex reg("^(\\w+) /(\\w*([.]\\w+)?) HTTP/1");
boost::smatch sm;
regex_search(body,sm,reg);
if(sm.size()==0) {
return false;
}else{
cout<<"Regex =>"<
正则表达式括号的内容代表我们要匹配的字符串,sm的第一个下标数据是匹配的完整的字符串,所以我们从sm[1]开始取值。
请求文件读取
前面读取到请求路径后,我们就可以去读取对应文件。
//读写文件
int fileSize=0;
FILE *file=fopen(path.c_str(),"r");
if(file!=NULL) {
fseek(file,0L,SEEK_END);
//获取文件大小
fileSize=ftell(file);
fseek(file,0L,SEEK_SET);
}else{
cout<<"file read error"<
接着就可以将文件数据发送出去
if(file!=NULL) {
char dataBuf[1024*100]={0};
while(true) {
int len= fread(dataBuf,1,sizeof(dataBuf),file);
if(len<=0) {
break;
}
cout<<"===>send Data "<sendData(dataBuf,len);
}
fclose(file);
}
注意在发送文件数据前,我们先要想头部发送出去
resp.setContentLength(fileSize);
// if(sm[3].str().empty()) {
resp.addHead("Content-Type","text/html");
// }else{
// resp.addHead("Content-Type","image/jpeg");
// }
resp.addHead("Server","eric http");
string headsStr=resp.getData();
cout<<"<<< "<sendData(headsStr.c_str(),headsStr.size());
一个简单的http容器我们搭建完成了。接下来就是如何支持php。
php-cgi解析php文件
首先先要安装php。可以参考Linux下php安装
我们可以先写一个简单的php代码测试一下。
然后在命令行使用php-cgi
php-cgi index.php > index.php.html
成功生成了index.php.html。接下来我们回到代码:
if(req.getPostfix()==".php") {
string cmd="php-cgi ";
string resFilePath;
resFilePath.append(path).append(".html");
cmd.append(path).append(" > ").append(resFilePath);
cout<
首先我们判断请求路径的后缀是否为.php。如果是则执行php-cgi的命令并生成文件。然后得到文件路径,接下来就和正常读取文件并发送数据的流程一致了。