从零编写c++之http服务器(3)-http服务

        http全称超文本传输协议,可调试性高,扩展性也强。上两个篇章我们已经拥有了epoll事件驱动框架和线程池处理网络事件,接下来我们要先写一个基础网络套接字,然后在此基础上扩展出http的套接字。献上类图如下

完整源码见<https://github.com/kwansoner/panda.git>

                                            

        可以看到我们有一个最顶层的基类ISocket,拥有一个方法fd返回描述符,增加这个接口时由于事件中心注册到epoll里需要int型的描述符。接下来在ISocket基础上派生出IServer与IClient两个基类。实例化出两个TCP类型的套接字类CStreamServer与CStreamClient。然后我们就可以组合的方式扩展出两个http套接字类,不用继承的原因是继承增加耦合,也没有必要用继承。由于HttpStream与HttpServer需要收听事件中心的事件回调,因此需要继承IEventHandle。

class ISocket
{
    public:
        virtual ~ISocket(){};
        
        // desc: 获取套接字描述符
        // param: void
        // return: 套接字描述符
        virtual int fd() = 0;
};
 
class IClient: public ISocket
{
    public:
        virtual ~IClient(){};
 
        // desc: 打开套接字
        // param: void
        // return: 0/成功 -1/失败
        virtual int start() = 0;
 
        // desc: 关闭套接字
        // param: void
        // return: 0/成功 -1/失败
        virtual int close() = 0;
 
        // desc: 设置套接字为非阻塞
        // param: block/是否非阻塞
        // return: 0/成功 -1/失败
        virtual int set_nonblock(bool nonblock) = 0;
 
        // desc: 连接到server
        // param: addr/server地址 port/端口
        // return: 0/成功 -1/失败
        virtual int connect(const std::string &addr, uint16_t port) = 0;
 
        // desc: 读取数据
        // param: buf/存放接收数据缓冲区 len/buf长度 flag/见man recv
        // return: 读取数据长度
        virtual ssize_t recv(void *buf, size_t len, int flags) = 0;
 
        // desc: 发送数据
        // param: buf/存放发送数据缓冲区 len/buf长度 flag/见man recv
        // return: 发送数据长度
        virtual ssize_t send(const void *buf, size_t len, int flags) = 0;
};
 
 
class IServer: public ISocket
{
    public: 
        virtual ~IServer(){};    
 
        // desc: 设置套接字为非阻塞
        // param: block/是否非阻塞
        // return: 0/成功 -1/失败
        virtual int set_nonblock(bool block) = 0;
 
        // desc: 打开套接字
        // param: backlog/内核连接队列最大长度
        // return: 0/成功 -1/失败
        virtual int start(size_t backlog) = 0;
 
        // desc: 关闭套接字
        // param: void
        // return: 0/成功 -1/失败
        virtual int close() = 0;
 
        // desc: 套接字是否关闭
        // param: void
        // return: true/关闭 false/未关闭
        virtual bool isclose() = 0;
 
        // desc: 返回一个新的连接, 需要手动释放内存
        // param: void
        // return: NULL/错误    NOT NULL/新的连接
        virtual IClient *accept() = 0;
};
        首先我们在HttpServer的start函数中启动一个CStreamServer。接下来设置为非阻塞套接字,这样才更高效。然后将套接字注册到事件中心中去,就可以等待事件通知了。

        有可读事件到达时会回调handle_in接口,对于server可读就是新的连接建立了。这里需要注意一点就是,一次可读事件产生并未意味着一个连接建立,在这里需要循环的读取直到返回EAGAIN或者EWOULDBLOCK(先前设置为非阻塞)。同时由于我们事件中心的epoll检测是设置为边缘触发的。所以这里不读取完全是很可怕的。accept读取连接会返回一个新建的CStreamClien对象,代表着这个新的连接。然后传入HttpStream构造函数。新创建的HttpStream对象会自行释放自己。

