GH-721 Redo internal NetJob implementation.
NetJob is now using its own task queue and does not start more than 6 actions at the same time
This commit is contained in:
parent
1151037f96
commit
0886786bb5
@ -478,6 +478,9 @@ SET(MULTIMC_SOURCES
|
|||||||
logic/OneSixInstance.h
|
logic/OneSixInstance.h
|
||||||
logic/OneSixInstance.cpp
|
logic/OneSixInstance.cpp
|
||||||
|
|
||||||
|
# a smart pointer wrapper intended for safer use with Qt signal/slot mechanisms
|
||||||
|
logic/QObjectPtr.h
|
||||||
|
|
||||||
# Common utils for instances
|
# Common utils for instances
|
||||||
logic/JarUtils.h
|
logic/JarUtils.h
|
||||||
logic/JarUtils.cpp
|
logic/JarUtils.cpp
|
||||||
|
@ -297,7 +297,9 @@ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent), ui(new Ui::MainWi
|
|||||||
connect(job, SIGNAL(succeeded()), SLOT(skinJobFinished()));
|
connect(job, SIGNAL(succeeded()), SLOT(skinJobFinished()));
|
||||||
connect(job, SIGNAL(failed()), SLOT(skinJobFinished()));
|
connect(job, SIGNAL(failed()), SLOT(skinJobFinished()));
|
||||||
for (auto action : skin_dls)
|
for (auto action : skin_dls)
|
||||||
|
{
|
||||||
job->addNetAction(action);
|
job->addNetAction(action);
|
||||||
|
}
|
||||||
skin_download_job.reset(job);
|
skin_download_job.reset(job);
|
||||||
job->start();
|
job->start();
|
||||||
}
|
}
|
||||||
|
@ -201,6 +201,7 @@ void OneSixUpdate::jarlibStart()
|
|||||||
auto entry = metacache->resolveEntry("versions", localPath);
|
auto entry = metacache->resolveEntry("versions", localPath);
|
||||||
job->addNetAction(CacheDownload::make(QUrl(urlstr), entry));
|
job->addNetAction(CacheDownload::make(QUrl(urlstr), entry));
|
||||||
jarHashOnEntry = entry->md5sum;
|
jarHashOnEntry = entry->md5sum;
|
||||||
|
|
||||||
jarlibDownloadJob.reset(job);
|
jarlibDownloadJob.reset(job);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
52
logic/QObjectPtr.h
Normal file
52
logic/QObjectPtr.h
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A pointer class with the usual shared pointer semantics intended for derivates of QObject
|
||||||
|
* Calls deleteLater() instead of destroying the contained object immediately
|
||||||
|
*/
|
||||||
|
template <typename T>
|
||||||
|
class QObjectPtr
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
QObjectPtr(){}
|
||||||
|
QObjectPtr(T * wrap)
|
||||||
|
{
|
||||||
|
reset(wrap);
|
||||||
|
}
|
||||||
|
QObjectPtr(const QObjectPtr<T>& other)
|
||||||
|
{
|
||||||
|
m_ptr = other.m_ptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
void reset(T * wrap)
|
||||||
|
{
|
||||||
|
using namespace std::placeholders;
|
||||||
|
m_ptr.reset(wrap, std::bind(&QObject::deleteLater, _1));
|
||||||
|
}
|
||||||
|
void reset()
|
||||||
|
{
|
||||||
|
m_ptr.reset();
|
||||||
|
}
|
||||||
|
T * get() const
|
||||||
|
{
|
||||||
|
return m_ptr.get();
|
||||||
|
}
|
||||||
|
T * operator->() const
|
||||||
|
{
|
||||||
|
return m_ptr.get();
|
||||||
|
}
|
||||||
|
T & operator*() const
|
||||||
|
{
|
||||||
|
return *m_ptr.get();
|
||||||
|
}
|
||||||
|
operator bool() const
|
||||||
|
{
|
||||||
|
return m_ptr.get() != nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::shared_ptr <T> m_ptr;
|
||||||
|
};
|
@ -29,7 +29,7 @@ enum JobStatus
|
|||||||
};
|
};
|
||||||
|
|
||||||
typedef std::shared_ptr<class NetAction> NetActionPtr;
|
typedef std::shared_ptr<class NetAction> NetActionPtr;
|
||||||
class NetAction : public QObject
|
class NetAction : public QObject, public std::enable_shared_from_this<NetAction>
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
protected:
|
protected:
|
||||||
@ -51,6 +51,11 @@ public:
|
|||||||
{
|
{
|
||||||
return m_failures;
|
return m_failures;
|
||||||
}
|
}
|
||||||
|
NetActionPtr getSharedPtr()
|
||||||
|
{
|
||||||
|
return shared_from_this();
|
||||||
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
/// the network reply
|
/// the network reply
|
||||||
std::shared_ptr<QNetworkReply> m_reply;
|
std::shared_ptr<QNetworkReply> m_reply;
|
||||||
|
@ -28,46 +28,27 @@ void NetJob::partSucceeded(int index)
|
|||||||
auto &slot = parts_progress[index];
|
auto &slot = parts_progress[index];
|
||||||
partProgress(index, slot.total_progress, slot.total_progress);
|
partProgress(index, slot.total_progress, slot.total_progress);
|
||||||
|
|
||||||
num_succeeded++;
|
m_doing.remove(index);
|
||||||
QLOG_INFO() << m_job_name.toLocal8Bit() << "progress:" << num_succeeded << "/"
|
m_done.insert(index);
|
||||||
<< downloads.size();
|
disconnect(downloads[index].get(), 0, this, 0);
|
||||||
|
startMoreParts();
|
||||||
if (num_failed + num_succeeded == downloads.size())
|
|
||||||
{
|
|
||||||
if (num_failed)
|
|
||||||
{
|
|
||||||
QLOG_ERROR() << m_job_name.toLocal8Bit() << "failed.";
|
|
||||||
emit failed();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
QLOG_INFO() << m_job_name.toLocal8Bit() << "succeeded.";
|
|
||||||
emit succeeded();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void NetJob::partFailed(int index)
|
void NetJob::partFailed(int index)
|
||||||
{
|
{
|
||||||
|
m_doing.remove(index);
|
||||||
auto &slot = parts_progress[index];
|
auto &slot = parts_progress[index];
|
||||||
if (slot.failures == 3)
|
if (slot.failures == 3)
|
||||||
{
|
{
|
||||||
QLOG_ERROR() << "Part" << index << "failed 3 times (" << downloads[index]->m_url << ")";
|
m_failed.insert(index);
|
||||||
num_failed++;
|
|
||||||
if (num_failed + num_succeeded == downloads.size())
|
|
||||||
{
|
|
||||||
QLOG_ERROR() << m_job_name.toLocal8Bit() << "failed.";
|
|
||||||
emit failed();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
QLOG_ERROR() << "Part" << index << "failed, restarting (" << downloads[index]->m_url
|
|
||||||
<< ")";
|
|
||||||
// restart the job
|
|
||||||
slot.failures++;
|
slot.failures++;
|
||||||
downloads[index]->start();
|
m_todo.enqueue(index);
|
||||||
}
|
}
|
||||||
|
disconnect(downloads[index].get(), 0, this, 0);
|
||||||
|
startMoreParts();
|
||||||
}
|
}
|
||||||
|
|
||||||
void NetJob::partProgress(int index, qint64 bytesReceived, qint64 bytesTotal)
|
void NetJob::partProgress(int index, qint64 bytesReceived, qint64 bytesTotal)
|
||||||
@ -88,25 +69,58 @@ void NetJob::start()
|
|||||||
{
|
{
|
||||||
QLOG_INFO() << m_job_name.toLocal8Bit() << " started.";
|
QLOG_INFO() << m_job_name.toLocal8Bit() << " started.";
|
||||||
m_running = true;
|
m_running = true;
|
||||||
for (auto iter : downloads)
|
for (int i = 0; i < downloads.size(); i++)
|
||||||
{
|
{
|
||||||
connect(iter.get(), SIGNAL(succeeded(int)), SLOT(partSucceeded(int)));
|
m_todo.enqueue(i);
|
||||||
connect(iter.get(), SIGNAL(failed(int)), SLOT(partFailed(int)));
|
}
|
||||||
connect(iter.get(), SIGNAL(progress(int, qint64, qint64)),
|
startMoreParts();
|
||||||
|
}
|
||||||
|
|
||||||
|
void NetJob::startMoreParts()
|
||||||
|
{
|
||||||
|
// check for final conditions if there's nothing in the queue
|
||||||
|
if(!m_todo.size())
|
||||||
|
{
|
||||||
|
if(!m_doing.size())
|
||||||
|
{
|
||||||
|
if(!m_failed.size())
|
||||||
|
{
|
||||||
|
QLOG_INFO() << m_job_name.toLocal8Bit() << "succeeded.";
|
||||||
|
emit succeeded();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
QLOG_ERROR() << m_job_name.toLocal8Bit() << "failed.";
|
||||||
|
emit failed();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// otherwise try to start more parts
|
||||||
|
while (m_doing.size() < 6)
|
||||||
|
{
|
||||||
|
if(!m_todo.size())
|
||||||
|
return;
|
||||||
|
int doThis = m_todo.dequeue();
|
||||||
|
m_doing.insert(doThis);
|
||||||
|
auto part = downloads[doThis];
|
||||||
|
// connect signals :D
|
||||||
|
connect(part.get(), SIGNAL(succeeded(int)), SLOT(partSucceeded(int)));
|
||||||
|
connect(part.get(), SIGNAL(failed(int)), SLOT(partFailed(int)));
|
||||||
|
connect(part.get(), SIGNAL(progress(int, qint64, qint64)),
|
||||||
SLOT(partProgress(int, qint64, qint64)));
|
SLOT(partProgress(int, qint64, qint64)));
|
||||||
iter->start();
|
part->start();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
QStringList NetJob::getFailedFiles()
|
QStringList NetJob::getFailedFiles()
|
||||||
{
|
{
|
||||||
QStringList failed;
|
QStringList failed;
|
||||||
for (auto download : downloads)
|
for (auto index: m_failed)
|
||||||
{
|
{
|
||||||
if (download->m_status == Job_Failed)
|
failed.push_back(downloads[index]->m_url.toString());
|
||||||
{
|
|
||||||
failed.push_back(download->m_url.toString());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
failed.sort();
|
||||||
return failed;
|
return failed;
|
||||||
}
|
}
|
||||||
|
@ -22,9 +22,10 @@
|
|||||||
#include "CacheDownload.h"
|
#include "CacheDownload.h"
|
||||||
#include "HttpMetaCache.h"
|
#include "HttpMetaCache.h"
|
||||||
#include "logic/tasks/ProgressProvider.h"
|
#include "logic/tasks/ProgressProvider.h"
|
||||||
|
#include "logic/QObjectPtr.h"
|
||||||
|
|
||||||
class NetJob;
|
class NetJob;
|
||||||
typedef std::shared_ptr<NetJob> NetJobPtr;
|
typedef QObjectPtr<NetJob> NetJobPtr;
|
||||||
|
|
||||||
class NetJob : public ProgressProvider
|
class NetJob : public ProgressProvider
|
||||||
{
|
{
|
||||||
@ -81,18 +82,22 @@ public:
|
|||||||
return m_running;
|
return m_running;
|
||||||
}
|
}
|
||||||
QStringList getFailedFiles();
|
QStringList getFailedFiles();
|
||||||
|
|
||||||
|
private:
|
||||||
|
void startMoreParts();
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void started();
|
void started();
|
||||||
void progress(qint64 current, qint64 total);
|
void progress(qint64 current, qint64 total);
|
||||||
void succeeded();
|
void succeeded();
|
||||||
void failed();
|
void failed();
|
||||||
public
|
|
||||||
slots:
|
public slots:
|
||||||
virtual void start();
|
virtual void start();
|
||||||
// FIXME: implement
|
// FIXME: implement
|
||||||
virtual void abort() {};
|
virtual void abort() {};
|
||||||
private
|
|
||||||
slots:
|
private slots:
|
||||||
void partProgress(int index, qint64 bytesReceived, qint64 bytesTotal);
|
void partProgress(int index, qint64 bytesReceived, qint64 bytesTotal);
|
||||||
void partSucceeded(int index);
|
void partSucceeded(int index);
|
||||||
void partFailed(int index);
|
void partFailed(int index);
|
||||||
@ -107,9 +112,11 @@ private:
|
|||||||
QString m_job_name;
|
QString m_job_name;
|
||||||
QList<NetActionPtr> downloads;
|
QList<NetActionPtr> downloads;
|
||||||
QList<part_info> parts_progress;
|
QList<part_info> parts_progress;
|
||||||
|
QQueue<int> m_todo;
|
||||||
|
QSet<int> m_doing;
|
||||||
|
QSet<int> m_done;
|
||||||
|
QSet<int> m_failed;
|
||||||
qint64 current_progress = 0;
|
qint64 current_progress = 0;
|
||||||
qint64 total_progress = 0;
|
qint64 total_progress = 0;
|
||||||
int num_succeeded = 0;
|
|
||||||
int num_failed = 0;
|
|
||||||
bool m_running = false;
|
bool m_running = false;
|
||||||
};
|
};
|
||||||
|
Loading…
x
Reference in New Issue
Block a user