322 lines
8.4 KiB
C++
322 lines
8.4 KiB
C++
/*
|
||
* Copyright (c) 2016-present The ZLMediaKit project authors. All Rights Reserved.
|
||
*
|
||
* This file is part of ZLMediaKit(https://github.com/ZLMediaKit/ZLMediaKit).
|
||
*
|
||
* Use of this source code is governed by MIT-like license that can be found in the
|
||
* LICENSE file in the root of the source tree. All contributing project authors
|
||
* may be found in the AUTHORS file in the root of the source tree.
|
||
*/
|
||
|
||
#ifndef Http_HttpClient_h
|
||
#define Http_HttpClient_h
|
||
|
||
#include <stdio.h>
|
||
#include <string.h>
|
||
#include <functional>
|
||
#include <memory>
|
||
#include "Util/util.h"
|
||
#include "Util/mini.h"
|
||
#include "Network/TcpClient.h"
|
||
#include "Common/Parser.h"
|
||
#include "HttpRequestSplitter.h"
|
||
#include "HttpCookie.h"
|
||
#include "HttpChunkedSplitter.h"
|
||
#include "Common/strCoding.h"
|
||
#include "HttpBody.h"
|
||
|
||
namespace mediakit {
|
||
|
||
class HttpArgs : public std::map<std::string, toolkit::variant, StrCaseCompare> {
|
||
public:
|
||
std::string make() const {
|
||
std::string ret;
|
||
for (auto &pr : *this) {
|
||
ret.append(pr.first);
|
||
ret.append("=");
|
||
ret.append(strCoding::UrlEncodeComponent(pr.second));
|
||
ret.append("&");
|
||
}
|
||
if (ret.size()) {
|
||
ret.pop_back();
|
||
}
|
||
return ret;
|
||
}
|
||
};
|
||
|
||
class HttpClient : public toolkit::TcpClient, public HttpRequestSplitter {
|
||
public:
|
||
using HttpHeader = StrCaseMap;
|
||
using Ptr = std::shared_ptr<HttpClient>;
|
||
|
||
/**
|
||
* 发送http[s]请求
|
||
* @param url 请求url
|
||
* Send http[s] request
|
||
* @param url Request url
|
||
|
||
* [AUTO-TRANSLATED:01b6c9ac]
|
||
*/
|
||
virtual void sendRequest(const std::string &url);
|
||
|
||
/**
|
||
* 重置对象
|
||
* Reset object
|
||
|
||
* [AUTO-TRANSLATED:d23b5bbb]
|
||
*/
|
||
virtual void clear();
|
||
|
||
/**
|
||
* 设置http方法
|
||
* @param method GET/POST等
|
||
* Set http method
|
||
* @param method GET/POST etc.
|
||
|
||
* [AUTO-TRANSLATED:5199546a]
|
||
*/
|
||
void setMethod(std::string method);
|
||
|
||
/**
|
||
* 覆盖http头
|
||
* @param header
|
||
* Override http header
|
||
* @param header
|
||
|
||
* [AUTO-TRANSLATED:ea31a471]
|
||
*/
|
||
void setHeader(HttpHeader header);
|
||
|
||
HttpClient &addHeader(std::string key, std::string val, bool force = false);
|
||
|
||
/**
|
||
* 设置http content
|
||
* @param body http content
|
||
* Set http content
|
||
* @param body http content
|
||
|
||
* [AUTO-TRANSLATED:9993580c]
|
||
*/
|
||
void setBody(std::string body);
|
||
|
||
/**
|
||
* 设置http content
|
||
* @param body http content
|
||
* Set http content
|
||
* @param body http content
|
||
|
||
* [AUTO-TRANSLATED:9993580c]
|
||
*/
|
||
void setBody(HttpBody::Ptr body);
|
||
|
||
/**
|
||
* 获取回复,在收到完整回复后有效
|
||
* Get response, valid after receiving the complete response
|
||
|
||
* [AUTO-TRANSLATED:b107995e]
|
||
*/
|
||
const Parser &response() const;
|
||
|
||
/**
|
||
* 获取回复header声明的body大小
|
||
* Get the body size declared in the response header
|
||
|
||
* [AUTO-TRANSLATED:65f8e782]
|
||
*/
|
||
ssize_t responseBodyTotalSize() const;
|
||
|
||
/**
|
||
* 获取已经下载body的大小
|
||
* Get the size of the body that has been downloaded
|
||
|
||
* [AUTO-TRANSLATED:a3cde7b4]
|
||
*/
|
||
size_t responseBodySize() const;
|
||
|
||
/**
|
||
* 获取请求url
|
||
* Get the request url
|
||
|
||
* [AUTO-TRANSLATED:cc7fe537]
|
||
*/
|
||
const std::string &getUrl() const;
|
||
|
||
/**
|
||
* 判断是否正在等待响应
|
||
* Determine if the response is pending
|
||
|
||
* [AUTO-TRANSLATED:058719d7]
|
||
*/
|
||
bool waitResponse() const;
|
||
|
||
/**
|
||
* 判断是否为https
|
||
* Determine if it is https
|
||
|
||
* [AUTO-TRANSLATED:9b3a0254]
|
||
*/
|
||
bool isHttps() const;
|
||
|
||
/**
|
||
* 设置从发起连接到接收header完毕的延时,默认10秒
|
||
* 此参数必须大于0
|
||
* Set the delay from initiating the connection to receiving the header, default 10 seconds
|
||
* This parameter must be greater than 0
|
||
|
||
* [AUTO-TRANSLATED:4cce3e85]
|
||
*/
|
||
void setHeaderTimeout(size_t timeout_ms);
|
||
|
||
/**
|
||
* 设置接收body数据超时时间, 默认5秒
|
||
* 此参数可以用于处理超大body回复的超时问题
|
||
* 此参数可以等于0
|
||
* Set the timeout for receiving body data, default 5 seconds
|
||
* This parameter can be used to handle timeout issues for large body responses
|
||
* This parameter can be equal to 0
|
||
|
||
* [AUTO-TRANSLATED:48585852]
|
||
*/
|
||
void setBodyTimeout(size_t timeout_ms);
|
||
|
||
/**
|
||
* 设置整个链接超时超时时间, 默认0
|
||
* 该值设置不为0后,HeaderTimeout和BodyTimeout无效
|
||
* Set the timeout for the entire link, default 0
|
||
* After this value is set to non-zero, HeaderTimeout and BodyTimeout are invalid
|
||
|
||
* [AUTO-TRANSLATED:df094868]
|
||
*/
|
||
void setCompleteTimeout(size_t timeout_ms);
|
||
|
||
/**
|
||
* 设置http代理url
|
||
* Set http proxy url
|
||
|
||
* [AUTO-TRANSLATED:95df17e7]
|
||
*/
|
||
void setProxyUrl(std::string proxy_url);
|
||
|
||
/**
|
||
* 当重用连接失败时, 是否允许重新发起请求
|
||
* If the reuse connection fails, whether to allow the request to be resent
|
||
* @param allow true:允许重新发起请求 / true: allow the request to be resent
|
||
* When the reuse connection fails, whether to allow the request to be resent
|
||
* @param allow true: allow the request to be resent
|
||
|
||
* [AUTO-TRANSLATED:71bd8e67]
|
||
*/
|
||
void setAllowResendRequest(bool allow);
|
||
|
||
protected:
|
||
/**
|
||
* 收到http回复头
|
||
* @param status 状态码,譬如:200 OK
|
||
* @param headers http头
|
||
* Receive http response header
|
||
* @param status Status code, such as: 200 OK
|
||
* @param headers http header
|
||
|
||
* [AUTO-TRANSLATED:a685f8ef]
|
||
*/
|
||
virtual void onResponseHeader(const std::string &status, const HttpHeader &headers) = 0;
|
||
|
||
/**
|
||
* 收到http conten数据
|
||
* @param buf 数据指针
|
||
* @param size 数据大小
|
||
* Receive http content data
|
||
* @param buf Data pointer
|
||
* @param size Data size
|
||
|
||
* [AUTO-TRANSLATED:bee3bf62]
|
||
*/
|
||
virtual void onResponseBody(const char *buf, size_t size) = 0;
|
||
|
||
/**
|
||
* 接收http回复完毕,
|
||
* Receive http response complete,
|
||
|
||
* [AUTO-TRANSLATED:b96ed715]
|
||
*/
|
||
virtual void onResponseCompleted(const toolkit::SockException &ex) = 0;
|
||
|
||
/**
|
||
* 重定向事件
|
||
* @param url 重定向url
|
||
* @param temporary 是否为临时重定向
|
||
* @return 是否继续
|
||
* Redirect event
|
||
* @param url Redirect url
|
||
* @param temporary Whether it is a temporary redirect
|
||
* @return Whether to continue
|
||
* [AUTO-TRANSLATED:b64d5f8b]
|
||
*/
|
||
virtual bool onRedirectUrl(const std::string &url, bool temporary) { return true; };
|
||
|
||
protected:
|
||
//// HttpRequestSplitter override ////
|
||
ssize_t onRecvHeader(const char *data, size_t len) override;
|
||
void onRecvContent(const char *data, size_t len) override;
|
||
|
||
//// TcpClient override ////
|
||
void onConnect(const toolkit::SockException &ex) override;
|
||
void onRecv(const toolkit::Buffer::Ptr &pBuf) override;
|
||
void onError(const toolkit::SockException &ex) override;
|
||
void onFlush() override;
|
||
void onManager() override;
|
||
|
||
void clearResponse();
|
||
|
||
bool checkProxyConnected(const char *data, size_t len);
|
||
bool isUsedProxy() const;
|
||
bool isProxyConnected() const;
|
||
|
||
private:
|
||
void onResponseCompleted_l(const toolkit::SockException &ex);
|
||
void onConnect_l(const toolkit::SockException &ex);
|
||
void checkCookie(HttpHeader &headers);
|
||
|
||
private:
|
||
//for http response
|
||
bool _complete = false;
|
||
bool _header_recved = false;
|
||
bool _http_persistent = true;
|
||
bool _allow_resend_request = false;
|
||
size_t _recved_body_size;
|
||
ssize_t _total_body_size;
|
||
Parser _parser;
|
||
std::shared_ptr<HttpChunkedSplitter> _chunked_splitter;
|
||
|
||
//for request args
|
||
bool _is_https;
|
||
std::string _url;
|
||
HttpHeader _user_set_header;
|
||
HttpBody::Ptr _body;
|
||
std::string _method;
|
||
std::string _last_host;
|
||
|
||
//for this request
|
||
std::string _path;
|
||
HttpHeader _header;
|
||
|
||
//for timeout
|
||
size_t _wait_header_ms = 10 * 1000;
|
||
size_t _wait_body_ms = 10 * 1000;
|
||
size_t _wait_complete_ms = 0;
|
||
toolkit::Ticker _wait_header;
|
||
toolkit::Ticker _wait_body;
|
||
toolkit::Ticker _wait_complete;
|
||
|
||
bool _used_proxy = false;
|
||
bool _proxy_connected = false;
|
||
uint16_t _proxy_port;
|
||
std::string _proxy_url;
|
||
std::string _proxy_host;
|
||
std::string _proxy_auth;
|
||
};
|
||
|
||
} /* namespace mediakit */
|
||
|
||
#endif /* Http_HttpClient_h */
|