2019-03-01 06:54:03 +01:00
|
|
|
/*
|
|
|
|
* IXHttpClient.cpp
|
|
|
|
* Author: Benjamin Sergeant
|
|
|
|
* Copyright (c) 2019 Machine Zone, Inc. All rights reserved.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "IXHttpClient.h"
|
2019-09-23 19:25:23 +02:00
|
|
|
|
2020-09-28 19:19:27 +02:00
|
|
|
#include "IXGzipCodec.h"
|
2019-09-23 19:25:23 +02:00
|
|
|
#include "IXSocketFactory.h"
|
2019-03-01 06:54:03 +01:00
|
|
|
#include "IXUrlParser.h"
|
2019-08-30 21:48:18 +02:00
|
|
|
#include "IXUserAgent.h"
|
2019-03-01 06:54:03 +01:00
|
|
|
#include "IXWebSocketHttpHeaders.h"
|
2019-09-23 19:25:23 +02:00
|
|
|
#include <assert.h>
|
|
|
|
#include <cstring>
|
2019-03-01 06:54:03 +01:00
|
|
|
#include <iomanip>
|
2019-11-26 06:08:43 +01:00
|
|
|
#include <random>
|
2019-09-23 19:25:23 +02:00
|
|
|
#include <sstream>
|
2019-03-01 06:54:03 +01:00
|
|
|
#include <vector>
|
2020-08-01 07:54:57 +02:00
|
|
|
|
2019-03-01 06:54:03 +01:00
|
|
|
namespace ix
|
|
|
|
{
|
|
|
|
const std::string HttpClient::kPost = "POST";
|
|
|
|
const std::string HttpClient::kGet = "GET";
|
|
|
|
const std::string HttpClient::kHead = "HEAD";
|
2019-06-03 20:38:56 +02:00
|
|
|
const std::string HttpClient::kDel = "DEL";
|
|
|
|
const std::string HttpClient::kPut = "PUT";
|
2020-05-05 16:38:55 +02:00
|
|
|
const std::string HttpClient::kPatch = "PATCH";
|
2019-03-01 06:54:03 +01:00
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
HttpClient::HttpClient(bool async)
|
|
|
|
: _async(async)
|
|
|
|
, _stop(false)
|
2020-05-05 16:43:55 +02:00
|
|
|
, _forceBody(false)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
2019-06-06 02:04:24 +02:00
|
|
|
if (!_async) return;
|
2019-03-01 06:54:03 +01:00
|
|
|
|
2019-06-06 02:04:24 +02:00
|
|
|
_thread = std::thread(&HttpClient::run, this);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
HttpClient::~HttpClient()
|
|
|
|
{
|
2019-06-06 02:04:24 +02:00
|
|
|
if (!_thread.joinable()) return;
|
2019-03-01 06:54:03 +01:00
|
|
|
|
2019-06-06 02:04:24 +02:00
|
|
|
_stop = true;
|
|
|
|
_condition.notify_one();
|
|
|
|
_thread.join();
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
2019-09-30 06:13:11 +02:00
|
|
|
void HttpClient::setTLSOptions(const SocketTLSOptions& tlsOptions)
|
|
|
|
{
|
|
|
|
_tlsOptions = tlsOptions;
|
|
|
|
}
|
|
|
|
|
2020-05-05 16:43:55 +02:00
|
|
|
void HttpClient::setForceBody(bool value)
|
|
|
|
{
|
2020-05-05 16:38:55 +02:00
|
|
|
_forceBody = value;
|
|
|
|
}
|
2020-05-05 16:43:55 +02:00
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
HttpRequestArgsPtr HttpClient::createRequest(const std::string& url, const std::string& verb)
|
2019-06-06 02:04:24 +02:00
|
|
|
{
|
|
|
|
auto request = std::make_shared<HttpRequestArgs>();
|
|
|
|
request->url = url;
|
|
|
|
request->verb = verb;
|
|
|
|
return request;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool HttpClient::performRequest(HttpRequestArgsPtr args,
|
|
|
|
const OnResponseCallback& onResponseCallback)
|
|
|
|
{
|
2019-08-15 04:53:57 +02:00
|
|
|
assert(_async && "HttpClient needs its async parameter set to true "
|
|
|
|
"in order to call performRequest");
|
2019-06-06 02:04:24 +02:00
|
|
|
if (!_async) return false;
|
|
|
|
|
|
|
|
// Enqueue the task
|
|
|
|
{
|
|
|
|
// acquire lock
|
|
|
|
std::unique_lock<std::mutex> lock(_queueMutex);
|
|
|
|
|
|
|
|
// add the task
|
|
|
|
_queue.push(std::make_pair(args, onResponseCallback));
|
|
|
|
} // release lock
|
|
|
|
|
|
|
|
// wake up one thread
|
|
|
|
_condition.notify_one();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void HttpClient::run()
|
|
|
|
{
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
HttpRequestArgsPtr args;
|
|
|
|
OnResponseCallback onResponseCallback;
|
|
|
|
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> lock(_queueMutex);
|
|
|
|
|
|
|
|
while (!_stop && _queue.empty())
|
|
|
|
{
|
|
|
|
_condition.wait(lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_stop) return;
|
|
|
|
|
|
|
|
auto p = _queue.front();
|
|
|
|
_queue.pop();
|
|
|
|
|
|
|
|
args = p.first;
|
|
|
|
onResponseCallback = p.second;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_stop) return;
|
|
|
|
|
|
|
|
HttpResponsePtr response = request(args->url, args->verb, args->body, args);
|
|
|
|
onResponseCallback(response);
|
|
|
|
|
|
|
|
if (_stop) return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
HttpResponsePtr HttpClient::request(const std::string& url,
|
|
|
|
const std::string& verb,
|
|
|
|
const std::string& body,
|
|
|
|
HttpRequestArgsPtr args,
|
|
|
|
int redirects)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
2019-06-06 22:48:53 +02:00
|
|
|
// We only have one socket connection, so we cannot
|
2019-06-06 03:47:48 +02:00
|
|
|
// make multiple requests concurrently.
|
2020-06-19 09:11:06 +02:00
|
|
|
std::lock_guard<std::recursive_mutex> lock(_mutex);
|
2019-06-06 03:47:48 +02:00
|
|
|
|
2019-03-01 06:54:03 +01:00
|
|
|
uint64_t uploadSize = 0;
|
|
|
|
uint64_t downloadSize = 0;
|
|
|
|
int code = 0;
|
|
|
|
WebSocketHttpHeaders headers;
|
|
|
|
std::string payload;
|
2019-06-23 23:54:21 +02:00
|
|
|
std::string description;
|
2019-03-01 06:54:03 +01:00
|
|
|
|
|
|
|
std::string protocol, host, path, query;
|
|
|
|
int port;
|
|
|
|
|
2019-05-06 23:45:02 +02:00
|
|
|
if (!UrlParser::parse(url, protocol, host, path, query, port))
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << "Cannot parse url: " << url;
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::UrlMalformed,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
ss.str(),
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool tls = protocol == "https";
|
|
|
|
std::string errorMsg;
|
2019-10-01 07:06:46 +02:00
|
|
|
_socket = createSocket(tls, -1, errorMsg, _tlsOptions);
|
2019-03-01 06:54:03 +01:00
|
|
|
|
|
|
|
if (!_socket)
|
|
|
|
{
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::CannotCreateSocket,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
errorMsg,
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Build request string
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << verb << " " << path << " HTTP/1.1\r\n";
|
|
|
|
ss << "Host: " << host << "\r\n";
|
|
|
|
|
2020-08-01 07:54:57 +02:00
|
|
|
#ifdef IXWEBSOCKET_USE_ZLIB
|
2019-06-06 02:04:24 +02:00
|
|
|
if (args->compress)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
2019-09-23 19:25:23 +02:00
|
|
|
ss << "Accept-Encoding: gzip"
|
|
|
|
<< "\r\n";
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
2020-08-01 07:54:57 +02:00
|
|
|
#endif
|
2019-03-01 06:54:03 +01:00
|
|
|
|
|
|
|
// Append extra headers
|
2019-06-06 02:04:24 +02:00
|
|
|
for (auto&& it : args->extraHeaders)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
|
|
|
ss << it.first << ": " << it.second << "\r\n";
|
|
|
|
}
|
|
|
|
|
2019-06-03 23:02:54 +02:00
|
|
|
// Set a default Accept header if none is present
|
|
|
|
if (headers.find("Accept") == headers.end())
|
|
|
|
{
|
2019-09-23 19:25:23 +02:00
|
|
|
ss << "Accept: */*"
|
|
|
|
<< "\r\n";
|
2019-06-03 23:02:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Set a default User agent if none is present
|
|
|
|
if (headers.find("User-Agent") == headers.end())
|
|
|
|
{
|
2019-08-30 21:48:18 +02:00
|
|
|
ss << "User-Agent: " << userAgent() << "\r\n";
|
2019-06-03 23:02:54 +02:00
|
|
|
}
|
|
|
|
|
2020-05-05 16:38:55 +02:00
|
|
|
if (verb == kPost || verb == kPut || verb == kPatch || _forceBody)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
2020-10-10 02:51:56 +02:00
|
|
|
// Set request compression header
|
2020-10-19 22:36:04 +02:00
|
|
|
#ifdef IXWEBSOCKET_USE_ZLIB
|
2020-10-10 02:51:56 +02:00
|
|
|
if (args->compressRequest)
|
|
|
|
{
|
|
|
|
ss << "Content-Encoding: gzip"
|
|
|
|
<< "\r\n";
|
|
|
|
}
|
2020-10-19 22:36:04 +02:00
|
|
|
#endif
|
2020-10-10 02:51:56 +02:00
|
|
|
|
2019-03-01 06:54:03 +01:00
|
|
|
ss << "Content-Length: " << body.size() << "\r\n";
|
|
|
|
|
|
|
|
// Set default Content-Type if unspecified
|
2019-06-06 02:04:24 +02:00
|
|
|
if (args->extraHeaders.find("Content-Type") == args->extraHeaders.end())
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
2019-11-26 06:08:43 +01:00
|
|
|
if (args->multipartBoundary.empty())
|
|
|
|
{
|
|
|
|
ss << "Content-Type: application/x-www-form-urlencoded"
|
|
|
|
<< "\r\n";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ss << "Content-Type: multipart/form-data; boundary=" << args->multipartBoundary
|
|
|
|
<< "\r\n";
|
|
|
|
}
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
ss << "\r\n";
|
|
|
|
ss << body;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ss << "\r\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string req(ss.str());
|
|
|
|
std::string errMsg;
|
|
|
|
|
|
|
|
// Make a cancellation object dealing with connection timeout
|
|
|
|
auto isCancellationRequested =
|
2020-04-29 20:53:23 +02:00
|
|
|
makeCancellationRequestWithTimeout(args->connectTimeout, _stop);
|
2019-03-01 06:54:03 +01:00
|
|
|
|
|
|
|
bool success = _socket->connect(host, port, errMsg, isCancellationRequested);
|
|
|
|
if (!success)
|
|
|
|
{
|
|
|
|
std::stringstream ss;
|
2019-06-02 02:41:48 +02:00
|
|
|
ss << "Cannot connect to url: " << url << " / error : " << errMsg;
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::CannotConnect,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
ss.str(),
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Make a new cancellation object dealing with transfer timeout
|
2020-05-05 02:19:25 +02:00
|
|
|
isCancellationRequested = makeCancellationRequestWithTimeout(args->transferTimeout, _stop);
|
2019-03-01 06:54:03 +01:00
|
|
|
|
2019-06-06 02:04:24 +02:00
|
|
|
if (args->verbose)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << "Sending " << verb << " request "
|
|
|
|
<< "to " << host << ":" << port << std::endl
|
|
|
|
<< "request size: " << req.size() << " bytes" << std::endl
|
|
|
|
<< "=============" << std::endl
|
2019-09-23 19:25:23 +02:00
|
|
|
<< req << "=============" << std::endl
|
2019-03-01 06:54:03 +01:00
|
|
|
<< std::endl;
|
|
|
|
|
|
|
|
log(ss.str(), args);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!_socket->writeBytes(req, isCancellationRequested))
|
|
|
|
{
|
|
|
|
std::string errorMsg("Cannot send request");
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::SendError,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
errorMsg,
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
uploadSize = req.size();
|
|
|
|
|
|
|
|
auto lineResult = _socket->readLine(isCancellationRequested);
|
|
|
|
auto lineValid = lineResult.first;
|
|
|
|
auto line = lineResult.second;
|
|
|
|
|
|
|
|
if (!lineValid)
|
|
|
|
{
|
|
|
|
std::string errorMsg("Cannot retrieve status line");
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::CannotReadStatusLine,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
errorMsg,
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
2019-06-06 02:04:24 +02:00
|
|
|
if (args->verbose)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << "Status line " << line;
|
|
|
|
log(ss.str(), args);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sscanf(line.c_str(), "HTTP/1.1 %d", &code) != 1)
|
|
|
|
{
|
|
|
|
std::string errorMsg("Cannot parse response code from status line");
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::MissingStatus,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
errorMsg,
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
auto result = parseHttpHeaders(_socket, isCancellationRequested);
|
|
|
|
auto headersValid = result.first;
|
|
|
|
headers = result.second;
|
|
|
|
|
|
|
|
if (!headersValid)
|
|
|
|
{
|
|
|
|
std::string errorMsg("Cannot parse http headers");
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::HeaderParsingError,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
errorMsg,
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Redirect ?
|
2019-06-06 02:04:24 +02:00
|
|
|
if ((code >= 301 && code <= 308) && args->followRedirects)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
|
|
|
if (headers.find("Location") == headers.end())
|
|
|
|
{
|
|
|
|
std::string errorMsg("Missing location header for redirect");
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::MissingLocation,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
errorMsg,
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
2019-06-06 02:04:24 +02:00
|
|
|
if (redirects >= args->maxRedirects)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << "Too many redirects: " << redirects;
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::TooManyRedirects,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
ss.str(),
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Recurse
|
|
|
|
std::string location = headers["Location"];
|
2019-09-23 19:25:23 +02:00
|
|
|
return request(location, verb, body, args, redirects + 1);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (verb == "HEAD")
|
|
|
|
{
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::Ok,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
std::string(),
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Parse response:
|
|
|
|
if (headers.find("Content-Length") != headers.end())
|
|
|
|
{
|
|
|
|
ssize_t contentLength = -1;
|
|
|
|
ss.str("");
|
|
|
|
ss << headers["Content-Length"];
|
|
|
|
ss >> contentLength;
|
|
|
|
|
|
|
|
payload.reserve(contentLength);
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
auto chunkResult = _socket->readBytes(
|
|
|
|
contentLength, args->onProgressCallback, isCancellationRequested);
|
2019-03-02 20:01:51 +01:00
|
|
|
if (!chunkResult.first)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
2019-03-02 20:01:51 +01:00
|
|
|
errorMsg = "Cannot read chunk";
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::ChunkReadError,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
errorMsg,
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
2019-03-02 20:01:51 +01:00
|
|
|
payload += chunkResult.second;
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
else if (headers.find("Transfer-Encoding") != headers.end() &&
|
|
|
|
headers["Transfer-Encoding"] == "chunked")
|
|
|
|
{
|
|
|
|
std::stringstream ss;
|
|
|
|
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
lineResult = _socket->readLine(isCancellationRequested);
|
|
|
|
line = lineResult.second;
|
|
|
|
|
|
|
|
if (!lineResult.first)
|
|
|
|
{
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::ChunkReadError,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
errorMsg,
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t chunkSize;
|
|
|
|
ss.str("");
|
|
|
|
ss << std::hex << line;
|
|
|
|
ss >> chunkSize;
|
|
|
|
|
2019-06-06 02:04:24 +02:00
|
|
|
if (args->verbose)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
|
|
|
std::stringstream oss;
|
2019-09-23 19:25:23 +02:00
|
|
|
oss << "Reading " << chunkSize << " bytes" << std::endl;
|
2019-03-01 06:54:03 +01:00
|
|
|
log(oss.str(), args);
|
|
|
|
}
|
|
|
|
|
2019-06-03 05:46:20 +02:00
|
|
|
payload.reserve(payload.size() + (size_t) chunkSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
|
2019-03-02 20:01:51 +01:00
|
|
|
// Read a chunk
|
2019-09-23 19:25:23 +02:00
|
|
|
auto chunkResult = _socket->readBytes(
|
|
|
|
(size_t) chunkSize, args->onProgressCallback, isCancellationRequested);
|
2019-03-02 20:01:51 +01:00
|
|
|
if (!chunkResult.first)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
2019-03-02 20:01:51 +01:00
|
|
|
errorMsg = "Cannot read chunk";
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::ChunkReadError,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
errorMsg,
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
2019-03-02 20:01:51 +01:00
|
|
|
payload += chunkResult.second;
|
2019-03-01 06:54:03 +01:00
|
|
|
|
2019-03-02 20:01:51 +01:00
|
|
|
// Read the line that terminates the chunk (\r\n)
|
2019-03-01 06:54:03 +01:00
|
|
|
lineResult = _socket->readLine(isCancellationRequested);
|
|
|
|
|
|
|
|
if (!lineResult.first)
|
|
|
|
{
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::ChunkReadError,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
errorMsg,
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (chunkSize == 0) break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (code == 204)
|
|
|
|
{
|
|
|
|
; // 204 is NoContent response code
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::string errorMsg("Cannot read http body");
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::CannotReadBody,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
errorMsg,
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
downloadSize = payload.size();
|
|
|
|
|
|
|
|
// If the content was compressed with gzip, decode it
|
|
|
|
if (headers["Content-Encoding"] == "gzip")
|
|
|
|
{
|
2020-09-28 19:19:27 +02:00
|
|
|
#ifdef IXWEBSOCKET_USE_ZLIB
|
2019-03-01 06:54:03 +01:00
|
|
|
std::string decompressedPayload;
|
2020-09-28 19:19:27 +02:00
|
|
|
if (!gzipDecompress(payload, decompressedPayload))
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
|
|
|
std::string errorMsg("Error decompressing payload");
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::Gzip,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
errorMsg,
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
payload = decompressedPayload;
|
2020-09-28 19:19:27 +02:00
|
|
|
#else
|
|
|
|
std::string errorMsg("ixwebsocket was not compiled with gzip support on");
|
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::Gzip,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
errorMsg,
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2020-08-01 07:54:57 +02:00
|
|
|
#endif
|
2020-09-28 19:19:27 +02:00
|
|
|
}
|
2019-03-01 06:54:03 +01:00
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
return std::make_shared<HttpResponse>(code,
|
|
|
|
description,
|
|
|
|
HttpErrorCode::Ok,
|
|
|
|
headers,
|
|
|
|
payload,
|
|
|
|
std::string(),
|
|
|
|
uploadSize,
|
|
|
|
downloadSize);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
HttpResponsePtr HttpClient::get(const std::string& url, HttpRequestArgsPtr args)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
|
|
|
return request(url, kGet, std::string(), args);
|
|
|
|
}
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
HttpResponsePtr HttpClient::head(const std::string& url, HttpRequestArgsPtr args)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
|
|
|
return request(url, kHead, std::string(), args);
|
|
|
|
}
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
HttpResponsePtr HttpClient::del(const std::string& url, HttpRequestArgsPtr args)
|
2019-06-03 20:38:56 +02:00
|
|
|
{
|
|
|
|
return request(url, kDel, std::string(), args);
|
|
|
|
}
|
|
|
|
|
2020-10-10 02:51:56 +02:00
|
|
|
HttpResponsePtr HttpClient::request(const std::string& url,
|
|
|
|
const std::string& verb,
|
|
|
|
const HttpParameters& httpParameters,
|
|
|
|
const HttpFormDataParameters& httpFormDataParameters,
|
|
|
|
HttpRequestArgsPtr args)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
2020-10-10 02:51:56 +02:00
|
|
|
std::string body;
|
|
|
|
|
2020-10-08 21:43:18 +02:00
|
|
|
if (httpFormDataParameters.empty())
|
|
|
|
{
|
2020-10-10 02:51:56 +02:00
|
|
|
body = serializeHttpParameters(httpParameters);
|
2020-10-08 21:43:18 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::string multipartBoundary = generateMultipartBoundary();
|
|
|
|
args->multipartBoundary = multipartBoundary;
|
2020-10-10 02:51:56 +02:00
|
|
|
body = serializeHttpFormDataParameters(
|
2020-10-08 21:43:18 +02:00
|
|
|
multipartBoundary, httpFormDataParameters, httpParameters);
|
|
|
|
}
|
2020-10-10 02:51:56 +02:00
|
|
|
|
2020-10-19 22:36:04 +02:00
|
|
|
#ifdef IXWEBSOCKET_USE_ZLIB
|
2020-10-10 02:51:56 +02:00
|
|
|
if (args->compressRequest)
|
|
|
|
{
|
|
|
|
body = gzipCompress(body);
|
|
|
|
}
|
2020-10-19 22:36:04 +02:00
|
|
|
#endif
|
2020-10-10 02:51:56 +02:00
|
|
|
|
|
|
|
return request(url, verb, body, args);
|
|
|
|
}
|
|
|
|
|
|
|
|
HttpResponsePtr HttpClient::post(const std::string& url,
|
|
|
|
const HttpParameters& httpParameters,
|
|
|
|
const HttpFormDataParameters& httpFormDataParameters,
|
|
|
|
HttpRequestArgsPtr args)
|
|
|
|
{
|
|
|
|
return request(url, kPost, httpParameters, httpFormDataParameters, args);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
|
2019-06-06 02:04:24 +02:00
|
|
|
HttpResponsePtr HttpClient::post(const std::string& url,
|
|
|
|
const std::string& body,
|
|
|
|
HttpRequestArgsPtr args)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
|
|
|
return request(url, kPost, body, args);
|
|
|
|
}
|
|
|
|
|
2019-06-06 02:04:24 +02:00
|
|
|
HttpResponsePtr HttpClient::put(const std::string& url,
|
|
|
|
const HttpParameters& httpParameters,
|
2020-10-08 21:43:18 +02:00
|
|
|
const HttpFormDataParameters& httpFormDataParameters,
|
2019-06-06 02:04:24 +02:00
|
|
|
HttpRequestArgsPtr args)
|
2019-06-03 20:38:56 +02:00
|
|
|
{
|
2020-10-10 02:51:56 +02:00
|
|
|
return request(url, kPut, httpParameters, httpFormDataParameters, args);
|
2019-06-03 20:38:56 +02:00
|
|
|
}
|
|
|
|
|
2019-06-06 02:04:24 +02:00
|
|
|
HttpResponsePtr HttpClient::put(const std::string& url,
|
|
|
|
const std::string& body,
|
|
|
|
const HttpRequestArgsPtr args)
|
2019-06-03 20:38:56 +02:00
|
|
|
{
|
|
|
|
return request(url, kPut, body, args);
|
|
|
|
}
|
|
|
|
|
2020-05-05 16:43:55 +02:00
|
|
|
HttpResponsePtr HttpClient::patch(const std::string& url,
|
|
|
|
const HttpParameters& httpParameters,
|
2020-10-08 21:43:18 +02:00
|
|
|
const HttpFormDataParameters& httpFormDataParameters,
|
2020-05-05 16:43:55 +02:00
|
|
|
HttpRequestArgsPtr args)
|
|
|
|
{
|
2020-10-10 02:51:56 +02:00
|
|
|
return request(url, kPatch, httpParameters, httpFormDataParameters, args);
|
2020-05-05 16:43:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
HttpResponsePtr HttpClient::patch(const std::string& url,
|
|
|
|
const std::string& body,
|
|
|
|
const HttpRequestArgsPtr args)
|
|
|
|
{
|
|
|
|
return request(url, kPatch, body, args);
|
|
|
|
}
|
|
|
|
|
2019-03-01 06:54:03 +01:00
|
|
|
std::string HttpClient::urlEncode(const std::string& value)
|
|
|
|
{
|
|
|
|
std::ostringstream escaped;
|
|
|
|
escaped.fill('0');
|
|
|
|
escaped << std::hex;
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
for (std::string::const_iterator i = value.begin(), n = value.end(); i != n; ++i)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
|
|
|
std::string::value_type c = (*i);
|
|
|
|
|
|
|
|
// Keep alphanumeric and other accepted characters intact
|
|
|
|
if (isalnum(c) || c == '-' || c == '_' || c == '.' || c == '~')
|
|
|
|
{
|
|
|
|
escaped << c;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Any other characters are percent-encoded
|
|
|
|
escaped << std::uppercase;
|
|
|
|
escaped << '%' << std::setw(2) << int((unsigned char) c);
|
|
|
|
escaped << std::nouppercase;
|
|
|
|
}
|
|
|
|
|
|
|
|
return escaped.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string HttpClient::serializeHttpParameters(const HttpParameters& httpParameters)
|
|
|
|
{
|
|
|
|
std::stringstream ss;
|
|
|
|
size_t count = httpParameters.size();
|
|
|
|
size_t i = 0;
|
|
|
|
|
|
|
|
for (auto&& it : httpParameters)
|
|
|
|
{
|
2019-09-23 19:25:23 +02:00
|
|
|
ss << urlEncode(it.first) << "=" << urlEncode(it.second);
|
2019-03-01 06:54:03 +01:00
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
if (i++ < (count - 1))
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
2019-09-23 19:25:23 +02:00
|
|
|
ss << "&";
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return ss.str();
|
|
|
|
}
|
|
|
|
|
2019-11-26 06:08:43 +01:00
|
|
|
std::string HttpClient::serializeHttpFormDataParameters(
|
|
|
|
const std::string& multipartBoundary,
|
|
|
|
const HttpFormDataParameters& httpFormDataParameters,
|
|
|
|
const HttpParameters& httpParameters)
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// --AaB03x
|
|
|
|
// Content-Disposition: form-data; name="submit-name"
|
|
|
|
|
|
|
|
// Larry
|
|
|
|
// --AaB03x
|
|
|
|
// Content-Disposition: form-data; name="foo.txt"; filename="file1.txt"
|
|
|
|
// Content-Type: text/plain
|
|
|
|
|
|
|
|
// ... contents of file1.txt ...
|
|
|
|
// --AaB03x--
|
|
|
|
//
|
|
|
|
std::stringstream ss;
|
|
|
|
|
|
|
|
for (auto&& it : httpFormDataParameters)
|
|
|
|
{
|
|
|
|
ss << "--" << multipartBoundary << "\r\n"
|
|
|
|
<< "Content-Disposition:"
|
|
|
|
<< " form-data; name=\"" << it.first << "\";"
|
|
|
|
<< " filename=\"" << it.first << "\""
|
|
|
|
<< "\r\n"
|
|
|
|
<< "Content-Type: application/octet-stream"
|
|
|
|
<< "\r\n"
|
|
|
|
<< "\r\n"
|
|
|
|
<< it.second << "\r\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto&& it : httpParameters)
|
|
|
|
{
|
|
|
|
ss << "--" << multipartBoundary << "\r\n"
|
|
|
|
<< "Content-Disposition:"
|
|
|
|
<< " form-data; name=\"" << it.first << "\";"
|
|
|
|
<< "\r\n"
|
|
|
|
<< "\r\n"
|
|
|
|
<< it.second << "\r\n";
|
|
|
|
}
|
|
|
|
|
2020-02-13 21:25:18 +01:00
|
|
|
ss << "--" << multipartBoundary << "--\r\n";
|
2019-11-26 06:08:43 +01:00
|
|
|
|
|
|
|
return ss.str();
|
|
|
|
}
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
void HttpClient::log(const std::string& msg, HttpRequestArgsPtr args)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
2019-06-06 02:04:24 +02:00
|
|
|
if (args->logger)
|
2019-03-01 06:54:03 +01:00
|
|
|
{
|
2019-06-06 02:04:24 +02:00
|
|
|
args->logger(msg);
|
2019-03-01 06:54:03 +01:00
|
|
|
}
|
|
|
|
}
|
2019-11-26 06:08:43 +01:00
|
|
|
|
|
|
|
std::string HttpClient::generateMultipartBoundary()
|
|
|
|
{
|
|
|
|
std::string str("0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz");
|
|
|
|
|
|
|
|
static std::random_device rd;
|
|
|
|
static std::mt19937 generator(rd());
|
|
|
|
|
|
|
|
std::shuffle(str.begin(), str.end(), generator);
|
|
|
|
|
|
|
|
return str;
|
|
|
|
}
|
2019-09-23 19:25:23 +02:00
|
|
|
} // namespace ix
|