You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
440 lines
17 KiB
C++
440 lines
17 KiB
C++
// -*- c++ -*-
|
|
|
|
/*!
|
|
*
|
|
* Copyright (C) 2015 Jolla Ltd.
|
|
*
|
|
* Contact: Valerio Valerio <valerio.valerio@jolla.com>
|
|
* Author: Andres Gomez <andres.gomez@jolla.com>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
|
|
#include "dbusextendedpendingcallwatcher_p.h"
|
|
|
|
#include "dbusextendedabstractinterface.h"
|
|
|
|
#include <QtDBus/QDBusMetaType>
|
|
#include <QtDBus/QDBusMessage>
|
|
#include <QtDBus/QDBusPendingCall>
|
|
#include <QtDBus/QDBusPendingCallWatcher>
|
|
#include <QtDBus/QDBusPendingReply>
|
|
|
|
#include <QtCore/QDebug>
|
|
#include <QtCore/QMetaProperty>
|
|
|
|
|
|
Q_GLOBAL_STATIC_WITH_ARGS(QByteArray, dBusPropertiesInterface, ("org.freedesktop.DBus.Properties"))
|
|
Q_GLOBAL_STATIC_WITH_ARGS(QByteArray, dBusPropertiesChangedSignal, ("PropertiesChanged"))
|
|
Q_GLOBAL_STATIC_WITH_ARGS(QByteArray, propertyChangedSignature, ("propertyChanged(QString,QVariant)"))
|
|
Q_GLOBAL_STATIC_WITH_ARGS(QByteArray, propertyInvalidatedSignature, ("propertyInvalidated(QString)"))
|
|
|
|
|
|
DBusExtendedAbstractInterface::DBusExtendedAbstractInterface(const QString &service, const QString &path, const char *interface, const QDBusConnection &connection, QObject *parent)
|
|
: QDBusAbstractInterface(service, path, interface, connection, parent)
|
|
, m_sync(false)
|
|
, m_useCache(false)
|
|
, m_getAllPendingCallWatcher(0)
|
|
, m_propertiesChangedConnected(false)
|
|
{
|
|
}
|
|
|
|
DBusExtendedAbstractInterface::~DBusExtendedAbstractInterface()
|
|
{
|
|
}
|
|
|
|
void DBusExtendedAbstractInterface::getAllProperties()
|
|
{
|
|
m_lastExtendedError = QDBusError();
|
|
|
|
if (!isValid()) {
|
|
QString errorMessage = QStringLiteral("This Extended DBus interface is not valid yet.");
|
|
m_lastExtendedError = QDBusMessage::createError(QDBusError::Failed, errorMessage);
|
|
qDebug() << Q_FUNC_INFO << errorMessage;
|
|
return;
|
|
}
|
|
|
|
if (!m_sync && m_getAllPendingCallWatcher) {
|
|
// Call already in place, not repeating ...
|
|
return;
|
|
}
|
|
|
|
QDBusMessage msg = QDBusMessage::createMethodCall(service(), path(), *dBusPropertiesInterface(), QStringLiteral("GetAll"));
|
|
msg << interface();
|
|
|
|
if (m_sync) {
|
|
QDBusMessage reply = connection().call(msg);
|
|
|
|
if (reply.type() != QDBusMessage::ReplyMessage) {
|
|
m_lastExtendedError = QDBusError(reply);
|
|
qWarning() << Q_FUNC_INFO << m_lastExtendedError.message();
|
|
return;
|
|
}
|
|
|
|
if (reply.signature() != QLatin1String("a{sv}")) {
|
|
QString errorMessage = QStringLiteral("Invalid signature \"%1\" in return from call to %2")
|
|
.arg(reply.signature(),
|
|
QString(*dBusPropertiesInterface()));
|
|
qWarning() << Q_FUNC_INFO << errorMessage;
|
|
m_lastExtendedError = QDBusError(QDBusError::InvalidSignature, errorMessage);
|
|
return;
|
|
}
|
|
|
|
QVariantMap value = reply.arguments().at(0).toMap();
|
|
onPropertiesChanged(interface(), value, QStringList());
|
|
} else {
|
|
QDBusPendingReply<QVariantMap> async = connection().asyncCall(msg);
|
|
m_getAllPendingCallWatcher = new QDBusPendingCallWatcher(async, this);
|
|
|
|
connect(m_getAllPendingCallWatcher, SIGNAL(finished(QDBusPendingCallWatcher*)), this, SLOT(onAsyncGetAllPropertiesFinished(QDBusPendingCallWatcher*)));
|
|
return;
|
|
}
|
|
}
|
|
|
|
void DBusExtendedAbstractInterface::connectNotify(const QMetaMethod &signal)
|
|
{
|
|
if (signal.methodType() == QMetaMethod::Signal
|
|
&& (signal.methodSignature() == *propertyChangedSignature()
|
|
|| signal.methodSignature() == *propertyInvalidatedSignature())) {
|
|
if (!m_propertiesChangedConnected) {
|
|
QStringList argumentMatch;
|
|
argumentMatch << interface();
|
|
connection().connect(service(), path(), *dBusPropertiesInterface(), *dBusPropertiesChangedSignal(),
|
|
argumentMatch, QString(),
|
|
this, SLOT(onPropertiesChanged(QString, QVariantMap, QStringList)));
|
|
|
|
m_propertiesChangedConnected = true;
|
|
return;
|
|
}
|
|
} else {
|
|
QDBusAbstractInterface::connectNotify(signal);
|
|
}
|
|
}
|
|
|
|
void DBusExtendedAbstractInterface::disconnectNotify(const QMetaMethod &signal)
|
|
{
|
|
if (signal.methodType() == QMetaMethod::Signal
|
|
&& (signal.methodSignature() == *propertyChangedSignature()
|
|
|| signal.methodSignature() == *propertyInvalidatedSignature())) {
|
|
if (m_propertiesChangedConnected
|
|
&& 0 == receivers(propertyChangedSignature()->constData())
|
|
&& 0 == receivers(propertyInvalidatedSignature()->constData())) {
|
|
QStringList argumentMatch;
|
|
argumentMatch << interface();
|
|
connection().disconnect(service(), path(), *dBusPropertiesInterface(), *dBusPropertiesChangedSignal(),
|
|
argumentMatch, QString(),
|
|
this, SLOT(onPropertiesChanged(QString, QVariantMap, QStringList)));
|
|
|
|
m_propertiesChangedConnected = false;
|
|
return;
|
|
}
|
|
} else {
|
|
QDBusAbstractInterface::disconnectNotify(signal);
|
|
}
|
|
}
|
|
|
|
QVariant DBusExtendedAbstractInterface::internalPropGet(const char *propname, void *propertyPtr)
|
|
{
|
|
m_lastExtendedError = QDBusError();
|
|
|
|
if (m_useCache) {
|
|
int propertyIndex = metaObject()->indexOfProperty(propname);
|
|
QMetaProperty metaProperty = metaObject()->property(propertyIndex);
|
|
return QVariant(metaProperty.type(), propertyPtr);
|
|
}
|
|
|
|
if (m_sync) {
|
|
return property(propname);
|
|
} else {
|
|
if (!isValid()) {
|
|
QString errorMessage = QStringLiteral("This Extended DBus interface is not valid yet.");
|
|
m_lastExtendedError = QDBusMessage::createError(QDBusError::Failed, errorMessage);
|
|
qDebug() << Q_FUNC_INFO << errorMessage;
|
|
return QVariant();
|
|
}
|
|
|
|
int propertyIndex = metaObject()->indexOfProperty(propname);
|
|
|
|
if (-1 == propertyIndex) {
|
|
QString errorMessage = QStringLiteral("Got unknown property \"%1\" to read")
|
|
.arg(QString::fromLatin1(propname));
|
|
m_lastExtendedError = QDBusMessage::createError(QDBusError::Failed, errorMessage);
|
|
qWarning() << Q_FUNC_INFO << errorMessage;
|
|
return QVariant();
|
|
}
|
|
|
|
QMetaProperty metaProperty = metaObject()->property(propertyIndex);
|
|
|
|
if (!metaProperty.isReadable()) {
|
|
QString errorMessage = QStringLiteral("Property \"%1\" is NOT readable")
|
|
.arg(QString::fromLatin1(propname));
|
|
m_lastExtendedError = QDBusMessage::createError(QDBusError::Failed, errorMessage);
|
|
qWarning() << Q_FUNC_INFO << errorMessage;
|
|
return QVariant();
|
|
}
|
|
|
|
// is this metatype registered?
|
|
const char *expectedSignature = "";
|
|
if (int(metaProperty.type()) != QMetaType::QVariant) {
|
|
expectedSignature = QDBusMetaType::typeToSignature(metaProperty.userType());
|
|
if (0 == expectedSignature) {
|
|
QString errorMessage =
|
|
QStringLiteral("Type %1 must be registered with Qt D-Bus "
|
|
"before it can be used to read property "
|
|
"%2.%3")
|
|
.arg(metaProperty.typeName(),
|
|
interface(),
|
|
propname);
|
|
m_lastExtendedError = QDBusMessage::createError(QDBusError::Failed, errorMessage);
|
|
qWarning() << Q_FUNC_INFO << errorMessage;
|
|
return QVariant();
|
|
}
|
|
}
|
|
|
|
asyncProperty(propname);
|
|
return QVariant(metaProperty.type(), propertyPtr);
|
|
}
|
|
}
|
|
|
|
void DBusExtendedAbstractInterface::internalPropSet(const char *propname, const QVariant &value, void *propertyPtr)
|
|
{
|
|
m_lastExtendedError = QDBusError();
|
|
|
|
if (m_sync) {
|
|
setProperty(propname, value);
|
|
} else {
|
|
if (!isValid()) {
|
|
QString errorMessage = QStringLiteral("This interface is not yet valid");
|
|
m_lastExtendedError = QDBusMessage::createError(QDBusError::Failed, errorMessage);
|
|
qDebug() << Q_FUNC_INFO << errorMessage;
|
|
return;
|
|
}
|
|
|
|
int propertyIndex = metaObject()->indexOfProperty(propname);
|
|
|
|
if (-1 == propertyIndex) {
|
|
QString errorMessage = QStringLiteral("Got unknown property \"%1\" to write")
|
|
.arg(QString::fromLatin1(propname));
|
|
m_lastExtendedError = QDBusMessage::createError(QDBusError::Failed, errorMessage);
|
|
qWarning() << Q_FUNC_INFO << errorMessage;
|
|
return;
|
|
}
|
|
|
|
QMetaProperty metaProperty = metaObject()->property(propertyIndex);
|
|
|
|
if (!metaProperty.isWritable()) {
|
|
QString errorMessage = QStringLiteral("Property \"%1\" is NOT writable")
|
|
.arg(QString::fromLatin1(propname));
|
|
m_lastExtendedError = QDBusMessage::createError(QDBusError::Failed, errorMessage);
|
|
qWarning() << Q_FUNC_INFO << errorMessage;
|
|
return;
|
|
}
|
|
|
|
asyncSetProperty(propname, QVariant(metaProperty.type(), propertyPtr));
|
|
}
|
|
}
|
|
|
|
QVariant DBusExtendedAbstractInterface::asyncProperty(const QString &propertyName)
|
|
{
|
|
QDBusMessage msg = QDBusMessage::createMethodCall(service(), path(), *dBusPropertiesInterface(), QStringLiteral("Get"));
|
|
msg << interface() << propertyName;
|
|
QDBusPendingReply<QVariant> async = connection().asyncCall(msg);
|
|
DBusExtendedPendingCallWatcher *watcher = new DBusExtendedPendingCallWatcher(async, propertyName, QVariant(), this);
|
|
|
|
connect(watcher, SIGNAL(finished(QDBusPendingCallWatcher*)), this, SLOT(onAsyncPropertyFinished(QDBusPendingCallWatcher*)));
|
|
|
|
return QVariant();
|
|
}
|
|
|
|
void DBusExtendedAbstractInterface::asyncSetProperty(const QString &propertyName, const QVariant &value)
|
|
{
|
|
QDBusMessage msg = QDBusMessage::createMethodCall(service(), path(), *dBusPropertiesInterface(), QStringLiteral("Set"));
|
|
msg << interface() << propertyName << value;
|
|
QDBusPendingReply<QVariant> async = connection().asyncCall(msg);
|
|
DBusExtendedPendingCallWatcher *watcher = new DBusExtendedPendingCallWatcher(async, propertyName, value, this);
|
|
|
|
connect(watcher, SIGNAL(finished(QDBusPendingCallWatcher*)), this, SLOT(onAsyncSetPropertyFinished(QDBusPendingCallWatcher*)));
|
|
}
|
|
|
|
void DBusExtendedAbstractInterface::onAsyncPropertyFinished(DBusExtendedPendingCallWatcher *watcher)
|
|
{
|
|
QDBusPendingReply<QVariant> reply = *watcher;
|
|
|
|
if (reply.isError()) {
|
|
m_lastExtendedError = reply.error();
|
|
} else {
|
|
int propertyIndex = metaObject()->indexOfProperty(watcher->asyncProperty().toLatin1().constData());
|
|
QVariant value = demarshall(interface(),
|
|
metaObject()->property(propertyIndex),
|
|
reply.value(),
|
|
&m_lastExtendedError);
|
|
|
|
if (m_lastExtendedError.isValid()) {
|
|
emit propertyInvalidated(watcher->asyncProperty());
|
|
} else {
|
|
emit propertyChanged(watcher->asyncProperty(), value);
|
|
}
|
|
}
|
|
|
|
emit asyncPropertyFinished(watcher->asyncProperty());
|
|
watcher->deleteLater();
|
|
}
|
|
|
|
void DBusExtendedAbstractInterface::onAsyncSetPropertyFinished(DBusExtendedPendingCallWatcher *watcher)
|
|
{
|
|
QDBusPendingReply<QVariant> reply = *watcher;
|
|
|
|
if (reply.isError()) {
|
|
m_lastExtendedError = reply.error();
|
|
} else {
|
|
m_lastExtendedError = QDBusError();
|
|
}
|
|
|
|
emit asyncSetPropertyFinished(watcher->asyncProperty());
|
|
|
|
// Resetting the property to its previous value after sending the
|
|
// finished signal
|
|
if (reply.isError()) {
|
|
m_lastExtendedError = QDBusError();
|
|
emit propertyChanged(watcher->asyncProperty(), watcher->previousValue());
|
|
}
|
|
|
|
watcher->deleteLater();
|
|
}
|
|
|
|
void DBusExtendedAbstractInterface::onAsyncGetAllPropertiesFinished(QDBusPendingCallWatcher *watcher)
|
|
{
|
|
m_getAllPendingCallWatcher = 0;
|
|
|
|
QDBusPendingReply<QVariantMap> reply = *watcher;
|
|
|
|
if (reply.isError()) {
|
|
m_lastExtendedError = reply.error();
|
|
} else {
|
|
m_lastExtendedError = QDBusError();
|
|
}
|
|
|
|
emit asyncGetAllPropertiesFinished();
|
|
|
|
if (!reply.isError()) {
|
|
onPropertiesChanged(interface(), reply.value(), QStringList());
|
|
}
|
|
|
|
watcher->deleteLater();
|
|
}
|
|
|
|
void DBusExtendedAbstractInterface::onPropertiesChanged(const QString& interfaceName,
|
|
const QVariantMap& changedProperties,
|
|
const QStringList& invalidatedProperties)
|
|
{
|
|
if (interfaceName == interface()) {
|
|
QVariantMap::const_iterator i = changedProperties.constBegin();
|
|
while (i != changedProperties.constEnd()) {
|
|
int propertyIndex = metaObject()->indexOfProperty(i.key().toLatin1().constData());
|
|
|
|
if (-1 == propertyIndex) {
|
|
qDebug() << Q_FUNC_INFO << "Got unknown changed property" << i.key();
|
|
} else {
|
|
QVariant value = demarshall(interface(), metaObject()->property(propertyIndex), i.value(), &m_lastExtendedError);
|
|
|
|
if (m_lastExtendedError.isValid()) {
|
|
emit propertyInvalidated(i.key());
|
|
} else {
|
|
emit propertyChanged(i.key(), value);
|
|
}
|
|
}
|
|
|
|
++i;
|
|
}
|
|
|
|
QStringList::const_iterator j = invalidatedProperties.constBegin();
|
|
while (j != invalidatedProperties.constEnd()) {
|
|
if (-1 == metaObject()->indexOfProperty(j->toLatin1().constData())) {
|
|
qDebug() << Q_FUNC_INFO << "Got unknown invalidated property" << *j;
|
|
} else {
|
|
m_lastExtendedError = QDBusError();
|
|
emit propertyInvalidated(*j);
|
|
}
|
|
|
|
++j;
|
|
}
|
|
}
|
|
}
|
|
|
|
QVariant DBusExtendedAbstractInterface::demarshall(const QString &interface, const QMetaProperty &metaProperty, const QVariant &value, QDBusError *error)
|
|
{
|
|
Q_ASSERT(metaProperty.isValid());
|
|
Q_ASSERT(error != 0);
|
|
|
|
if (value.userType() == metaProperty.userType()) {
|
|
// No need demarshalling. Passing back straight away ...
|
|
*error = QDBusError();
|
|
return value;
|
|
}
|
|
|
|
QVariant result = QVariant(metaProperty.userType(), (void*)0);
|
|
QString errorMessage;
|
|
const char *expectedSignature = QDBusMetaType::typeToSignature(metaProperty.userType());
|
|
|
|
if (value.userType() == qMetaTypeId<QDBusArgument>()) {
|
|
// demarshalling a DBus argument ...
|
|
QDBusArgument dbusArg = value.value<QDBusArgument>();
|
|
|
|
if (expectedSignature == dbusArg.currentSignature().toLatin1()) {
|
|
QDBusMetaType::demarshall(dbusArg, metaProperty.userType(), result.data());
|
|
if (!result.isValid()) {
|
|
errorMessage = QStringLiteral("Unexpected failure demarshalling "
|
|
"upon PropertiesChanged signal arrival "
|
|
"for property `%3.%4' (expected type `%5' (%6))")
|
|
.arg(interface,
|
|
QString::fromLatin1(metaProperty.name()),
|
|
QString::fromLatin1(metaProperty.typeName()),
|
|
expectedSignature);
|
|
}
|
|
} else {
|
|
errorMessage = QStringLiteral("Unexpected `user type' (%2) "
|
|
"upon PropertiesChanged signal arrival "
|
|
"for property `%3.%4' (expected type `%5' (%6))")
|
|
.arg(dbusArg.currentSignature(),
|
|
interface,
|
|
QString::fromLatin1(metaProperty.name()),
|
|
QString::fromLatin1(metaProperty.typeName()),
|
|
QString::fromLatin1(expectedSignature));
|
|
}
|
|
} else {
|
|
const char *actualSignature = QDBusMetaType::typeToSignature(value.userType());
|
|
|
|
errorMessage = QStringLiteral("Unexpected `%1' (%2) "
|
|
"upon PropertiesChanged signal arrival "
|
|
"for property `%3.%4' (expected type `%5' (%6))")
|
|
.arg(QString::fromLatin1(value.typeName()),
|
|
QString::fromLatin1(actualSignature),
|
|
interface,
|
|
QString::fromLatin1(metaProperty.name()),
|
|
QString::fromLatin1(metaProperty.typeName()),
|
|
QString::fromLatin1(expectedSignature));
|
|
}
|
|
|
|
if (errorMessage.isEmpty()) {
|
|
*error = QDBusError();
|
|
} else {
|
|
*error = QDBusMessage::createError(QDBusError::InvalidSignature, errorMessage);
|
|
qDebug() << Q_FUNC_INFO << errorMessage;
|
|
}
|
|
|
|
return result;
|
|
}
|