mirror of
https://github.com/Chatterino/chatterino2.git
synced 2024-11-21 22:24:07 +01:00
276 lines
7.5 KiB
C++
276 lines
7.5 KiB
C++
#include "common/NetworkPrivate.hpp"
|
|
|
|
#include "common/NetworkManager.hpp"
|
|
#include "common/NetworkResult.hpp"
|
|
#include "common/Outcome.hpp"
|
|
#include "debug/AssertInGuiThread.hpp"
|
|
#include "singletons/Paths.hpp"
|
|
#include "util/DebugCount.hpp"
|
|
#include "util/PostToThread.hpp"
|
|
|
|
#include <QCryptographicHash>
|
|
#include <QFile>
|
|
#include <QtConcurrent>
|
|
|
|
namespace chatterino {
|
|
|
|
NetworkData::NetworkData()
|
|
: timer_(new QTimer())
|
|
, lifetimeManager_(new QObject)
|
|
{
|
|
timer_->setSingleShot(true);
|
|
|
|
DebugCount::increase("NetworkData");
|
|
}
|
|
|
|
NetworkData::~NetworkData()
|
|
{
|
|
this->timer_->deleteLater();
|
|
this->lifetimeManager_->deleteLater();
|
|
|
|
DebugCount::decrease("NetworkData");
|
|
}
|
|
|
|
QString NetworkData::getHash()
|
|
{
|
|
static std::mutex mu;
|
|
|
|
std::lock_guard lock(mu);
|
|
|
|
if (this->hash_.isEmpty())
|
|
{
|
|
QByteArray bytes;
|
|
|
|
bytes.append(this->request_.url().toString());
|
|
|
|
for (const auto &header : this->request_.rawHeaderList())
|
|
{
|
|
bytes.append(header);
|
|
}
|
|
|
|
QByteArray hashBytes(
|
|
QCryptographicHash::hash(bytes, QCryptographicHash::Sha256));
|
|
|
|
this->hash_ = hashBytes.toHex();
|
|
}
|
|
|
|
return this->hash_;
|
|
}
|
|
|
|
void writeToCache(const std::shared_ptr<NetworkData> &data,
|
|
const QByteArray &bytes)
|
|
{
|
|
if (data->cache_)
|
|
{
|
|
QtConcurrent::run([data, bytes] {
|
|
QFile cachedFile(getPaths()->cacheDirectory() + "/" +
|
|
data->getHash());
|
|
|
|
if (cachedFile.open(QIODevice::WriteOnly))
|
|
{
|
|
cachedFile.write(bytes);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
void loadUncached(const std::shared_ptr<NetworkData> &data)
|
|
{
|
|
DebugCount::increase("http request started");
|
|
|
|
NetworkRequester requester;
|
|
NetworkWorker *worker = new NetworkWorker;
|
|
|
|
worker->moveToThread(&NetworkManager::workerThread);
|
|
|
|
if (data->hasTimeout_)
|
|
{
|
|
data->timer_->setSingleShot(true);
|
|
data->timer_->start();
|
|
}
|
|
|
|
auto onUrlRequested = [data, worker]() mutable {
|
|
auto reply = [&]() -> QNetworkReply * {
|
|
switch (data->requestType_)
|
|
{
|
|
case NetworkRequestType::Get:
|
|
return NetworkManager::accessManager.get(data->request_);
|
|
|
|
case NetworkRequestType::Put:
|
|
return NetworkManager::accessManager.put(data->request_,
|
|
data->payload_);
|
|
|
|
case NetworkRequestType::Delete:
|
|
return NetworkManager::accessManager.deleteResource(
|
|
data->request_);
|
|
|
|
case NetworkRequestType::Post:
|
|
if (data->multiPartPayload_)
|
|
{
|
|
assert(data->payload_.isNull());
|
|
|
|
return NetworkManager::accessManager.post(
|
|
data->request_, data->multiPartPayload_);
|
|
}
|
|
else
|
|
{
|
|
return NetworkManager::accessManager.post(
|
|
data->request_, data->payload_);
|
|
}
|
|
}
|
|
return nullptr;
|
|
}();
|
|
|
|
if (reply == nullptr)
|
|
{
|
|
qDebug() << "Unhandled request type";
|
|
return;
|
|
}
|
|
|
|
if (data->timer_->isActive())
|
|
{
|
|
QObject::connect(
|
|
data->timer_, &QTimer::timeout, worker, [reply, data]() {
|
|
qDebug() << "Aborted!";
|
|
reply->abort();
|
|
if (data->onError_)
|
|
{
|
|
postToThread([data] {
|
|
data->onError_(NetworkResult(
|
|
{}, NetworkResult::timedoutStatus));
|
|
});
|
|
}
|
|
});
|
|
}
|
|
|
|
if (data->onReplyCreated_)
|
|
{
|
|
data->onReplyCreated_(reply);
|
|
}
|
|
|
|
auto handleReply = [data, reply]() mutable {
|
|
if (data->hasCaller_ && !data->caller_.get())
|
|
{
|
|
return;
|
|
}
|
|
|
|
// TODO(pajlada): A reply was received, kill the timeout timer
|
|
if (reply->error() != QNetworkReply::NetworkError::NoError)
|
|
{
|
|
if (data->onError_)
|
|
{
|
|
auto error = reply->error();
|
|
postToThread([data, error] {
|
|
data->onError_(NetworkResult({}, error));
|
|
});
|
|
}
|
|
return;
|
|
}
|
|
|
|
QByteArray bytes = reply->readAll();
|
|
writeToCache(data, bytes);
|
|
|
|
auto status =
|
|
reply->attribute(QNetworkRequest::HttpStatusCodeAttribute);
|
|
|
|
NetworkResult result(bytes, status.toInt());
|
|
|
|
DebugCount::increase("http request success");
|
|
// log("starting {}", data->request_.url().toString());
|
|
if (data->onSuccess_)
|
|
{
|
|
if (data->executeConcurrently_)
|
|
QtConcurrent::run(
|
|
[onSuccess = std::move(data->onSuccess_),
|
|
result = std::move(result)] { onSuccess(result); });
|
|
else
|
|
data->onSuccess_(result);
|
|
}
|
|
// log("finished {}", data->request_.url().toString());
|
|
|
|
reply->deleteLater();
|
|
};
|
|
|
|
QObject::connect(
|
|
reply, &QNetworkReply::finished, worker,
|
|
[data, handleReply, worker]() mutable {
|
|
if (data->executeConcurrently_ || isGuiThread())
|
|
{
|
|
handleReply();
|
|
delete worker;
|
|
}
|
|
else
|
|
{
|
|
postToThread(
|
|
[worker, cb = std::move(handleReply)]() mutable {
|
|
cb();
|
|
delete worker;
|
|
});
|
|
}
|
|
});
|
|
};
|
|
|
|
QObject::connect(&requester, &NetworkRequester::requestUrl, worker,
|
|
onUrlRequested);
|
|
|
|
emit requester.requestUrl();
|
|
}
|
|
|
|
// First tried to load cached, then uncached.
|
|
void loadCached(const std::shared_ptr<NetworkData> &data)
|
|
{
|
|
QFile cachedFile(getPaths()->cacheDirectory() + "/" + data->getHash());
|
|
|
|
if (!cachedFile.exists() || !cachedFile.open(QIODevice::ReadOnly))
|
|
{
|
|
// File didn't exist OR File could not be opened
|
|
loadUncached(data);
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
// XXX: check if bytes is empty?
|
|
QByteArray bytes = cachedFile.readAll();
|
|
NetworkResult result(bytes, 200);
|
|
|
|
if (data->onSuccess_)
|
|
{
|
|
if (data->executeConcurrently_ || isGuiThread())
|
|
{
|
|
// XXX: If outcome is Failure, we should invalidate the cache file
|
|
// somehow/somewhere
|
|
/*auto outcome =*/
|
|
if (data->hasCaller_ && !data->caller_.get())
|
|
{
|
|
return;
|
|
}
|
|
data->onSuccess_(result);
|
|
}
|
|
else
|
|
{
|
|
postToThread([data, result]() {
|
|
if (data->hasCaller_ && !data->caller_.get())
|
|
{
|
|
return;
|
|
}
|
|
|
|
data->onSuccess_(result);
|
|
});
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void load(const std::shared_ptr<NetworkData> &data)
|
|
{
|
|
if (data->cache_)
|
|
{
|
|
QtConcurrent::run(loadCached, data);
|
|
}
|
|
else
|
|
{
|
|
loadUncached(data);
|
|
}
|
|
}
|
|
|
|
} // namespace chatterino
|