2017-07-13 11:26:03 +03:00
|
|
|
/*
|
|
|
|
* cdownloadmanager_moc.cpp, part of VCMI engine
|
|
|
|
*
|
|
|
|
* Authors: listed in file AUTHORS in main folder
|
|
|
|
*
|
|
|
|
* License: GNU General Public License v2.0 or later
|
|
|
|
* Full text of license available in license.txt file, in main folder
|
|
|
|
*
|
|
|
|
*/
|
2013-08-22 14:22:49 +00:00
|
|
|
#include "StdInc.h"
|
2013-08-24 20:11:51 +00:00
|
|
|
#include "cdownloadmanager_moc.h"
|
2013-08-22 14:22:49 +00:00
|
|
|
|
2024-10-07 08:47:00 +03:00
|
|
|
#include "../vcmiqt/launcherdirs.h"
|
2013-08-22 14:22:49 +00:00
|
|
|
|
2024-04-20 19:46:30 +03:00
|
|
|
#include "../../lib/CConfigHandler.h"
|
|
|
|
|
2013-08-22 14:22:49 +00:00
|
|
|
CDownloadManager::CDownloadManager()
|
|
|
|
{
|
2018-04-13 12:34:58 +07:00
|
|
|
connect(&manager, SIGNAL(finished(QNetworkReply *)),
|
|
|
|
SLOT(downloadFinished(QNetworkReply *)));
|
2024-04-20 19:46:30 +03:00
|
|
|
connect(&manager, &QNetworkAccessManager::sslErrors, [](QNetworkReply * reply, const QList<QSslError> & errors) {
|
|
|
|
if(settings["launcher"]["ignoreSslErrors"].Bool())
|
|
|
|
reply->ignoreSslErrors();
|
|
|
|
});
|
2013-08-22 14:22:49 +00:00
|
|
|
}
|
|
|
|
|
2023-09-01 18:43:26 +04:00
|
|
|
void CDownloadManager::downloadFile(const QUrl & url, const QString & file, qint64 bytesTotal)
|
2013-08-22 14:22:49 +00:00
|
|
|
{
|
|
|
|
QNetworkRequest request(url);
|
|
|
|
FileEntry entry;
|
2024-02-06 01:33:47 +03:00
|
|
|
entry.file.reset(new QFile(QString{QLatin1String{"%1/%2"}}.arg(CLauncherDirs::downloadsPath(), file)));
|
2013-08-22 14:22:49 +00:00
|
|
|
entry.bytesReceived = 0;
|
2023-09-01 18:43:26 +04:00
|
|
|
entry.totalSize = bytesTotal;
|
2023-04-14 12:47:13 +03:00
|
|
|
entry.filename = file;
|
2013-08-22 14:22:49 +00:00
|
|
|
|
2018-04-13 12:34:58 +07:00
|
|
|
if(entry.file->open(QIODevice::WriteOnly | QIODevice::Truncate))
|
2013-08-22 14:22:49 +00:00
|
|
|
{
|
|
|
|
entry.status = FileEntry::IN_PROGRESS;
|
|
|
|
entry.reply = manager.get(request);
|
|
|
|
|
2018-04-13 12:34:58 +07:00
|
|
|
connect(entry.reply, SIGNAL(downloadProgress(qint64,qint64)),
|
|
|
|
SLOT(downloadProgressChanged(qint64,qint64)));
|
2013-08-22 14:22:49 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
entry.status = FileEntry::FAILED;
|
2018-04-13 12:34:58 +07:00
|
|
|
entry.reply = nullptr;
|
2013-08-22 14:22:49 +00:00
|
|
|
encounteredErrors += entry.file->errorString();
|
|
|
|
}
|
|
|
|
|
|
|
|
// even if failed - add it into list to report it in finished() call
|
|
|
|
currentDownloads.push_back(entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
CDownloadManager::FileEntry & CDownloadManager::getEntry(QNetworkReply * reply)
|
|
|
|
{
|
|
|
|
assert(reply);
|
2018-04-13 12:34:58 +07:00
|
|
|
for(auto & entry : currentDownloads)
|
2013-08-22 14:22:49 +00:00
|
|
|
{
|
2018-04-13 12:34:58 +07:00
|
|
|
if(entry.reply == reply)
|
2013-08-22 14:22:49 +00:00
|
|
|
return entry;
|
|
|
|
}
|
2024-02-12 13:22:54 +02:00
|
|
|
throw std::runtime_error("Failed to find download entry");
|
2013-08-22 14:22:49 +00:00
|
|
|
}
|
|
|
|
|
2018-04-13 12:34:58 +07:00
|
|
|
void CDownloadManager::downloadFinished(QNetworkReply * reply)
|
2013-08-22 14:22:49 +00:00
|
|
|
{
|
|
|
|
FileEntry & file = getEntry(reply);
|
2023-06-27 22:34:34 +03:00
|
|
|
|
2022-09-04 20:32:48 +04:00
|
|
|
QVariant possibleRedirectUrl = reply->attribute(QNetworkRequest::RedirectionTargetAttribute);
|
|
|
|
QUrl qurl = possibleRedirectUrl.toUrl();
|
2023-06-27 22:34:34 +03:00
|
|
|
|
2022-09-04 20:32:48 +04:00
|
|
|
if(possibleRedirectUrl.isValid())
|
|
|
|
{
|
2023-04-14 12:47:13 +03:00
|
|
|
QString filename;
|
|
|
|
|
2022-09-04 20:32:48 +04:00
|
|
|
for(int i = 0; i< currentDownloads.size(); ++i)
|
|
|
|
{
|
|
|
|
if(currentDownloads[i].file == file.file)
|
|
|
|
{
|
2023-04-14 12:47:13 +03:00
|
|
|
filename = currentDownloads[i].filename;
|
2022-09-04 20:32:48 +04:00
|
|
|
currentDownloads.removeAt(i);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2023-09-01 18:43:26 +04:00
|
|
|
downloadFile(qurl, filename, file.totalSize);
|
2022-09-04 20:32:48 +04:00
|
|
|
return;
|
|
|
|
}
|
2013-08-22 14:22:49 +00:00
|
|
|
|
2018-04-13 12:34:58 +07:00
|
|
|
if(file.reply->error())
|
2013-08-22 14:22:49 +00:00
|
|
|
{
|
|
|
|
encounteredErrors += file.reply->errorString();
|
|
|
|
file.file->remove();
|
|
|
|
file.status = FileEntry::FAILED;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
file.file->write(file.reply->readAll());
|
|
|
|
file.file->close();
|
|
|
|
file.status = FileEntry::FINISHED;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool downloadComplete = true;
|
2018-04-13 12:34:58 +07:00
|
|
|
for(auto & entry : currentDownloads)
|
2013-08-22 14:22:49 +00:00
|
|
|
{
|
2018-04-13 12:34:58 +07:00
|
|
|
if(entry.status == FileEntry::IN_PROGRESS)
|
2013-08-22 14:22:49 +00:00
|
|
|
{
|
|
|
|
downloadComplete = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList successful;
|
|
|
|
QStringList failed;
|
|
|
|
|
2018-04-13 12:34:58 +07:00
|
|
|
for(auto & entry : currentDownloads)
|
2013-08-22 14:22:49 +00:00
|
|
|
{
|
2018-04-13 12:34:58 +07:00
|
|
|
if(entry.status == FileEntry::FINISHED)
|
2013-08-22 14:22:49 +00:00
|
|
|
successful += entry.file->fileName();
|
|
|
|
else
|
|
|
|
failed += entry.file->fileName();
|
|
|
|
}
|
|
|
|
|
2018-04-13 12:34:58 +07:00
|
|
|
if(downloadComplete)
|
2013-08-22 14:22:49 +00:00
|
|
|
emit finished(successful, failed, encounteredErrors);
|
2013-09-08 16:49:23 +00:00
|
|
|
|
|
|
|
file.reply->deleteLater();
|
|
|
|
file.reply = nullptr;
|
2013-08-22 14:22:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CDownloadManager::downloadProgressChanged(qint64 bytesReceived, qint64 bytesTotal)
|
|
|
|
{
|
|
|
|
auto reply = dynamic_cast<QNetworkReply *>(sender());
|
|
|
|
FileEntry & entry = getEntry(reply);
|
|
|
|
|
|
|
|
entry.file->write(entry.reply->readAll());
|
|
|
|
entry.bytesReceived = bytesReceived;
|
2023-09-19 23:38:00 +02:00
|
|
|
if(bytesTotal > entry.totalSize)
|
2023-09-01 18:43:26 +04:00
|
|
|
entry.totalSize = bytesTotal;
|
2013-08-22 14:22:49 +00:00
|
|
|
|
|
|
|
quint64 total = 0;
|
2018-04-13 12:34:58 +07:00
|
|
|
for(auto & entry : currentDownloads)
|
2023-06-27 22:34:34 +03:00
|
|
|
total += entry.totalSize > 0 ? entry.totalSize : entry.bytesReceived;
|
2013-08-22 14:22:49 +00:00
|
|
|
|
|
|
|
quint64 received = 0;
|
2018-04-13 12:34:58 +07:00
|
|
|
for(auto & entry : currentDownloads)
|
2013-08-22 14:22:49 +00:00
|
|
|
received += entry.bytesReceived > 0 ? entry.bytesReceived : 0;
|
2023-09-02 14:10:50 +04:00
|
|
|
|
|
|
|
if(received > total)
|
|
|
|
total = received;
|
2013-08-22 14:22:49 +00:00
|
|
|
|
|
|
|
emit downloadProgress(received, total);
|
|
|
|
}
|
2014-03-23 12:08:01 +00:00
|
|
|
|
2023-09-01 18:43:26 +04:00
|
|
|
bool CDownloadManager::downloadInProgress(const QUrl & url) const
|
2014-03-23 12:08:01 +00:00
|
|
|
{
|
2018-04-13 12:34:58 +07:00
|
|
|
for(auto & entry : currentDownloads)
|
2014-03-23 12:08:01 +00:00
|
|
|
{
|
2018-04-13 12:34:58 +07:00
|
|
|
if(entry.reply->url() == url)
|
2014-03-23 12:08:01 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|