Bootstrap

⚡通信管理模块⚡

目录

🌳.通信管理模块的介绍

🌳.通信模块的设计

🌳.ListShow函数

🌳.UpLoad函数

🌳.DownLoad函数

下载的断点续传


🌳.通信管理模块的介绍

通信管理模块管理着服务器与浏览器之间的通信:

  • 当服务器收到浏览器的 get /listshow/ http/1.1请求时,服务器就会給浏览器响应一个文件展示页面,用户可以再这个页面上传文件或下载文件。
  • 当服务器收到浏览器的 个 get /download/文件名 http1.1请求时,服务器就会到backdir目录下或
  • packdir目录下查找该文件,如果找到了,就将该文件发送給浏览器。
  • 当服务器收到浏览器的 个post /upload/ http1.1请求时,则浏览器将文件上传到服务器上,文件名和文件数据存储在http响应中,服务器则将该文件放入backdir目录下。

🌳.通信模块的设计

将通信模块设计为单例模式,保证系统中只有一个通信模块对象.

服务器在运行的时候,先注册各个请求与对应函数的关心

   class Service{
      private:
        std::string ip;
        int port;
        httplib::Server server;
        static std::mutex lock;
        static Service* instant;
      private:
        Service(){
          ip=sjp::Config::GetInstant()->GetIp();
          port=sjp::Config::GetInstant()->GetServerPort();
        }
  
        Service(Service&)=delete;
        static std::string mtimeToString(time_t mtime){
          std::string res= ctime(&mtime);
          return res;
        }
      public:
        void Run(){
          server.Post("/upload",UpLoad); 
          server.Get("/download/(.*)",DownLoad);
          server.Get("/listshow",ListShow);                                                                                                                                                 
          server.Get("/",ListShow);
          server.listen("0.0.0.0",port);//启动服务器
        }

      static Service* GetInstant(){
          if(instant==nullptr){
            lock.lock();
            if(instant==nullptr){
              instant=new Service();
            }
            lock.unlock();
          }
          return instant;
        }
    };
  
    std::mutex Service::lock;
    Service* Service::instant=nullptr;
  }                                                              

🌳.ListShow函数

  listshow页面的前端源代码如下:

  •  我们需要使用stringstream对象将前端页面的源代码组织起来,然后将在将stringstream组织在http响应中,httplib服务器在底层会将http响应发送給浏览器。
  • 为了保证浏览器能够将前端代码翻译成一个页面,需要在httplib响应中的Content-type字段设置为text/html.
static void ListShow(const httplib::Request& req,httplib::Response& rep){    
          cout<<"listshow"<<endl;    
          std::vector<sjp::FileInfo> arry;    
          sjp::FileInfoManger::GetInstant()->GetAllInfo(arry);    
          std::stringstream ss;    
      
          //    
          ss<<"<!DOCTYPE html>"<<"<html><meta charset=\"UTF-8\"><head><title>Download</title></head><body><h1>Download</h1><table>";      
                        ss<<"<tr><td><h3>文件名</h3></a></td>";        
          ss<<"<td align=\"left\">"<<"<h3>最近修改时间</h3>"<<"</td>";                                                 
          ss<<"<td align=\"right\">"<<"<h3>文件大小</h3>"<<"</td></tr>";     
W>        for(int i=0;i<arry.size();i++){    
            //>    
            /*<td><a href="/download/test.txt">test.txt</a></td>    
              <td align="right">2021-12-29 10:10:10</td>    
              <td align="right">28k</td>    
             * */    
            std::string path=arry[i].back_path;    
            sjp::FileUtil fu(path);    
            std::string atimestr=mtimeToString(arry[i].modify_time);    
            ss<<"<tr><td><a href=\"/download/"<<fu.GetFilename()<<"\">"<<fu.GetFilename()<<"</a></td>";    
            ss<<"<td align=\"right\">"<<atimestr<<"    </td>";    
            ss<<"<td align=\"right\">"<<arry[i].file_size/1024<<"k"<<"</td></tr>";                                                                                                          
          }    
          ss<<"</table></body></html>";
          ss<<"<form action=\"http://119.23.41.13:8081/upload\" method=\"post\" enctype=\"multipart/form-data\"><div><input type=\"file\" name=\"file\"></div><div><input type=\"submit\" va  lue=\"上传\"></div></form>";
          rep.body=ss.str();
          rep.status=200;
          rep.set_header("Content-Type","text/html;");
        }
  

🌳.UpLoad函数

post /upload/ http1.1请求如下

上传文件的请求中请求正文包含文件的字段名和文件名,文件正文(文件包含的数据)。

实现:

  • 服务器收到上传请求后,需要先判断是否有文件字段名,如果有则提取该文件的信息。
  • 然后使用FileUtil文件工具打开一个文件存放在backdir下,并将文件正文写入到该文件中,并新插入一个文件信息到文件管理系统中_table表中,则完成服务器上传文件的请求。
  • 重新返回一个文件页面。
        static void UpLoad(const httplib::Request& req,httplib::Response& rep)
        {
          /* 1.判断是否有文件上传
           * 2.如果有,则获取该文件的名称和内容
           * 3.在backdir中创建一个文件
           * 4.将文件信息存储到FileInfoManger中
           * */
          //判断是否有文件上传
          std::cout<<"upload begin"<<endl;
            auto res=req.has_file("file");
            if(res==false){
              rep.status=400;                                                                                                                                                               
              return;
            } 
          const auto& file=req.get_file_value("file");
            std::string filepath=sjp::Config::GetInstant()->GetBackDir()+file.filename;
            std::string body=file.content;
  
            sjp::FileUtil fu(filepath);
            fu.SetContent(body);
  
            sjp::FileInfo fileinfo(filepath);
            sjp::FileInfoManger::GetInstant()->Insert(fileinfo);  
            sjp::FileInfoManger::GetInstant()->Storage();
            ListShow(req,rep);
        }
  

