Сигнал QDBus не получен

Я хочу получить сигнал DBus в приложении Qt:

приемник.ч

#ifndef RECEIVER_H
#define RECEIVER_H

#include <iostream>
#include <QObject>

class Receiver : public QObject {
    Q_OBJECT
public slots:
    void receive() {
        std::cout << "Received!";
    }
};

#endif // RECEIVER_H

main.cpp

#include <QCoreApplication>
#include <QDBusConnection>
#include "receiver.h"

int main(int argc, char *argv[])
{
    QCoreApplication a(argc, argv);
    auto recv = new Receiver;
    QDBusConnection::sessionBus().connect("com.ControllerBoard",
                                          "/com/ControllerBoard/Buttons",
                                          "com.ControllerBoard.Buttons",
                                          "PowerButtonPushed", recv, SLOT(receive()));
    return a.exec();
}

К сожалению, это не работает, однако я получил сигнал, указанный в dbus-monitor:

signal time=1516297427.711775 sender=:1.89 -> destination=(null destination) serial=313 path=/com/ControllerBoard/Buttons; interface=com.ControllerBoard.Buttons; member=PowerButtonPushed
signal time=1516297427.807397 sender=:1.89 -> destination=(null destination) serial=314 path=/com/ControllerBoard/Buttons; interface=com.ControllerBoard.Buttons; member=PowerButtonReleased

РЕДАКТИРОВАТЬ:

Я пытался использовать QDBusInterface, но он все еще не работает. Я проверил QDBusInterface::isValid() и QDBusConnection::isConnected(), оба возвращают true.

EDIT2: Это очень странно, когда я запустил это приложение с QDBUS_DEBUG=1, я получил сигнал, указанный в отладке QDBus:

QDBusConnectionPrivate(0x7fe7780032f0) : connected successfully
QDBusConnectionPrivate(0x7fe7780032f0) got message (signal): QDBusMessage(type=Signal, service="org.freedesktop.DBus", path="/org/freedesktop/DBus", interface="org.freedesktop.DBus", member="NameAcquired", signature="s", contents=(":1.104") )
QDBusConnectionPrivate(0x7fe7780032f0) sending message: QDBusMessage(type=MethodCall, service="org.freedesktop.DBus", path="/org/freedesktop/DBus", interface="org.freedesktop.DBus", member="GetNameOwner", signature="", contents=("com.ControllerBoard") )
QDBusConnectionPrivate(0x7fe7780032f0) got message reply: QDBusMessage(type=MethodReturn, service="org.freedesktop.DBus", signature="s", contents=(":1.98") )
QDBusConnectionPrivate(0x7fe7780032f0) sending message: QDBusMessage(type=MethodCall, service="com.ControllerBoard", path="/com/ControllerBoard/Buttons", interface="org.freedesktop.DBus.Introspectable", member="Introspect", signature="", contents=() )
QDBusConnectionPrivate(0x7fe7780032f0) got message reply: QDBusMessage(type=MethodReturn, service=":1.98", signature="s", contents=("<!DOCTYPE node PUBLIC "-//freedesktop//DTD D-BUS Object Introspection 1.0//EN"
"http://www.freedesktop.org/standards/dbus/1.0/introspect.dtd">
<node>
  <interface name="com.ControllerBoard.Buttons">
    <signal name="PowerButtonPushed"/>
    <signal name="PowerButtonReleased"/>
    <signal name="ScalesButtonPushed"/>
    <signal name="ScalesButtonReleased"/>
    <signal name="TimerButtonPushed"/>
    <signal name="TimerButtonReleased"/>
  </interface>
  <interface name="org.freedesktop.DBus.Properties">
    <method name="Get">
      <arg name="interface_name" type="s" direction="in"/>
      <arg name="property_name" type="s" direction="in"/>
      <arg name="value" type="v" direction="out"/>
    </method>
    <method name="Set">
      <arg name="interface_name" type="s" direction="in"/>
      <arg name="property_name" type="s" direction="in"/>
      <arg name="value" type="v" direction="in"/>
    </method>
    <method name="GetAll">
      <arg name="interface_name" type="s" direction="in"/>
      <arg name="values" type="a{sv}" direction="out"/>
      <annotation name="org.qtproject.QtDBus.QtTypeName.Out0" value="QVariantMap"/>
    </method>
    <signal name="PropertiesChanged">
      <arg name="interface_name" type="s" direction="out"/>
      <arg name="changed_properties" type="a{sv}" direction="out"/>
      <annotation name="org.qtproject.QtDBus.QtTypeName.Out1" value="QVariantMap"/>
      <arg name="invalidated_properties" type="as" direction="out"/>
    </signal>
  </interface>
  <interface name="org.freedesktop.DBus.Introspectable">
    <method name="Introspect">
      <arg name="xml_data" type="s" direction="out"/>
    </method>
  </interface>
  <interface name="org.freedesktop.DBus.Peer">
    <method name="Ping"/>
    <method name="GetMachineId">
      <arg name="machine_uuid" type="s" direction="out"/>
    </method>
  </interface>
</node>
") )
QDBusConnectionPrivate(0x7fe7780032f0) Adding rule: "type='signal',sender='org.freedesktop.DBus',interface='org.freedesktop.DBus',member='NameOwnerChanged',arg0='com.ControllerBoard'"
QDBusConnectionPrivate(0x7fe7780032f0) Adding rule: "type='signal',sender='com.ControllerBoard',path='/com/ControllerBoard/Buttons',interface='com.ControllerBoard.Buttons',member='PowerButtonPushed'"
QDBusConnectionPrivate(0x7fe7780032f0) sending message: QDBusMessage(type=MethodCall, service="org.freedesktop.DBus", path="/org/freedesktop/DBus", interface="org.freedesktop.DBus", member="GetNameOwner", signature="", contents=("com.ControllerBoard") )
QDBusConnectionPrivate(0x7fe7780032f0) got message reply: QDBusMessage(type=MethodReturn, service="org.freedesktop.DBus", signature="s", contents=(":1.98") )
QDBusConnectionPrivate(0x7fe7780032f0) Watching service "com.ControllerBoard" for owner changes (current owner: ":1.98" )
QDBusConnectionPrivate(0x7fe7780032f0) got message (signal): QDBusMessage(type=Signal, service=":1.98", path="/com/ControllerBoard/Buttons", interface="com.ControllerBoard.Buttons", member="PowerButtonPushed", signature="", contents=() )

person Sajmplus    schedule 18.01.2018    source источник
comment
разве вы не должны просто иметь SLOT(receive()) в своем вызове подключения?   -  person csunday95    schedule 18.01.2018
comment
@ csunday95 Извините, это просто моя ошибка. Исправлено.   -  person Sajmplus    schedule 19.01.2018


Ответы (1)


Попробуйте использовать этот код. И убедитесь, что ваш приемник остается живым при передаче сигнала.

QDBusInterface *iface = new QDBusInterface(serviceName, servicePath, serviceInterface, QDBusConnection::sessionBus(), this);
if(iface->isValid()) {
    connect(iface, SIGNAL(PowerButtonPushed()), recv, SLOT(receiveSlot()));
}
person Andrey Semenov    schedule 18.01.2018