2015-08-21 17:38:54 +01:00
|
|
|
#include "responsewaiter.h"
|
2014-01-28 22:22:59 +00:00
|
|
|
|
|
|
|
#include <QDBusPendingCall>
|
|
|
|
#include <QDBusPendingReply>
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QCoreApplication>
|
|
|
|
|
|
|
|
Q_DECLARE_METATYPE(QDBusPendingReply<>)
|
|
|
|
Q_DECLARE_METATYPE(QDBusPendingReply<QVariant>)
|
|
|
|
Q_DECLARE_METATYPE(QDBusPendingReply<bool>)
|
|
|
|
Q_DECLARE_METATYPE(QDBusPendingReply<int>)
|
|
|
|
Q_DECLARE_METATYPE(QDBusPendingReply<QString>)
|
|
|
|
|
2015-09-08 09:46:59 +01:00
|
|
|
DBusResponseWaiter* DBusResponseWaiter::m_instance = nullptr;
|
2014-01-31 10:06:21 +00:00
|
|
|
|
|
|
|
DBusResponseWaiter* DBusResponseWaiter::instance()
|
|
|
|
{
|
|
|
|
if (!m_instance)
|
|
|
|
{
|
|
|
|
m_instance = new DBusResponseWaiter();
|
|
|
|
}
|
|
|
|
return m_instance;
|
|
|
|
}
|
|
|
|
|
2014-01-28 22:22:59 +00:00
|
|
|
DBusResponseWaiter::DBusResponseWaiter()
|
|
|
|
: QObject()
|
|
|
|
{
|
|
|
|
m_registered
|
|
|
|
<< qRegisterMetaType<QDBusPendingReply<> >("QDBusPendingReply<>")
|
|
|
|
<< qRegisterMetaType<QDBusPendingReply<QVariant> >("QDBusPendingReply<QVariant>")
|
|
|
|
<< qRegisterMetaType<QDBusPendingReply<bool> >("QDBusPendingReply<bool>")
|
|
|
|
<< qRegisterMetaType<QDBusPendingReply<int> >("QDBusPendingReply<int>")
|
|
|
|
<< qRegisterMetaType<QDBusPendingReply<QString> >("QDBusPendingReply<QString>")
|
|
|
|
;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant DBusResponseWaiter::waitForReply(QVariant variant) const
|
|
|
|
{
|
|
|
|
if (QDBusPendingCall* call = const_cast<QDBusPendingCall*>(extractPendingCall(variant)))
|
|
|
|
{
|
|
|
|
call->waitForFinished();
|
2014-02-05 20:36:51 +00:00
|
|
|
|
|
|
|
if (call->isError())
|
|
|
|
{
|
2016-06-12 21:26:32 +01:00
|
|
|
qWarning() << "error:" << call->error();
|
2014-02-05 20:36:51 +00:00
|
|
|
return QVariant("error");
|
|
|
|
}
|
|
|
|
|
2014-01-28 22:22:59 +00:00
|
|
|
QDBusMessage reply = call->reply();
|
|
|
|
|
|
|
|
if (reply.arguments().count() > 0)
|
|
|
|
{
|
2015-09-12 08:53:05 +01:00
|
|
|
return reply.arguments().at(0);
|
2014-01-28 22:22:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
2014-02-06 15:25:22 +00:00
|
|
|
DBusAsyncResponse::DBusAsyncResponse(QObject* parent)
|
|
|
|
: QObject(parent)
|
|
|
|
, m_autodelete(false)
|
|
|
|
{
|
|
|
|
m_timeout.setSingleShot(true);
|
|
|
|
m_timeout.setInterval(15000);
|
2016-11-26 14:12:38 +00:00
|
|
|
connect(&m_timeout, &QTimer::timeout, this, &DBusAsyncResponse::onTimeout);
|
2014-02-06 15:25:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-31 10:06:21 +00:00
|
|
|
void DBusAsyncResponse::setPendingCall(QVariant variant)
|
2014-01-28 22:22:59 +00:00
|
|
|
{
|
2014-01-31 10:06:21 +00:00
|
|
|
if (QDBusPendingCall* call = const_cast<QDBusPendingCall*>(DBusResponseWaiter::instance()->extractPendingCall(variant)))
|
2014-01-28 22:22:59 +00:00
|
|
|
{
|
|
|
|
QDBusPendingCallWatcher* watcher = new QDBusPendingCallWatcher(*call);
|
2014-02-06 15:25:22 +00:00
|
|
|
watcher->setProperty("pengingCallVariant", variant);
|
2016-11-26 14:12:38 +00:00
|
|
|
connect(watcher, &QDBusPendingCallWatcher::finished, this, &DBusAsyncResponse::onCallFinished);
|
|
|
|
connect(watcher, &QDBusPendingCallWatcher::finished, watcher, &QObject::deleteLater);
|
|
|
|
connect(&m_timeout, &QTimer::timeout, watcher, &QObject::deleteLater);
|
2014-02-06 15:25:22 +00:00
|
|
|
m_timeout.start();
|
2016-06-15 22:49:37 +01:00
|
|
|
}
|
2014-01-28 22:22:59 +00:00
|
|
|
}
|
|
|
|
|
2014-01-31 10:06:21 +00:00
|
|
|
void DBusAsyncResponse::onCallFinished(QDBusPendingCallWatcher* watcher)
|
2014-01-28 22:22:59 +00:00
|
|
|
{
|
2014-02-06 15:25:22 +00:00
|
|
|
m_timeout.stop();
|
|
|
|
QVariant variant = watcher->property("pengingCallVariant");
|
2014-01-28 22:22:59 +00:00
|
|
|
|
2014-01-31 10:06:21 +00:00
|
|
|
if (QDBusPendingCall* call = const_cast<QDBusPendingCall*>(DBusResponseWaiter::instance()->extractPendingCall(variant)))
|
2014-01-28 22:22:59 +00:00
|
|
|
{
|
|
|
|
if (call->isError())
|
|
|
|
{
|
2014-01-30 16:25:22 +00:00
|
|
|
Q_EMIT error(call->error().message());
|
2014-01-28 22:22:59 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
QDBusMessage reply = call->reply();
|
|
|
|
|
|
|
|
if (reply.arguments().count() > 0)
|
|
|
|
{
|
2015-09-12 08:53:05 +01:00
|
|
|
Q_EMIT success(reply.arguments().at(0));
|
2014-01-28 22:22:59 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-01-30 16:25:22 +00:00
|
|
|
Q_EMIT success(QVariant());
|
2014-01-28 22:22:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-02-05 20:36:51 +00:00
|
|
|
if (m_autodelete)
|
|
|
|
{
|
|
|
|
deleteLater();
|
|
|
|
}
|
2014-01-28 22:22:59 +00:00
|
|
|
}
|
|
|
|
|
2014-02-06 15:25:22 +00:00
|
|
|
void DBusAsyncResponse::onTimeout()
|
|
|
|
{
|
2016-11-26 14:38:08 +00:00
|
|
|
Q_EMIT error(QStringLiteral("timeout when waiting dbus response!"));
|
2014-02-06 15:25:22 +00:00
|
|
|
}
|
|
|
|
|
2014-01-28 22:22:59 +00:00
|
|
|
const QDBusPendingCall* DBusResponseWaiter::extractPendingCall(QVariant& variant) const
|
|
|
|
{
|
2017-07-20 15:14:07 +01:00
|
|
|
for (int type : qAsConst(m_registered))
|
2014-01-28 22:22:59 +00:00
|
|
|
{
|
|
|
|
if (variant.canConvert(QVariant::Type(type)))
|
|
|
|
{
|
|
|
|
return reinterpret_cast<const QDBusPendingCall*>(variant.constData());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-08 09:46:59 +01:00
|
|
|
return nullptr;
|
2014-01-28 22:22:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|