🌳.DownLoad函数

  • 当服务器收到一个下载请求时,服务器需要执行的步骤:
  • 文件信息管理模块通过请求url获取该文件信息,如果没有该文件信息,则说明该文件不存在,返回404.
  • 如果存在该文件,则通过文件信息中的packsign判断是否被压缩
  • 如果packsign=false,说明该文件没有没有被压缩,则服务器直接从backdir目录下将获取文件内容发送給浏览器。
  • 如果packsign=true,说明该文件已经被压缩,则服务器从packdir目录下将该文件进行解压缩到backdir,并删除packdir目录下的压缩包,将文件信息中的packsign修改为false,最后将文件内容发送給浏览器。
  • 如果浏览器想要以下载的形式接受文件,则服务器需要将http响应中的Content-Type字段设置为application/octet-stream

下载的断点续传

用户在下载文件的过程中,可能因为网络或者其它原因导致服务器下载失败了,如果没有断点续传,则用户重新下载文件则需要从文件的最开始的位置下载,如果实现了断点续传,则用户重新下载 文件则会从上次失败的位置开始下载

http下载的断点续传的思想

当浏览器第一次下载的时候,服务器給浏览器发送的http响应添加Accept-Ranges和Etag这两个字段:

  • Accept-Ranges一般被设置为bytes ,表示这个字段表示服务器支持断点续传,以字节为单位传输数据.
  • ETag表示的是服务器上某一版本资源的唯一标识,如果资源被改动过,则ETag会改变,客户端收到后则会保证这个信息。

如果下载中断了,浏览器重新下载文件,则第二次的http请求中需要包含If-Range字段和Range字段.

  • If-Range字段:"保存服务端的响应的ETag字段的信息",用于判断与服务端中与上一次请求资源是否一致,如果一致则断点续传,如果不一致,则重新开始下载。
  • Range字段:bytes start-end,表示的是请求服务器资源从start个字节开始到end个字节结束的数据。

则服务端发送http响应給客户端需要包含Content-Range字段和ETag字段:

  • Content-Range:start-end/文件大小,表示http响应包含文件数据从start开始到end结束的文件数据,文件大小表示文件总大小。
  • ETag:表示的是服务器上资源的唯一标识。

ETag字段的设定

ETag某一版本资源的唯一标识,如果资源被改动过,则ETag则会改变,所以我们可以通过"文件名-文件大小-文件修改时间"来设定ETag(当然这不是唯一的方式)。

UpDown函数的实现

如果浏览器发生断点续传请求,则在httplib内部实现了对Range字段的解析,开发人员只需要读取文件中的内容,设定好字段,底层的httplib库则会自动定位好具体的数据范围,发送給浏览器。


        
      //文件名-文件大小-最后一次修改时间
        static  std::string GetEtag(std::string filename){
           sjp::FileUtil fu(filename);
           std::string etag=fu.GetFilename();
           etag+="-";
           etag+=std::to_string(fu.GetFileSize());
           etag+="-";
           etag+=std::to_string(fu.GetFileModfityTime());
           return etag;
        }
  

     static void DownLoad(const httplib::Request& req,httplib::Response& rep)
        {
 
          cout<<"download begin"<<endl;
          std::string url=req.path;
          sjp::FileInfo fileinfo;
          sjp::FileInfoManger::GetInstant()->GetoneByURL(url,fileinfo);                                

         if(!sjp::FileInfoManger::GetInstant()->Exist(fileinfo.back_path)){
            /* sjp::FileUtil fu(back_dir); 
             std::vector<std::string> arry;
             fu.GetPathInDir(arry);
             */
            rep.status=404;
            return ;
          }
  
          if(fileinfo.pack_sign==true)
          {
            //解压缩
            sjp::FileUtil fu(fileinfo.pack_path);
            fu.UnpackFile(fileinfo.back_path);
            fileinfo.pack_sign=false;
            sjp::FileInfoManger::GetInstant()->Insert(fileinfo);
            fu.Remove();
          }
  
  
          bool flag=false;
          //判断是否为断点续传
          if(req.has_header("If-Range")){
            //断点续传
            std::string oldetag=req.get_header_value("If-Range"); 
            if(oldetag==GetEtag(fileinfo.back_path)){
               //需要断点续传
               cout<<"oldetag: "<<oldetag<<endl;
               cout<<"etag: "<<GetEtag(fileinfo.back_path)<<endl;                                                                                                                           
               flag=true;
            }
          }
         sjp::FileUtil fu(fileinfo.back_path);
          //获取文件内容
          std::string body;
          fu.GetContent(body);
          rep.body=body;
          rep.content_length_=body.size();
          if(flag==false){
          //不需要断点续传
           rep.status=200;
           rep.set_header("Accept-Ranges","bytes");
           rep.set_header("Content-Type","application/octet-stream");
           rep.set_header("Etag",GetEtag(fileinfo.back_path));
          }
          else{
            //断点续传
            rep.status=206;
            rep.set_header("Accept-Ranges","bytes");
            rep.set_header("Content-Type","application/octet-stream");
            rep.set_header("ETag",GetEtag(fileinfo.back_path));
          }
          return ;
        }

;