Как можно представить методы и свойства в DBus, используя QDBusAbstractAdaptor с PyQt4?
Я пытаюсь получить некоторый базовый код, работающий на DBus с использованием PyQt4, в частности, QtDBus. Я использую Python3 версию PyQt4. Я уже получил код, который хочу запустить на Qt (C++), но я хочу, чтобы аналогичный код выполнялся только на Python. Я хочу выставить методы, сигналы / слоты и свойства на DBus для вызова другого кода Python.
В Qt вы используете макрос / функцию Q_CLASSINFO для выполнения самоанализа DBus. Хотя я использовал метод Q_CLASSINFO, я не могу заставить его создавать функциональность того же типа. Насколько я могу судить, метод Q_CLASSINFO не содержит никакой документации, поэтому я не уверен, есть ли другой способ. Используя D-Feet, я ясно вижу, что ни один метод не выставляется автоматически, поэтому я застрял.
Вот что у меня так далеко.
from PyQt4 import QtDBus
from PyQt4.QtCore import QCoreApplication, QObject, Q_CLASSINFO, pyqtSlot, pyqtProperty
from PyQt4.QtDBus import QDBusConnection, QDBusAbstractAdaptor
SERVICE = 'com.home.dbus'
class MyServer(QObject):
def __init__(self):
QObject.__init__(self)
self.__dbusAdaptor = ServerAdaptor(self)
def close(self):
pass
def echo(self, value):
echoed = 'Received {0}'.format(value)
return echoed
def name(self):
return 'myname'
def dbus_adaptor(self):
return self.__dbusAdaptor
class ServerAdaptor(QDBusAbstractAdaptor):
""" This provides the DBus adaptor to the outside world"""
def __init__(self, parent):
super().__init__(parent)
self.__parent = parent
Q_CLASSINFO("D-Bus Introspection",
" <interface name=\"com.home.dbus\">\n"
" <method name=\"name\">\n"
" <arg direction=\"out\" type=\"s\" name=\"name\"/>\n"
" </method>\n"
" <method name=\"echo\">\n"
" <arg direction=\"in\" type=\"s\" name=\"phrase\"/>\n"
" <arg directory=\"out\" type=\"s\" name=\"echoed\"/>\n"
" </method>\n"
" </interface>\n")
def close(self):
parent.close()
def echo(self, value):
return parent.echo(value)
def name(self):
return parent.name
def start():
app = QCoreApplication([])
if QDBusConnection.sessionBus().isConnected() == False:
print('Cannot connect to D-Bus session bus')
return
print('Starting')
server = MyServer()
if not QDBusConnection.sessionBus().registerService(SERVICE):
print('Unable to register service name')
return
if not QDBusConnection.sessionBus().registerObject('/mydbus', server.dbus_adaptor):
print('Unable to register object at service path')
return
app.exec();
print('Exited')
if __name__ == '__main__':
start()
Хотя мне очень нравится использование QtDBus в C++ из-за того, как я хочу структурировать этот мой большой проект, мне действительно нужно, чтобы объект, доступ к которому осуществлялся через DBus, был написан на Python3.
1 ответ
Есть несколько проблем с вашей программой. Я рекомендую взглянуть на remotecontrolledcar
а также pingpong
примеры из последних источников PyQt, они довольно полезны. Основные моменты, на которые следует обратить внимание:
- Вы должны пройти
MyServer
экземпляр (неServerAdaptor
) чтобыregisterObject()
- добавлять
pyqtSlot()
декораторы для функций, которые вы хотите выставить через D-Bus - Вызов
Q_CLASSINFO()
в верхней части класса адаптера, а не в его__init__()
функция - Также установите "Интерфейс D-Bus", используя
Q_CLASSINFO()
- Ваш самоанализ XML содержал опечатку ("каталог" вместо "направление")
Вот урезанный пример, который работает для меня (Python 3.2.3/Qt 4.8.2/PyQt 4.9.4):
from PyQt4 import QtDBus
from PyQt4.QtCore import (QCoreApplication, QObject, Q_CLASSINFO, pyqtSlot,
pyqtProperty)
from PyQt4.QtDBus import QDBusConnection, QDBusAbstractAdaptor
class MyServer(QObject):
def __init__(self):
QObject.__init__(self)
self.__dbusAdaptor = ServerAdaptor(self)
self.__name = 'myname'
def echo(self, value):
return'Received: {0}'.format(value)
@property
def name(self):
return self.__name
@name.setter
def name(self, value):
self.__name = value
class ServerAdaptor(QDBusAbstractAdaptor):
""" This provides the DBus adaptor to the outside world"""
Q_CLASSINFO("D-Bus Interface", "com.home.dbus")
Q_CLASSINFO("D-Bus Introspection",
' <interface name="com.home.dbus">\n'
' <property name="name" type="s" access="readwrite"/>\n'
' <method name="echo">\n'
' <arg direction="in" type="s" name="phrase"/>\n'
' <arg direction="out" type="s" name="echoed"/>\n'
' </method>\n'
' </interface>\n')
def __init__(self, parent):
super().__init__(parent)
@pyqtSlot(str, result=str)
def echo(self, phrase):
return self.parent().echo(phrase)
@pyqtProperty(str)
def name(self):
return self.parent().name
@name.setter
def name(self, value):
self.parent().name = value
def start():
app = QCoreApplication([])
bus = QDBusConnection.sessionBus()
server = MyServer()
bus.registerObject('/mydbus', server)
bus.registerService('com.home.dbus')
app.exec()
if __name__ == '__main__':
start()