2017-10-09 22:11:01 +08:00
|
|
|
|
/*
|
2018-09-23 21:10:17 +08:00
|
|
|
|
* MIT License
|
|
|
|
|
*
|
|
|
|
|
* Copyright (c) 2016 xiongziliang <771730766@qq.com>
|
|
|
|
|
*
|
|
|
|
|
* 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
|
|
|
|
|
|
|
|
|
#include "HttpClient.h"
|
|
|
|
|
#include "Rtsp/Rtsp.h"
|
|
|
|
|
|
|
|
|
|
namespace ZL {
|
|
|
|
|
namespace Http {
|
|
|
|
|
|
|
|
|
|
|
2018-09-23 21:10:17 +08:00
|
|
|
|
HttpClient::HttpClient() {
|
2017-05-05 18:02:54 +08:00
|
|
|
|
}
|
2018-09-23 21:10:17 +08:00
|
|
|
|
|
|
|
|
|
HttpClient::~HttpClient() {
|
2017-05-05 18:02:54 +08:00
|
|
|
|
}
|
2018-09-23 21:10:17 +08:00
|
|
|
|
|
|
|
|
|
void HttpClient::sendRequest(const string &strUrl, float fTimeOutSec) {
|
2018-06-21 14:03:43 +08:00
|
|
|
|
_aliveTicker.resetTime();
|
2018-09-23 21:10:17 +08:00
|
|
|
|
auto protocol = FindField(strUrl.data(), NULL, "://");
|
2017-05-05 18:02:54 +08:00
|
|
|
|
uint16_t defaultPort;
|
|
|
|
|
bool isHttps;
|
|
|
|
|
if (strcasecmp(protocol.data(), "http") == 0) {
|
|
|
|
|
defaultPort = 80;
|
|
|
|
|
isHttps = false;
|
2018-09-23 21:10:17 +08:00
|
|
|
|
} else if (strcasecmp(protocol.data(), "https") == 0) {
|
2017-05-05 18:02:54 +08:00
|
|
|
|
defaultPort = 443;
|
|
|
|
|
isHttps = true;
|
2018-09-23 21:10:17 +08:00
|
|
|
|
} else {
|
2017-08-09 18:39:30 +08:00
|
|
|
|
auto strErr = StrPrinter << "非法的协议:" << protocol << endl;
|
2017-05-05 18:02:54 +08:00
|
|
|
|
throw std::invalid_argument(strErr);
|
|
|
|
|
}
|
2018-09-23 21:10:17 +08:00
|
|
|
|
|
2017-05-05 18:02:54 +08:00
|
|
|
|
auto host = FindField(strUrl.data(), "://", "/");
|
|
|
|
|
if (host.empty()) {
|
|
|
|
|
host = FindField(strUrl.data(), "://", NULL);
|
|
|
|
|
}
|
|
|
|
|
_path = FindField(strUrl.data(), host.data(), NULL);
|
|
|
|
|
if (_path.empty()) {
|
|
|
|
|
_path = "/";
|
|
|
|
|
}
|
|
|
|
|
auto port = atoi(FindField(host.data(), ":", NULL).data());
|
|
|
|
|
if (port <= 0) {
|
|
|
|
|
//默认端口
|
|
|
|
|
port = defaultPort;
|
|
|
|
|
} else {
|
|
|
|
|
//服务器域名
|
|
|
|
|
host = FindField(host.data(), NULL, ":");
|
|
|
|
|
}
|
2018-09-23 21:10:17 +08:00
|
|
|
|
_header.emplace(string("Host"), host);
|
|
|
|
|
_header.emplace(string("Tools"), "ZLMediaKit");
|
|
|
|
|
_header.emplace(string("Connection"), "keep-alive");
|
|
|
|
|
_header.emplace(string("Accept"), "*/*");
|
|
|
|
|
_header.emplace(string("Accept-Language"), "zh-CN,zh;q=0.8");
|
|
|
|
|
_header.emplace(string("User-Agent"),
|
|
|
|
|
"Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/57.0.2987.133 Safari/537.36");
|
|
|
|
|
|
|
|
|
|
if (_body && _body->remainSize()) {
|
|
|
|
|
_header.emplace(string("Content-Length"), to_string(_body->remainSize()));
|
|
|
|
|
_header.emplace(string("Content-Type"), "application/x-www-form-urlencoded; charset=UTF-8");
|
2017-05-05 18:02:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool bChanged = (_lastHost != host + ":" + to_string(port)) || (_isHttps != isHttps);
|
|
|
|
|
_lastHost = host + ":" + to_string(port);
|
|
|
|
|
_isHttps = isHttps;
|
2018-06-21 14:03:43 +08:00
|
|
|
|
_fTimeOutSec = fTimeOutSec;
|
2018-09-23 21:10:17 +08:00
|
|
|
|
if (!alive() || bChanged) {
|
2018-02-06 15:28:27 +08:00
|
|
|
|
//InfoL << "reconnet:" << _lastHost;
|
2018-09-23 21:10:17 +08:00
|
|
|
|
startConnect(host, port, fTimeOutSec);
|
|
|
|
|
} else {
|
2017-05-05 18:02:54 +08:00
|
|
|
|
SockException ex;
|
|
|
|
|
onConnect(ex);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2018-09-23 21:10:17 +08:00
|
|
|
|
void HttpClient::onConnect(const SockException &ex) {
|
2018-06-21 14:03:43 +08:00
|
|
|
|
_aliveTicker.resetTime();
|
2018-09-23 21:10:17 +08:00
|
|
|
|
if (ex) {
|
|
|
|
|
onDisconnect(ex);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_totalBodySize = 0;
|
|
|
|
|
_recvedBodySize = 0;
|
|
|
|
|
HttpRequestSplitter::reset();
|
|
|
|
|
|
2018-06-21 14:03:43 +08:00
|
|
|
|
_StrPrinter printer;
|
|
|
|
|
printer << _method + " " << _path + " HTTP/1.1\r\n";
|
2017-05-05 18:02:54 +08:00
|
|
|
|
for (auto &pr : _header) {
|
2018-09-23 21:10:17 +08:00
|
|
|
|
printer << pr.first + ": ";
|
2018-06-21 14:03:43 +08:00
|
|
|
|
printer << pr.second + "\r\n";
|
2017-05-05 18:02:54 +08:00
|
|
|
|
}
|
2018-06-21 14:03:43 +08:00
|
|
|
|
send(printer << "\r\n");
|
|
|
|
|
onSend();
|
2017-05-05 18:02:54 +08:00
|
|
|
|
}
|
2018-09-23 21:10:17 +08:00
|
|
|
|
|
|
|
|
|
void HttpClient::onRecv(const Buffer::Ptr &pBuf) {
|
|
|
|
|
onRecvBytes(pBuf->data(), pBuf->size());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void HttpClient::onRecvBytes(const char *data, int size) {
|
|
|
|
|
_aliveTicker.resetTime();
|
|
|
|
|
HttpRequestSplitter::input(data, size);
|
2017-05-05 18:02:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
2018-09-23 21:10:17 +08:00
|
|
|
|
void HttpClient::onErr(const SockException &ex) {
|
|
|
|
|
if (ex.getErrCode() == Err_eof && _totalBodySize == INT64_MAX) {
|
2018-06-21 14:03:43 +08:00
|
|
|
|
//如果Content-Length未指定 但服务器断开链接
|
|
|
|
|
//则认为本次http请求完成
|
2018-09-23 21:10:17 +08:00
|
|
|
|
onResponseCompleted_l();
|
2018-06-21 14:03:43 +08:00
|
|
|
|
}
|
2018-09-23 21:10:17 +08:00
|
|
|
|
onDisconnect(ex);
|
2017-05-05 18:02:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
2018-09-23 21:10:17 +08:00
|
|
|
|
int64_t HttpClient::onRecvHeader(const char *data, uint64_t len) {
|
|
|
|
|
_parser.Parse(data);
|
|
|
|
|
onResponseHeader(_parser.Url(), _parser.getValues());
|
|
|
|
|
|
2018-09-23 21:19:24 +08:00
|
|
|
|
if(_parser["Content-Length"].empty()){
|
|
|
|
|
//没有Content-Length字段
|
|
|
|
|
if(!_parser.Content().empty()){
|
|
|
|
|
//如果http回复未声明Content-Length字段,但是却有content内容,那说明可能是个不限长度的content
|
|
|
|
|
_totalBodySize = INT64_MAX;
|
|
|
|
|
_recvedBodySize = 0;
|
|
|
|
|
//返回-1代表不限制content回复大小
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
//content长度为0,本次http请求结束
|
|
|
|
|
onResponseCompleted_l();
|
|
|
|
|
return 0;
|
2018-09-23 21:10:17 +08:00
|
|
|
|
}
|
2018-09-23 21:19:24 +08:00
|
|
|
|
|
2018-09-23 21:10:17 +08:00
|
|
|
|
_recvedBodySize = 0;
|
2018-09-23 21:19:24 +08:00
|
|
|
|
_totalBodySize = atoll(_parser["Content-Length"].data());
|
|
|
|
|
|
|
|
|
|
if(_totalBodySize == 0){
|
|
|
|
|
//content长度为0,本次http请求结束
|
|
|
|
|
onResponseCompleted_l();
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2018-09-23 21:10:17 +08:00
|
|
|
|
|
|
|
|
|
//虽然我们知道content的确切大小,
|
|
|
|
|
//但是由于我们没必要等content接收完毕才回调onRecvContent(因为这样浪费内存并且要多次拷贝数据)
|
|
|
|
|
//所以返回-1代表我们接下来分段接收content
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void HttpClient::onRecvContent(const char *data, uint64_t len) {
|
|
|
|
|
auto recvedBodySize = _recvedBodySize + len;
|
|
|
|
|
if (recvedBodySize < _totalBodySize) {
|
|
|
|
|
onResponseBody(data, len, recvedBodySize, _totalBodySize);
|
|
|
|
|
_recvedBodySize = recvedBodySize;
|
|
|
|
|
} else {
|
|
|
|
|
onResponseBody(data, _totalBodySize - _recvedBodySize, _totalBodySize, _totalBodySize);
|
|
|
|
|
bool biggerThanExpected = recvedBodySize > _totalBodySize;
|
|
|
|
|
onResponseCompleted_l();
|
|
|
|
|
if(biggerThanExpected) {
|
|
|
|
|
//声明的content数据比真实的小,那么我们只截取前面部分的并断开链接
|
|
|
|
|
shutdown();
|
|
|
|
|
onDisconnect(SockException(Err_other, "http response content size bigger than expected"));
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-05-05 18:02:54 +08:00
|
|
|
|
}
|
2018-06-21 14:03:43 +08:00
|
|
|
|
|
|
|
|
|
void HttpClient::onSend() {
|
|
|
|
|
_aliveTicker.resetTime();
|
2018-09-23 21:10:17 +08:00
|
|
|
|
while (_body && _body->remainSize() && !isSocketBusy()) {
|
2018-06-21 14:03:43 +08:00
|
|
|
|
auto buffer = _body->readData();
|
2018-09-23 21:10:17 +08:00
|
|
|
|
if (!buffer) {
|
2018-06-21 14:03:43 +08:00
|
|
|
|
//数据发送结束或读取数据异常
|
|
|
|
|
break;
|
|
|
|
|
}
|
2018-09-23 21:10:17 +08:00
|
|
|
|
if (send(buffer) <= 0) {
|
2018-06-21 14:03:43 +08:00
|
|
|
|
//发送数据失败,不需要回滚数据,因为发送前已经通过isSocketBusy()判断socket可写
|
|
|
|
|
//所以发送缓存区肯定未满,该buffer肯定已经写入socket
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void HttpClient::onManager() {
|
2018-09-23 21:10:17 +08:00
|
|
|
|
if (_aliveTicker.elapsedTime() > 3 * 1000 && _totalBodySize == INT64_MAX) {
|
2018-06-21 14:03:43 +08:00
|
|
|
|
//如果Content-Length未指定 但接收数据超时
|
|
|
|
|
//则认为本次http请求完成
|
2018-09-23 21:10:17 +08:00
|
|
|
|
onResponseCompleted_l();
|
2018-06-21 14:03:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
2018-09-23 21:10:17 +08:00
|
|
|
|
if (_fTimeOutSec > 0 && _aliveTicker.elapsedTime() > _fTimeOutSec * 1000) {
|
2018-06-21 14:03:43 +08:00
|
|
|
|
//超时
|
2018-09-23 21:10:17 +08:00
|
|
|
|
onDisconnect(SockException(Err_timeout, "http request timeout"));
|
2018-06-21 14:03:43 +08:00
|
|
|
|
shutdown();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-23 21:10:17 +08:00
|
|
|
|
void HttpClient::onResponseCompleted_l() {
|
|
|
|
|
_totalBodySize = 0;
|
|
|
|
|
_recvedBodySize = 0;
|
|
|
|
|
HttpRequestSplitter::reset();
|
|
|
|
|
onResponseCompleted();
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-21 14:03:43 +08:00
|
|
|
|
|
2017-05-05 18:02:54 +08:00
|
|
|
|
} /* namespace Http */
|
|
|
|
|
} /* namespace ZL */
|
|
|
|
|
|