美文网首页
Linux下Socket编程(五)——http容器(支持php)

Linux下Socket编程(五)——http容器(支持php)

作者: 第八区 | 来源:发表于2017-09-27 16:55 被阅读41次

简介

  • 理解http报文格式
  • socket+epoll+多线程框架
  • 请求和响应类封装
  • 正则表达式解析头部
  • 请求文件读取
  • php-cgi解析php文件

理解http报文格式

socket+epoll+多线程框架

请求和响应类封装

为了方便操作以及功能模块的拆分,这里我们将请求内容和响应内容进行封装。这里我们只列出头文件,封装类中我们仅仅做了关系字段的提取。

Request.h

请求报文进行封装,我们将请求的内容放到body中,然后调用parse进行解析,解析后的数据分别存放到定义的成员变量中。

#ifndef REQUEST_H
#define  REQUEST_H
#include <iostream>
using namespace std;
#include <string.h>
#include <map>
class Request{
private:
  string body;
  string method;
  string path;
  string protocol;
  string content;
  string postfix;
  map<string,string> headers;
  map<string,string> params;
public:
  Request();
  ~Request();
  void setBody(string body);
  bool parse();
  string getMethod();
  string getPath();
  string getProtocol();
  string getContent();
  string getPostfix();
  map<string,string> getParam();
  map<string,string> getHeaders();
};
#endif
Reponse.h

对响应报文进行封装,重要是响应状态吗、协议、Content-Length等。通过getData对头部进行拼装。

#ifndef RESPONSE_H
#define  RESPONSE_H
#include <iostream>
using namespace std;
#include <map>
#include <string.h>

class Response{
private:
  int code;
  string msg;
  string protocol;
  map<string,string> headers;
public:
  Response();
  ~Response();
  void setCode(int code);
  void setMsg(const char* msg);
  void setProtocol(const char protocol);
  void addHead(string,string);
  void setContentLength(int length);
  string getData();
};
#endif

正则表达式解析头部

由于我的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[1]<<"|"<<sm[2]<<"|"<<sm[3]<<endl;
        method=sm[1];
        path.clear();
        path.append("/").append(sm[2]);
        postfix=sm[3];
}

正则表达式括号的内容代表我们要匹配的字符串,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"<<endl;
}

接着就可以将文件数据发送出去

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 "<<len<<endl;
                client->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<<"<<< "<<headsStr.c_str();
client->sendData(headsStr.c_str(),headsStr.size());

一个简单的http容器我们搭建完成了。接下来就是如何支持php。

php-cgi解析php文件

首先先要安装php。可以参考Linux下php安装
我们可以先写一个简单的php代码测试一下。

<?php
phpinfo();
  ?>

然后在命令行使用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<<cmd<<endl;
        system(cmd.c_str());
        path=resFilePath;
}

首先我们判断请求路径的后缀是否为.php。如果是则执行php-cgi的命令并生成文件。然后得到文件路径,接下来就和正常读取文件并发送数据的流程一致了。

相关文章

网友评论

      本文标题:Linux下Socket编程(五)——http容器(支持php)

      本文链接:https://www.haomeiwen.com/subject/lcbuextx.html