ZLMediaKit/src/Http/HttpClient.h

340 lines
9.9 KiB
C++
Raw Normal View History

2017-10-09 22:11:01 +08:00
/*
2017-09-27 16:20:30 +08:00
* MIT License
*
2019-05-08 15:40:07 +08:00
* Copyright (c) 2016-2019 xiongziliang <771730766@qq.com>
2017-09-27 16:20:30 +08:00
*
* This file is part of ZLMediaKit(https://github.com/xiongziliang/ZLMediaKit).
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
2017-05-05 18:02:54 +08:00
#ifndef Http_HttpClient_h
#define Http_HttpClient_h
#include <stdio.h>
2017-05-05 18:02:54 +08:00
#include <string.h>
#include <functional>
#include <memory>
#include "Util/util.h"
2019-06-28 16:48:02 +08:00
#include "Util/mini.h"
2017-05-05 18:02:54 +08:00
#include "Network/TcpClient.h"
2019-06-28 16:48:02 +08:00
#include "Common/Parser.h"
2018-09-23 21:10:17 +08:00
#include "HttpRequestSplitter.h"
#include "HttpCookie.h"
#include "HttpChunkedSplitter.h"
2019-05-20 16:26:04 +08:00
#include "strCoding.h"
2017-05-05 18:02:54 +08:00
using namespace std;
2018-10-24 17:17:55 +08:00
using namespace toolkit;
2017-05-05 18:02:54 +08:00
2018-10-24 17:17:55 +08:00
namespace mediakit {
2017-05-05 18:02:54 +08:00
2019-05-20 16:26:04 +08:00
class HttpArgs : public map<string, variant, StrCaseCompare> {
2017-05-05 18:02:54 +08:00
public:
HttpArgs(){}
virtual ~HttpArgs(){}
string make() const {
string ret;
for(auto &pr : *this){
ret.append(pr.first);
ret.append("=");
2019-05-27 12:13:27 +08:00
ret.append(strCoding::UrlEncode(pr.second));
2017-05-05 18:02:54 +08:00
ret.append("&");
}
if(ret.size()){
ret.pop_back();
}
return ret;
}
};
class HttpBody{
public:
typedef std::shared_ptr<HttpBody> Ptr;
HttpBody(){}
virtual ~HttpBody(){}
//剩余数据大小
virtual uint64_t remainSize() = 0;
virtual Buffer::Ptr readData() = 0;
};
class HttpStringBody : public HttpBody{
public:
typedef std::shared_ptr<HttpStringBody> Ptr;
HttpStringBody(const string &str){
_str = str;
}
virtual ~HttpStringBody(){}
uint64_t remainSize() override {
return _str.size();
}
Buffer::Ptr readData() override {
auto ret = std::make_shared<BufferString>(_str);
_str.clear();
return ret;
}
private:
mutable string _str;
};
class HttpMultiFormBody : public HttpBody {
public:
typedef std::shared_ptr<HttpMultiFormBody> Ptr;
2019-05-20 16:26:04 +08:00
template<typename MapType>
HttpMultiFormBody(const MapType &args,const string &filePath,const string &boundary,uint32_t sliceSize = 4 * 1024){
_fp = fopen(filePath.data(),"rb");
if(!_fp){
throw std::invalid_argument(StrPrinter << "打开文件失败:" << filePath << " " << get_uv_errmsg());
}
auto fileName = filePath;
auto pos = filePath.rfind('/');
if(pos != string::npos){
fileName = filePath.substr(pos + 1);
}
_bodyPrefix = multiFormBodyPrefix(args,boundary,fileName);
_bodySuffix = multiFormBodySuffix(boundary);
_totalSize = _bodyPrefix.size() + _bodySuffix.size() + fileSize(_fp);
_sliceSize = sliceSize;
}
virtual ~HttpMultiFormBody(){
fclose(_fp);
}
uint64_t remainSize() override {
return _totalSize - _offset;
}
Buffer::Ptr readData() override{
if(_bodyPrefix.size()){
auto ret = std::make_shared<BufferString>(_bodyPrefix);
_offset += _bodyPrefix.size();
_bodyPrefix.clear();
return ret;
}
if(0 == feof(_fp)){
auto ret = std::make_shared<BufferRaw>(_sliceSize);
//读文件
int size;
do{
size = fread(ret->data(),1,_sliceSize,_fp);
}while(-1 == size && UV_EINTR == get_uv_error(false));
if(size == -1){
_offset = _totalSize;
WarnL << "fread failed:" << get_uv_errmsg();
return nullptr;
}
_offset += size;
ret->setSize(size);
return ret;
}
if(_bodySuffix.size()){
auto ret = std::make_shared<BufferString>(_bodySuffix);
_offset = _totalSize;
_bodySuffix.clear();
return ret;
}
return nullptr;
}
public:
2019-05-20 16:26:04 +08:00
template<typename MapType>
static string multiFormBodyPrefix(const MapType &args,const string &boundary,const string &fileName){
string MPboundary = string("--") + boundary;
_StrPrinter body;
for(auto &pr : args){
body << MPboundary << "\r\n";
body << "Content-Disposition: form-data; name=\"" << pr.first << "\"\r\n\r\n";
body << pr.second << "\r\n";
}
body << MPboundary << "\r\n";
body << "Content-Disposition: form-data; name=\"" << "file" << "\";filename=\"" << fileName << "\"\r\n";
body << "Content-Type: application/octet-stream\r\n\r\n" ;
return body;
}
static string multiFormBodySuffix(const string &boundary){
string MPboundary = string("--") + boundary;
string endMPboundary = MPboundary + "--";
_StrPrinter body;
body << "\r\n" << endMPboundary;
return body;
}
static uint64_t fileSize(FILE *fp) {
auto current = ftell(fp);
fseek(fp,0L,SEEK_END); /* 定位到文件末尾 */
auto end = ftell(fp); /* 得到文件大小 */
fseek(fp,current,SEEK_SET);
return end - current;
}
static string multiFormContentType(const string &boundary){
return StrPrinter << "multipart/form-data; boundary=" << boundary;
}
private:
FILE *_fp;
string _bodyPrefix;
string _bodySuffix;
uint64_t _offset = 0;
uint64_t _totalSize;
uint32_t _sliceSize;
};
2018-09-23 21:10:17 +08:00
class HttpClient : public TcpClient , public HttpRequestSplitter
2017-05-05 18:02:54 +08:00
{
public:
typedef StrCaseMap HttpHeader;
typedef std::shared_ptr<HttpClient> Ptr;
HttpClient();
virtual ~HttpClient();
2018-02-28 17:30:12 +08:00
virtual void sendRequest(const string &url,float fTimeOutSec);
2019-01-03 15:05:52 +08:00
virtual void clear(){
2017-05-05 18:02:54 +08:00
_header.clear();
_body.reset();
2017-05-05 18:02:54 +08:00
_method.clear();
_path.clear();
_parser.Clear();
2019-01-03 15:05:52 +08:00
_recvedBodySize = 0;
_totalBodySize = 0;
_aliveTicker.resetTime();
_chunkedSplitter.reset();
HttpRequestSplitter::reset();
2017-05-05 18:02:54 +08:00
}
2019-01-03 15:05:52 +08:00
2017-05-05 18:02:54 +08:00
void setMethod(const string &method){
_method = method;
}
void setHeader(const HttpHeader &header){
2017-06-09 16:05:33 +08:00
_header = header;
2017-05-05 18:02:54 +08:00
}
HttpClient & addHeader(const string &key,const string &val,bool force = false){
if(!force){
_header.emplace(key,val);
}else{
_header[key] = val;
}
return *this;
2017-05-05 18:02:54 +08:00
}
void setBody(const string &body){
_body.reset(new HttpStringBody(body));
}
void setBody(const HttpBody::Ptr &body){
2017-05-05 18:02:54 +08:00
_body = body;
}
2018-11-13 16:34:45 +08:00
const string &responseStatus() const{
2017-05-05 18:02:54 +08:00
return _parser.Url();
}
2018-11-13 16:34:45 +08:00
const HttpHeader &responseHeader() const{
2017-05-05 18:02:54 +08:00
return _parser.getValues();
}
2018-11-13 16:34:45 +08:00
const Parser& response() const{
return _parser;
}
2019-07-01 18:35:26 +08:00
const string &getUrl() const{
return _url;
}
2017-05-05 18:02:54 +08:00
protected:
2018-09-23 21:10:17 +08:00
/**
* http回复头
* @param status :200 OK
* @param headers http头
* @return content的长度-1:content>=0:content
* http头中带有Content-Length字段时
2018-09-23 21:10:17 +08:00
*/
virtual int64_t onResponseHeader(const string &status,const HttpHeader &headers){
2017-05-05 18:02:54 +08:00
DebugL << status;
//无Content-Length字段时默认后面全是content
return -1;
2017-05-05 18:02:54 +08:00
};
2018-09-23 21:10:17 +08:00
/**
* http conten数据
* @param buf
* @param size
* @param recvedSize (),totalSize时将触发onResponseCompleted回调
* @param totalSize
*/
2019-01-16 14:26:06 +08:00
virtual void onResponseBody(const char *buf,int64_t size,int64_t recvedSize,int64_t totalSize){
2017-05-05 18:02:54 +08:00
DebugL << size << " " << recvedSize << " " << totalSize;
};
2018-09-23 21:10:17 +08:00
/**
2018-11-13 17:59:12 +08:00
* http回复完毕,
2018-09-23 21:10:17 +08:00
*/
virtual void onResponseCompleted(){
2017-05-05 18:02:54 +08:00
DebugL;
}
2018-09-23 21:10:17 +08:00
/**
* http链接断开回调
* @param ex
*/
2017-05-05 18:02:54 +08:00
virtual void onDisconnect(const SockException &ex){}
2018-09-23 21:10:17 +08:00
2019-07-01 20:55:31 +08:00
/**
*
* @param url url
* @param temporary
* @return
*/
virtual bool onRedirectUrl(const string &url,bool temporary){ return true;};
2018-09-23 21:10:17 +08:00
//HttpRequestSplitter override
int64_t onRecvHeader(const char *data,uint64_t len) override ;
void onRecvContent(const char *data,uint64_t len) override;
protected:
2017-05-05 18:02:54 +08:00
virtual void onConnect(const SockException &ex) override;
2018-02-23 15:36:51 +08:00
virtual void onRecv(const Buffer::Ptr &pBuf) override;
2017-05-05 18:02:54 +08:00
virtual void onErr(const SockException &ex) override;
2019-05-29 18:08:50 +08:00
virtual void onFlush() override;
virtual void onManager() override;
2018-09-23 21:10:17 +08:00
private:
void onResponseCompleted_l();
2019-03-14 09:59:07 +08:00
void checkCookie(HttpHeader &headers );
protected:
bool _isHttps;
private:
2019-07-01 18:35:26 +08:00
string _url;
2017-05-05 18:02:54 +08:00
HttpHeader _header;
HttpBody::Ptr _body;
2017-05-05 18:02:54 +08:00
string _method;
string _path;
//recv
int64_t _recvedBodySize;
int64_t _totalBodySize;
2017-05-05 18:02:54 +08:00
Parser _parser;
string _lastHost;
Ticker _aliveTicker;
float _fTimeOutSec = 0;
std::shared_ptr<HttpChunkedSplitter> _chunkedSplitter;
2017-05-05 18:02:54 +08:00
};
2018-10-24 17:17:55 +08:00
} /* namespace mediakit */
2017-05-05 18:02:54 +08:00
#endif /* Http_HttpClient_h */