2018-06-26 15:33:51 +02:00
|
|
|
#include "common/NetworkRequest.hpp"
|
2018-01-19 22:45:33 +01:00
|
|
|
|
2018-06-26 14:09:39 +02:00
|
|
|
#include "Application.hpp"
|
2018-07-07 13:08:57 +02:00
|
|
|
#include "common/NetworkManager.hpp"
|
|
|
|
#include "providers/twitch/TwitchCommon.hpp"
|
|
|
|
#include "singletons/Paths.hpp"
|
|
|
|
|
|
|
|
#include <QFile>
|
|
|
|
|
|
|
|
#include <cassert>
|
2018-04-27 22:11:19 +02:00
|
|
|
|
2018-01-19 22:45:33 +01:00
|
|
|
namespace chatterino {
|
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
NetworkRequest::NetworkRequest(const std::string &url, NetworkRequestType requestType)
|
|
|
|
: timer(new NetworkTimer)
|
2018-01-19 22:45:33 +01:00
|
|
|
{
|
2018-07-07 13:08:57 +02:00
|
|
|
this->data.request_.setUrl(QUrl(QString::fromStdString(url)));
|
|
|
|
this->data.requestType_ = requestType;
|
2018-01-19 22:45:33 +01:00
|
|
|
}
|
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
NetworkRequest::NetworkRequest(QUrl url, NetworkRequestType requestType)
|
|
|
|
: timer(new NetworkTimer)
|
2018-01-19 22:45:33 +01:00
|
|
|
{
|
2018-07-07 13:08:57 +02:00
|
|
|
this->data.request_.setUrl(url);
|
|
|
|
this->data.requestType_ = requestType;
|
2018-01-19 22:45:33 +01:00
|
|
|
}
|
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
NetworkRequest::~NetworkRequest()
|
2018-01-19 22:45:33 +01:00
|
|
|
{
|
2018-07-07 13:08:57 +02:00
|
|
|
assert(this->executed_);
|
2018-01-19 22:45:33 +01:00
|
|
|
}
|
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
void NetworkRequest::setRequestType(NetworkRequestType newRequestType)
|
2018-07-06 21:31:58 +02:00
|
|
|
{
|
2018-07-07 13:08:57 +02:00
|
|
|
this->data.requestType_ = newRequestType;
|
2018-07-06 21:31:58 +02:00
|
|
|
}
|
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
void NetworkRequest::setCaller(const QObject *caller)
|
2018-07-06 17:56:11 +02:00
|
|
|
{
|
2018-07-07 13:08:57 +02:00
|
|
|
this->data.caller_ = caller;
|
2018-07-06 17:56:11 +02:00
|
|
|
}
|
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
void NetworkRequest::onReplyCreated(NetworkReplyCreatedCallback cb)
|
|
|
|
{
|
|
|
|
this->data.onReplyCreated_ = cb;
|
|
|
|
}
|
|
|
|
|
|
|
|
void NetworkRequest::onError(NetworkErrorCallback cb)
|
2018-07-06 17:56:11 +02:00
|
|
|
{
|
2018-07-07 13:08:57 +02:00
|
|
|
this->data.onError_ = cb;
|
2018-07-06 17:56:11 +02:00
|
|
|
}
|
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
void NetworkRequest::onSuccess(NetworkSuccessCallback cb)
|
2018-07-06 17:56:11 +02:00
|
|
|
{
|
2018-07-07 13:08:57 +02:00
|
|
|
this->data.onSuccess_ = cb;
|
2018-07-06 17:56:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void NetworkRequest::setRawHeader(const char *headerName, const char *value)
|
|
|
|
{
|
2018-07-07 13:08:57 +02:00
|
|
|
this->data.request_.setRawHeader(headerName, value);
|
2018-07-06 17:56:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void NetworkRequest::setRawHeader(const char *headerName, const QByteArray &value)
|
|
|
|
{
|
2018-07-07 13:08:57 +02:00
|
|
|
this->data.request_.setRawHeader(headerName, value);
|
2018-07-06 17:56:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void NetworkRequest::setRawHeader(const char *headerName, const QString &value)
|
|
|
|
{
|
2018-07-07 13:08:57 +02:00
|
|
|
this->data.request_.setRawHeader(headerName, value.toUtf8());
|
2018-07-06 17:56:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void NetworkRequest::setTimeout(int ms)
|
|
|
|
{
|
2018-07-07 13:08:57 +02:00
|
|
|
this->timer->timeoutMS_ = ms;
|
2018-07-06 17:56:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void NetworkRequest::makeAuthorizedV5(const QString &clientID, const QString &oauthToken)
|
|
|
|
{
|
|
|
|
this->setRawHeader("Client-ID", clientID);
|
|
|
|
this->setRawHeader("Accept", "application/vnd.twitchtv.v5+json");
|
2018-07-06 22:00:03 +02:00
|
|
|
if (!oauthToken.isEmpty()) {
|
|
|
|
this->setRawHeader("Authorization", "OAuth " + oauthToken);
|
|
|
|
}
|
2018-07-06 17:56:11 +02:00
|
|
|
}
|
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
void NetworkRequest::setPayload(const QByteArray &payload)
|
2018-01-19 22:45:33 +01:00
|
|
|
{
|
2018-07-07 13:08:57 +02:00
|
|
|
this->data.payload_ = payload;
|
2018-01-19 22:45:33 +01:00
|
|
|
}
|
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
void NetworkRequest::setUseQuickLoadCache(bool value)
|
2018-01-19 22:45:33 +01:00
|
|
|
{
|
2018-07-07 13:08:57 +02:00
|
|
|
this->data.useQuickLoadCache_ = value;
|
2018-01-19 22:45:33 +01:00
|
|
|
}
|
|
|
|
|
2018-07-06 17:56:11 +02:00
|
|
|
void NetworkRequest::execute()
|
|
|
|
{
|
2018-07-07 13:08:57 +02:00
|
|
|
this->executed_ = true;
|
|
|
|
|
|
|
|
switch (this->data.requestType_) {
|
|
|
|
case NetworkRequestType::Get: {
|
|
|
|
// Get requests try to load from cache, then perform the request
|
|
|
|
if (this->data.useQuickLoadCache_) {
|
|
|
|
if (this->tryLoadCachedFile()) {
|
|
|
|
Log("Loaded from cache");
|
|
|
|
// Successfully loaded from cache
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this->doRequest();
|
2018-07-06 17:56:11 +02:00
|
|
|
} break;
|
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
case NetworkRequestType::Put: {
|
|
|
|
// Put requests cannot be cached, therefore the request is called immediately
|
|
|
|
this->doRequest();
|
2018-07-06 17:56:11 +02:00
|
|
|
} break;
|
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
case NetworkRequestType::Delete: {
|
|
|
|
// Delete requests cannot be cached, therefore the request is called immediately
|
|
|
|
this->doRequest();
|
2018-07-06 17:56:11 +02:00
|
|
|
} break;
|
|
|
|
|
|
|
|
default: {
|
2018-07-07 13:08:57 +02:00
|
|
|
Log("[Execute] Unhandled request type");
|
2018-07-06 17:56:11 +02:00
|
|
|
} break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
bool NetworkRequest::tryLoadCachedFile()
|
2018-07-06 17:56:11 +02:00
|
|
|
{
|
2018-07-07 13:08:57 +02:00
|
|
|
auto app = getApp();
|
2018-07-06 17:56:11 +02:00
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
QFile cachedFile(app->paths->cacheDirectory + "/" + this->data.getHash());
|
2018-07-06 17:56:11 +02:00
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
if (!cachedFile.exists()) {
|
|
|
|
// File didn't exist
|
|
|
|
return false;
|
|
|
|
}
|
2018-07-06 17:56:11 +02:00
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
if (!cachedFile.open(QIODevice::ReadOnly)) {
|
|
|
|
// File could not be opened
|
|
|
|
return false;
|
|
|
|
}
|
2018-07-06 17:56:11 +02:00
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
QByteArray bytes = cachedFile.readAll();
|
|
|
|
NetworkResult result(bytes);
|
2018-07-06 17:56:11 +02:00
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
bool success = this->data.onSuccess_(result);
|
2018-07-06 17:56:11 +02:00
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
cachedFile.close();
|
2018-07-06 17:56:11 +02:00
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
// XXX: If success is false, we should invalidate the cache file somehow/somewhere
|
2018-07-06 17:56:11 +02:00
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
return success;
|
2018-07-06 17:56:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void NetworkRequest::doRequest()
|
|
|
|
{
|
|
|
|
NetworkRequester requester;
|
|
|
|
NetworkWorker *worker = new NetworkWorker;
|
|
|
|
|
|
|
|
worker->moveToThread(&NetworkManager::workerThread);
|
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
this->timer->start();
|
2018-07-06 17:56:11 +02:00
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
auto onUrlRequested = [data = std::move(this->data), timer = std::move(this->timer),
|
|
|
|
worker]() mutable {
|
|
|
|
QNetworkReply *reply = nullptr;
|
|
|
|
switch (data.requestType_) {
|
|
|
|
case NetworkRequestType::Get: {
|
|
|
|
reply = NetworkManager::NaM.get(data.request_);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case NetworkRequestType::Put: {
|
|
|
|
reply = NetworkManager::NaM.put(data.request_, data.payload_);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case NetworkRequestType::Delete: {
|
|
|
|
reply = NetworkManager::NaM.deleteResource(data.request_);
|
|
|
|
} break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (reply == nullptr) {
|
|
|
|
Log("Unhandled request type");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (timer->isStarted()) {
|
|
|
|
timer->onTimeout(worker, [reply, data]() {
|
|
|
|
Log("Aborted!");
|
|
|
|
reply->abort();
|
|
|
|
if (data.onError_) {
|
|
|
|
data.onError_(-2);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data.onReplyCreated_) {
|
|
|
|
data.onReplyCreated_(reply);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool directAction = (data.caller_ == nullptr);
|
|
|
|
|
|
|
|
auto handleReply = [data = std::move(data), timer = std::move(timer), reply]() mutable {
|
|
|
|
if (reply->error() != QNetworkReply::NetworkError::NoError) {
|
|
|
|
if (data.onError_) {
|
|
|
|
data.onError_(reply->error());
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QByteArray readBytes = reply->readAll();
|
|
|
|
QByteArray bytes;
|
|
|
|
bytes.setRawData(readBytes.data(), readBytes.size());
|
|
|
|
data.writeToCache(bytes);
|
|
|
|
|
|
|
|
NetworkResult result(bytes);
|
|
|
|
data.onSuccess_(result);
|
2018-07-06 17:56:11 +02:00
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
reply->deleteLater();
|
|
|
|
};
|
|
|
|
|
|
|
|
if (data.caller_ != nullptr) {
|
|
|
|
QObject::connect(worker, &NetworkWorker::doneUrl, data.caller_, std::move(handleReply));
|
|
|
|
QObject::connect(reply, &QNetworkReply::finished, worker, [worker]() mutable {
|
|
|
|
emit worker->doneUrl();
|
|
|
|
|
|
|
|
delete worker;
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
QObject::connect(reply, &QNetworkReply::finished, worker,
|
|
|
|
[handleReply = std::move(handleReply), worker]() mutable {
|
|
|
|
handleReply();
|
|
|
|
|
|
|
|
delete worker;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
QObject::connect(&requester, &NetworkRequester::requestUrl, worker, std::move(onUrlRequested));
|
2018-07-06 17:56:11 +02:00
|
|
|
|
|
|
|
emit requester.requestUrl();
|
|
|
|
}
|
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
// Helper creator functions
|
|
|
|
NetworkRequest NetworkRequest::twitchRequest(QUrl url)
|
2018-07-06 17:56:11 +02:00
|
|
|
{
|
2018-07-07 13:08:57 +02:00
|
|
|
NetworkRequest request(url);
|
2018-07-06 17:56:11 +02:00
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
request.makeAuthorizedV5(getDefaultClientID());
|
2018-07-06 17:56:11 +02:00
|
|
|
|
2018-07-07 13:08:57 +02:00
|
|
|
return request;
|
2018-07-06 17:56:11 +02:00
|
|
|
}
|
|
|
|
|
2018-01-19 22:45:33 +01:00
|
|
|
} // namespace chatterino
|