forked from Public/monero-gui
qt: implement FutureScheduler, always await async code to complete
This commit is contained in:
89
src/qt/FutureScheduler.cpp
Normal file
89
src/qt/FutureScheduler.cpp
Normal file
@@ -0,0 +1,89 @@
|
||||
#include "FutureScheduler.h"
|
||||
|
||||
FutureScheduler::FutureScheduler(QObject *parent)
|
||||
: QObject(parent), Alive(0), Stopping(false)
|
||||
{
|
||||
}
|
||||
|
||||
FutureScheduler::~FutureScheduler()
|
||||
{
|
||||
shutdownWaitForFinished();
|
||||
}
|
||||
|
||||
void FutureScheduler::shutdownWaitForFinished() noexcept
|
||||
{
|
||||
QMutexLocker locker(&Mutex);
|
||||
|
||||
Stopping = true;
|
||||
while (Alive > 0)
|
||||
{
|
||||
Condition.wait(&Mutex);
|
||||
}
|
||||
}
|
||||
|
||||
QPair<bool, QFuture<void>> FutureScheduler::run(std::function<void()> function) noexcept
|
||||
{
|
||||
return execute<void>([this, function](QFutureWatcher<void> *) {
|
||||
return QtConcurrent::run([this, function] {
|
||||
try
|
||||
{
|
||||
function();
|
||||
}
|
||||
catch (const std::exception &exception)
|
||||
{
|
||||
qWarning() << "Exception thrown from async function: " << exception.what();
|
||||
}
|
||||
done();
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
QPair<bool, QFuture<QJSValueList>> FutureScheduler::run(std::function<QJSValueList() noexcept> function, const QJSValue &callback) noexcept
|
||||
{
|
||||
if (!callback.isCallable())
|
||||
{
|
||||
throw std::runtime_error("js callback must be callable");
|
||||
}
|
||||
|
||||
return execute<QJSValueList>([this, function, callback](QFutureWatcher<QJSValueList> *watcher) {
|
||||
connect(watcher, &QFutureWatcher<QJSValueList>::finished, [watcher, callback] {
|
||||
QJSValue(callback).call(watcher->future().result());
|
||||
});
|
||||
return QtConcurrent::run([this, function] {
|
||||
QJSValueList result;
|
||||
try
|
||||
{
|
||||
result = function();
|
||||
}
|
||||
catch (const std::exception &exception)
|
||||
{
|
||||
qWarning() << "Exception thrown from async function: " << exception.what();
|
||||
}
|
||||
done();
|
||||
return result;
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
bool FutureScheduler::add() noexcept
|
||||
{
|
||||
QMutexLocker locker(&Mutex);
|
||||
|
||||
if (Stopping)
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
++Alive;
|
||||
return true;
|
||||
}
|
||||
|
||||
void FutureScheduler::done() noexcept
|
||||
{
|
||||
{
|
||||
QMutexLocker locker(&Mutex);
|
||||
--Alive;
|
||||
}
|
||||
|
||||
Condition.wakeAll();
|
||||
}
|
||||
79
src/qt/FutureScheduler.h
Normal file
79
src/qt/FutureScheduler.h
Normal file
@@ -0,0 +1,79 @@
|
||||
#ifndef FUTURE_SCHEDULER_H
|
||||
#define FUTURE_SCHEDULER_H
|
||||
|
||||
#include <functional>
|
||||
|
||||
#include <QtConcurrent/QtConcurrent>
|
||||
#include <QFuture>
|
||||
#include <QJSValue>
|
||||
#include <QMutex>
|
||||
#include <QMutexLocker>
|
||||
#include <QPair>
|
||||
#include <QWaitCondition>
|
||||
|
||||
class FutureScheduler : public QObject
|
||||
{
|
||||
Q_OBJECT
|
||||
|
||||
public:
|
||||
FutureScheduler(QObject *parent);
|
||||
~FutureScheduler();
|
||||
|
||||
void shutdownWaitForFinished() noexcept;
|
||||
|
||||
QPair<bool, QFuture<void>> run(std::function<void()> function) noexcept;
|
||||
QPair<bool, QFuture<QJSValueList>> run(std::function<QJSValueList() noexcept> function, const QJSValue &callback) noexcept;
|
||||
|
||||
private:
|
||||
bool add() noexcept;
|
||||
void done() noexcept;
|
||||
|
||||
template<typename T>
|
||||
QFutureWatcher<T> *newWatcher()
|
||||
{
|
||||
QFutureWatcher<T> *watcher = new QFutureWatcher<T>();
|
||||
QThread *schedulerThread = this->thread();
|
||||
if (watcher->thread() != schedulerThread)
|
||||
{
|
||||
watcher->moveToThread(schedulerThread);
|
||||
}
|
||||
watcher->setParent(this);
|
||||
|
||||
return watcher;
|
||||
}
|
||||
|
||||
template<typename T>
|
||||
QPair<bool, QFuture<T>> execute(std::function<QFuture<T>(QFutureWatcher<T> *)> makeFuture) noexcept
|
||||
{
|
||||
if (add())
|
||||
{
|
||||
try
|
||||
{
|
||||
auto *watcher = newWatcher<T>();
|
||||
watcher->setFuture(makeFuture(watcher));
|
||||
connect(watcher, &QFutureWatcher<T>::finished, [this, watcher] {
|
||||
watcher->deleteLater();
|
||||
});
|
||||
return qMakePair(true, watcher->future());
|
||||
}
|
||||
catch (const std::exception &exception)
|
||||
{
|
||||
qCritical() << "Failed to schedule async function: " << exception.what();
|
||||
done();
|
||||
}
|
||||
}
|
||||
|
||||
return qMakePair(false, QFuture<T>());
|
||||
}
|
||||
|
||||
QFutureWatcher<void> schedule(std::function<void()> function);
|
||||
QFutureWatcher<QJSValueList> schedule(std::function<QJSValueList() noexcept> function, const QJSValue &callback);
|
||||
|
||||
private:
|
||||
size_t Alive;
|
||||
QWaitCondition Condition;
|
||||
QMutex Mutex;
|
||||
bool Stopping;
|
||||
};
|
||||
|
||||
#endif // FUTURE_SCHEDULER_H
|
||||
Reference in New Issue
Block a user