int HttpServer::start(size_t backlog)
{
    if(!m_server.isclose())     //has start
        return 0;
        
    if(m_server.start(backlog) < 0)
        return -1;
 
    if(m_server.set_nonblock(true) < 0)
        errsys("set socket non block failed\n");
 
    return register_event(m_server);
}
void HttpServer::handle_in(int fd)
{
    /*
    * 读取所有建立的连接
    */
    do{
        Socket::IClient *newconn = m_server.accept();
        if(newconn == NULL){
            if(errno == EAGAIN || errno == EWOULDBLOCK || errno == EINTR){
                break;
            }else{
                errsys("sockfd[%d] accept error\n", fd);
                close();
                return;
            }
        }
        
        trace("socket[%d] accept a conn\n", fd);
        HttpStream *httpstream = new HttpStream(newconn);    // self release 
        assert(httpstream != NULL);
 
    }while(true);
}
      新的连接会接收到请求,然后我们对数据使用CHttpRequest对象进行解析。然后我们开始根据http的方法进行处理。这里只处理了GET方法。然后读取出文件后开始构造CHttpRespose对象,需要注意一点是CHttpRespose里配置的Body最大是64k,这里没有进行分包处理。因此不宜传输大文件试验。构造好CHttpRespose回复包后我们调用其serialize进行序列化后发送出去,这样就完成了一个http事务。同时这里为了处理简单,采取了短链接。即回复报文中Connection头部为close。

void HttpStream::handle_in(int fd)
{
    Pthread::CGuard guard(m_readbuffmutex);
    ssize_t nread = m_client->recv(m_readbuff, READBUFF_LEN, MSG_DONTWAIT);
    if((nread < 0 && nread != EAGAIN) || nread == 0){    // error or read EOF
        close();
        return;
    }else if(nread < 0 && nread == EAGAIN){         
        errsys("non data to read\n");
        return; 
    }    
    m_readbuff[nread] = 0x00;
 
    Http::CHttpRequest httprequest;
    if(httprequest.load_packet(m_readbuff, nread) < 0){
        error("parse package error\n");
        return;
    }
 
    trace("socket[%d] receive <--- %ld bytes\n", fd, nread);
    Http::IHttpRespose *respose = handle_request(httprequest);
    if(respose != NULL){
        m_client->send(respose->serialize(), respose->size(), 0);
        delete respose;
    }
}
void HttpStream::handle_close(int fd)
{
    trace("socket[%d] handle close\n", fd);
    delete this;
}
Http::IHttpRespose *HttpStream::handle_request(Http::IHttpRequest &request)
{
    const std::string &method = request.method();
    const std::string &url = request.url();
 
    std::string dname, bname;
    split_url(url, dname, bname);
 
    Http::CHttpRespose *respose = new Http::CHttpRespose;    
    std::string filepath = http_path_handle(dname, bname);
    if(method == "GET"){
 
        std::string extention = extention_name(filepath);
        if(extention.empty() || access(filepath.c_str(), R_OK) < 0){
 
            errsys("access %s error\n", filepath.c_str());
            
            respose->set_version(HTTP_VERSION);
            respose->set_status("404", "Not Found");
            respose->add_head(HTTP_HEAD_CONNECTION, "close");
            return respose;
        }
 
        
        struct stat filestat;
        stat(filepath.c_str(), &filestat);
        const size_t filesize = filestat.st_size;
 
        char *fbuff = new char[filesize];
        assert(fbuff != NULL);
 
        FILE *fp = fopen(filepath.c_str(), "rb");
        if(fp == NULL || fread(fbuff, filesize, 1, fp) != 0x01){
        
            delete fbuff;
 
            respose->set_version(HTTP_VERSION);
            respose->set_status("500", "Internal Server Error");
            respose->add_head(HTTP_HEAD_CONNECTION, "close");
            return respose;
        }
        
        fclose(fp);
 
        char sfilesize[16] = {0x00};
        snprintf(sfilesize, sizeof(sfilesize), "%ld", filesize);
 
        respose->set_version(HTTP_VERSION);
        respose->set_status("200", "OK");
        respose->add_head(HTTP_HEAD_CONTENT_TYPE, http_content_type(extention));
        respose->add_head(HTTP_HEAD_CONTENT_LEN, sfilesize);
        respose->add_head(HTTP_HEAD_CONNECTION, "close");
        respose->set_body(fbuff, filesize);
        delete fbuff;
    }
 
    return respose;
}
        最终完成代码后执行make编译,在Bin目录下生成名为panda的执行程序。执行后在浏览器上输入ip和8080端口即可打开网页, have fun!

 
--------------------- 
作者:kwanson 
来源:CSDN 
原文:https://blog.csdn.net/kwanson/article/details/81194214 
版权声明:本文为博主原创文章,转载请附上博文链接!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值