/**************************************************************************** ** ** Copyright (C) 2012 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of the test suite of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3.0 as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU General Public License version 3.0 requirements will be ** met: http://www.gnu.org/copyleft/gpl.html. ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include #include #include #include class BaseObject: public QObject { Q_OBJECT Q_CLASSINFO("D-Bus Interface", "local.BaseObject") public: BaseObject(QObject *parent = 0) : QObject(parent) { } public slots: void anotherMethod() { } }; class MyObject: public BaseObject { Q_OBJECT public slots: void method(const QDBusMessage &msg); public: static QString path; int callCount; MyObject(QObject *parent = 0) : BaseObject(parent), callCount(0) {} }; void MyObject::method(const QDBusMessage &msg) { path = msg.path(); ++callCount; //qDebug() << msg; } class tst_QDBusConnection: public QObject { Q_OBJECT int signalsReceived; public slots: void oneSlot() { ++signalsReceived; } void exitLoop() { ++signalsReceived; QTestEventLoop::instance().exitLoop(); } void secondCallWithCallback(); private slots: void noConnection(); void connectToBus(); void connectToPeer(); void connect(); void send(); void sendWithGui(); void sendAsync(); void sendSignal(); void registerObject_data(); void registerObject(); void registerObjectPeer_data(); void registerObjectPeer(); void registerObject2(); void registerObjectPeer2(); void registerQObjectChildren(); void registerQObjectChildrenPeer(); void callSelf(); void callSelfByAnotherName_data(); void callSelfByAnotherName(); void multipleInterfacesInQObject(); void slotsWithLessParameters(); void nestedCallWithCallback(); void serviceRegistrationRaceCondition(); void registerVirtualObject(); void callVirtualObject(); void callVirtualObjectLocal(); public: QString serviceName() const { return "com.trolltech.Qt.Autotests.QDBusConnection"; } bool callMethod(const QDBusConnection &conn, const QString &path); bool callMethodPeer(const QDBusConnection &conn, const QString &path); }; class QDBusSpy: public QObject { Q_OBJECT public slots: void handlePing(const QString &str) { args.clear(); args << str; } void asyncReply(const QDBusMessage &msg) { args = msg.arguments(); } public: QList args; }; void tst_QDBusConnection::noConnection() { QDBusConnection con = QDBusConnection::connectToBus("unix:path=/dev/null", "testconnection"); QVERIFY(!con.isConnected()); // try sending a message. This should fail QDBusMessage msg = QDBusMessage::createMethodCall("org.kde.selftest", "/org/kde/selftest", "org.kde.selftest", "Ping"); msg << QLatin1String("ping"); QVERIFY(!con.send(msg)); QDBusSpy spy; QVERIFY(con.callWithCallback(msg, &spy, SLOT(asyncReply)) == 0); QDBusMessage reply = con.call(msg); QVERIFY(reply.type() == QDBusMessage::ErrorMessage); QDBusReply voidreply(reply); QVERIFY(!voidreply.isValid()); QDBusConnection::disconnectFromBus("testconnection"); } void tst_QDBusConnection::sendSignal() { QDBusConnection con = QDBusConnection::sessionBus(); QVERIFY(con.isConnected()); QDBusMessage msg = QDBusMessage::createSignal("/org/kde/selftest", "org.kde.selftest", "Ping"); msg << QLatin1String("ping"); QVERIFY(con.send(msg)); QTest::qWait(1000); } void tst_QDBusConnection::send() { QDBusConnection con = QDBusConnection::sessionBus(); QVERIFY(con.isConnected()); QDBusMessage msg = QDBusMessage::createMethodCall("org.freedesktop.DBus", "/org/freedesktop/DBus", "org.freedesktop.DBus", "ListNames"); QDBusMessage reply = con.call(msg); QCOMPARE(reply.arguments().count(), 1); QCOMPARE(reply.arguments().at(0).typeName(), "QStringList"); QVERIFY(reply.arguments().at(0).toStringList().contains(con.baseService())); } void tst_QDBusConnection::sendWithGui() { QDBusConnection con = QDBusConnection::sessionBus(); QVERIFY(con.isConnected()); QDBusMessage msg = QDBusMessage::createMethodCall("org.freedesktop.DBus", "/org/freedesktop/DBus", "org.freedesktop.DBus", "ListNames"); QDBusMessage reply = con.call(msg, QDBus::BlockWithGui); QCOMPARE(reply.arguments().count(), 1); QCOMPARE(reply.arguments().at(0).typeName(), "QStringList"); QVERIFY(reply.arguments().at(0).toStringList().contains(con.baseService())); } void tst_QDBusConnection::sendAsync() { QDBusConnection con = QDBusConnection::sessionBus(); QVERIFY(con.isConnected()); QDBusSpy spy; QDBusMessage msg = QDBusMessage::createMethodCall("org.freedesktop.DBus", "/org/freedesktop/DBus", "org.freedesktop.DBus", "ListNames"); QVERIFY(con.callWithCallback(msg, &spy, SLOT(asyncReply(QDBusMessage)))); QTest::qWait(1000); QCOMPARE(spy.args.value(0).typeName(), "QStringList"); QVERIFY(spy.args.at(0).toStringList().contains(con.baseService())); } void tst_QDBusConnection::connect() { QDBusSpy spy; QDBusConnection con = QDBusConnection::sessionBus(); con.connect(con.baseService(), "/org/kde/selftest", "org.kde.selftest", "ping", &spy, SLOT(handlePing(QString))); QDBusMessage msg = QDBusMessage::createSignal("/org/kde/selftest", "org.kde.selftest", "ping"); msg << QLatin1String("ping"); QVERIFY(con.send(msg)); QTest::qWait(1000); QCOMPARE(spy.args.count(), 1); QCOMPARE(spy.args.at(0).toString(), QString("ping")); } void tst_QDBusConnection::connectToBus() { { QDBusConnection con = QDBusConnection::connectToBus( QDBusConnection::SessionBus, "bubu"); QVERIFY(con.isConnected()); QVERIFY(!con.lastError().isValid()); QDBusConnection con2("foo"); QVERIFY(!con2.isConnected()); QVERIFY(!con2.lastError().isValid()); con2 = con; QVERIFY(con.isConnected()); QVERIFY(con2.isConnected()); QVERIFY(!con.lastError().isValid()); QVERIFY(!con2.lastError().isValid()); } { QDBusConnection con("bubu"); QVERIFY(con.isConnected()); QVERIFY(!con.lastError().isValid()); } QDBusConnection::disconnectFromPeer("bubu"); { QDBusConnection con("bubu"); QVERIFY(con.isConnected()); QVERIFY(!con.lastError().isValid()); } QDBusConnection::disconnectFromBus("bubu"); { QDBusConnection con("bubu"); QVERIFY(!con.isConnected()); QVERIFY(!con.lastError().isValid()); } QByteArray address = qgetenv("DBUS_SESSION_BUS_ADDRESS"); if (!address.isEmpty()) { QDBusConnection con = QDBusConnection::connectToBus(address, "newconn"); QVERIFY(con.isConnected()); QVERIFY(!con.lastError().isValid()); QDBusConnection::disconnectFromBus("newconn"); } } void tst_QDBusConnection::connectToPeer() { { QDBusConnection con = QDBusConnection::connectToPeer( "", "newconn"); QVERIFY(!con.isConnected()); QVERIFY(con.lastError().isValid()); } QDBusServer server("unix:tmpdir=/tmp", 0); { QDBusConnection con = QDBusConnection::connectToPeer( "unix:abstract=/tmp/dbus-XXXXXXXXXX,guid=00000000000000000000000000000000", "newconn2"); QVERIFY(!con.isConnected()); QVERIFY(con.lastError().isValid()); } { QDBusConnection con = QDBusConnection::connectToPeer( server.address(), "bubu"); QVERIFY(con.isConnected()); QVERIFY(!con.lastError().isValid()); QDBusConnection con2("foo"); QVERIFY(!con2.isConnected()); QVERIFY(!con2.lastError().isValid()); con2 = con; QVERIFY(con.isConnected()); QVERIFY(con2.isConnected()); QVERIFY(!con.lastError().isValid()); QVERIFY(!con2.lastError().isValid()); } { QDBusConnection con("bubu"); QVERIFY(con.isConnected()); QVERIFY(!con.lastError().isValid()); } QDBusConnection::disconnectFromBus("bubu"); { QDBusConnection con("bubu"); QVERIFY(con.isConnected()); QVERIFY(!con.lastError().isValid()); } QDBusConnection::disconnectFromPeer("bubu"); { QDBusConnection con("bubu"); QVERIFY(!con.isConnected()); QVERIFY(!con.lastError().isValid()); } } void tst_QDBusConnection::registerObject_data() { QTest::addColumn("path"); QTest::newRow("/") << "/"; QTest::newRow("/p1") << "/p1"; QTest::newRow("/p2") << "/p2"; QTest::newRow("/p1/q") << "/p1/q"; QTest::newRow("/p1/q/r") << "/p1/q/r"; } void tst_QDBusConnection::registerObject() { QFETCH(QString, path); QDBusConnection con = QDBusConnection::sessionBus(); QVERIFY(con.isConnected()); //QVERIFY(!callMethod(con, path)); { // register one object at root: MyObject obj; QVERIFY(con.registerObject(path, &obj, QDBusConnection::ExportAllSlots)); QCOMPARE(con.objectRegisteredAt(path), static_cast(&obj)); QVERIFY(callMethod(con, path)); QCOMPARE(obj.path, path); } // make sure it's gone QVERIFY(!callMethod(con, path)); } class MyServer : public QDBusServer { Q_OBJECT public: MyServer(QString path, QString addr, QObject* parent) : QDBusServer(addr, parent), m_path(path), m_connections() { connect(this, SIGNAL(newConnection(const QDBusConnection&)), SLOT(handleConnection(const QDBusConnection&))); } bool registerObject(const QDBusConnection& c) { QDBusConnection conn(c); if (!conn.registerObject(m_path, &m_obj, QDBusConnection::ExportAllSlots)) return false; if (!(conn.objectRegisteredAt(m_path) == &m_obj)) return false; return true; } bool registerObject() { Q_FOREACH (const QString &name, m_connections) { if (!registerObject(QDBusConnection(name))) return false; } return true; } void unregisterObject() { Q_FOREACH (const QString &name, m_connections) { QDBusConnection c(name); c.unregisterObject(m_path); } } public slots: void handleConnection(const QDBusConnection& c) { m_connections << c.name(); QVERIFY(isConnected()); QVERIFY(c.isConnected()); QVERIFY(registerObject(c)); } private: MyObject m_obj; QString m_path; QStringList m_connections; }; void tst_QDBusConnection::registerObjectPeer_data() { QTest::addColumn("path"); QTest::newRow("/") << "/"; QTest::newRow("/p1") << "/p1"; QTest::newRow("/p2") << "/p2"; QTest::newRow("/p1/q") << "/p1/q"; QTest::newRow("/p1/q/r") << "/p1/q/r"; } void tst_QDBusConnection::registerObjectPeer() { QFETCH(QString, path); MyServer server(path, "unix:tmpdir=/tmp", 0); QDBusConnection::connectToPeer(server.address(), "beforeFoo"); { QDBusConnection con = QDBusConnection::connectToPeer(server.address(), "foo"); QCoreApplication::processEvents(); QVERIFY(con.isConnected()); MyObject obj; QVERIFY(callMethodPeer(con, path)); QCOMPARE(obj.path, path); } QDBusConnection::connectToPeer(server.address(), "afterFoo"); { QDBusConnection con("foo"); QVERIFY(con.isConnected()); QVERIFY(callMethodPeer(con, path)); } server.unregisterObject(); { QDBusConnection con("foo"); QVERIFY(con.isConnected()); QVERIFY(!callMethodPeer(con, path)); } server.registerObject(); { QDBusConnection con("foo"); QVERIFY(con.isConnected()); QVERIFY(callMethodPeer(con, path)); } QDBusConnection::disconnectFromPeer("foo"); { QDBusConnection con("foo"); QVERIFY(!con.isConnected()); QVERIFY(!callMethodPeer(con, path)); } QDBusConnection::disconnectFromPeer("beforeFoo"); QDBusConnection::disconnectFromPeer("afterFoo"); } void tst_QDBusConnection::registerObject2() { QDBusConnection con = QDBusConnection::sessionBus(); QVERIFY(con.isConnected()); // make sure nothing is using our paths: QVERIFY(!callMethod(con, "/")); QVERIFY(!callMethod(con, "/p1")); QVERIFY(!callMethod(con, "/p2")); QVERIFY(!callMethod(con, "/p1/q")); QVERIFY(!callMethod(con, "/p1/q/r")); { // register one object at root: MyObject obj; QVERIFY(con.registerObject("/", &obj, QDBusConnection::ExportAllSlots)); QVERIFY(callMethod(con, "/")); qDebug() << obj.path; QCOMPARE(obj.path, QString("/")); } // make sure it's gone QVERIFY(!callMethod(con, "/")); { // register one at an element: MyObject obj; QVERIFY(con.registerObject("/p1", &obj, QDBusConnection::ExportAllSlots)); QVERIFY(!callMethod(con, "/")); QVERIFY(callMethod(con, "/p1")); qDebug() << obj.path; QCOMPARE(obj.path, QString("/p1")); // re-register it somewhere else QVERIFY(con.registerObject("/p2", &obj, QDBusConnection::ExportAllSlots)); QVERIFY(callMethod(con, "/p1")); QCOMPARE(obj.path, QString("/p1")); QVERIFY(callMethod(con, "/p2")); QCOMPARE(obj.path, QString("/p2")); } // make sure it's gone QVERIFY(!callMethod(con, "/p1")); QVERIFY(!callMethod(con, "/p2")); { // register at a deep path MyObject obj; QVERIFY(con.registerObject("/p1/q/r", &obj, QDBusConnection::ExportAllSlots)); QVERIFY(!callMethod(con, "/")); QVERIFY(!callMethod(con, "/p1")); QVERIFY(!callMethod(con, "/p1/q")); QVERIFY(callMethod(con, "/p1/q/r")); QCOMPARE(obj.path, QString("/p1/q/r")); } // make sure it's gone QVERIFY(!callMethod(con, "/p1/q/r")); { MyObject obj; QVERIFY(con.registerObject("/p1/q2", &obj, QDBusConnection::ExportAllSlots)); QVERIFY(callMethod(con, "/p1/q2")); QCOMPARE(obj.path, QString("/p1/q2")); // try unregistering con.unregisterObject("/p1/q2"); QVERIFY(!callMethod(con, "/p1/q2")); // register it again QVERIFY(con.registerObject("/p1/q2", &obj, QDBusConnection::ExportAllSlots)); QVERIFY(callMethod(con, "/p1/q2")); QCOMPARE(obj.path, QString("/p1/q2")); // now try removing things around it: con.unregisterObject("/p2"); QVERIFY(callMethod(con, "/p1/q2")); // unrelated object shouldn't affect con.unregisterObject("/p1"); QVERIFY(callMethod(con, "/p1/q2")); // unregistering just the parent shouldn't affect it con.unregisterObject("/p1/q2/r"); QVERIFY(callMethod(con, "/p1/q2")); // unregistering non-existing child shouldn't affect it either con.unregisterObject("/p1/q"); QVERIFY(callMethod(con, "/p1/q2")); // unregistering sibling (before) shouldn't affect con.unregisterObject("/p1/r"); QVERIFY(callMethod(con, "/p1/q2")); // unregistering sibling (after) shouldn't affect // now remove it: con.unregisterObject("/p1", QDBusConnection::UnregisterTree); QVERIFY(!callMethod(con, "/p1/q2")); // we removed the full tree } } class MyServer2 : public QDBusServer { Q_OBJECT public: MyServer2(QString addr, QObject* parent) : QDBusServer(addr, parent), m_conn("none") { connect(this, SIGNAL(newConnection(const QDBusConnection&)), SLOT(handleConnection(const QDBusConnection&))); } QDBusConnection connection() { return m_conn; } public slots: void handleConnection(const QDBusConnection& c) { m_conn = c; QVERIFY(isConnected()); QVERIFY(m_conn.isConnected()); } private: MyObject m_obj; QDBusConnection m_conn; }; void tst_QDBusConnection::registerObjectPeer2() { MyServer2 server("unix:tmpdir=/tmp", 0); QDBusConnection con = QDBusConnection::connectToPeer(server.address(), "foo"); QCoreApplication::processEvents(); QVERIFY(con.isConnected()); QDBusConnection srv_con = server.connection(); // make sure nothing is using our paths: QVERIFY(!callMethodPeer(srv_con, "/")); QVERIFY(!callMethodPeer(srv_con, "/p1")); QVERIFY(!callMethodPeer(srv_con, "/p2")); QVERIFY(!callMethodPeer(srv_con, "/p1/q")); QVERIFY(!callMethodPeer(srv_con, "/p1/q/r")); { // register one object at root: MyObject obj; QVERIFY(con.registerObject("/", &obj, QDBusConnection::ExportAllSlots)); QVERIFY(callMethodPeer(srv_con, "/")); qDebug() << obj.path; QCOMPARE(obj.path, QString("/")); } // make sure it's gone QVERIFY(!callMethodPeer(srv_con, "/")); { // register one at an element: MyObject obj; QVERIFY(con.registerObject("/p1", &obj, QDBusConnection::ExportAllSlots)); QVERIFY(!callMethodPeer(srv_con, "/")); QVERIFY(callMethodPeer(srv_con, "/p1")); qDebug() << obj.path; QCOMPARE(obj.path, QString("/p1")); // re-register it somewhere else QVERIFY(con.registerObject("/p2", &obj, QDBusConnection::ExportAllSlots)); QVERIFY(callMethodPeer(srv_con, "/p1")); QCOMPARE(obj.path, QString("/p1")); QVERIFY(callMethodPeer(srv_con, "/p2")); QCOMPARE(obj.path, QString("/p2")); } // make sure it's gone QVERIFY(!callMethodPeer(srv_con, "/p1")); QVERIFY(!callMethodPeer(srv_con, "/p2")); { // register at a deep path MyObject obj; QVERIFY(con.registerObject("/p1/q/r", &obj, QDBusConnection::ExportAllSlots)); QVERIFY(!callMethodPeer(srv_con, "/")); QVERIFY(!callMethodPeer(srv_con, "/p1")); QVERIFY(!callMethodPeer(srv_con, "/p1/q")); QVERIFY(callMethodPeer(srv_con, "/p1/q/r")); QCOMPARE(obj.path, QString("/p1/q/r")); } // make sure it's gone QVERIFY(!callMethodPeer(srv_con, "/p1/q/r")); { MyObject obj; QVERIFY(con.registerObject("/p1/q2", &obj, QDBusConnection::ExportAllSlots)); QVERIFY(callMethodPeer(srv_con, "/p1/q2")); QCOMPARE(obj.path, QString("/p1/q2")); // try unregistering con.unregisterObject("/p1/q2"); QVERIFY(!callMethodPeer(srv_con, "/p1/q2")); // register it again QVERIFY(con.registerObject("/p1/q2", &obj, QDBusConnection::ExportAllSlots)); QVERIFY(callMethodPeer(srv_con, "/p1/q2")); QCOMPARE(obj.path, QString("/p1/q2")); // now try removing things around it: con.unregisterObject("/p2"); QVERIFY(callMethodPeer(srv_con, "/p1/q2")); // unrelated object shouldn't affect con.unregisterObject("/p1"); QVERIFY(callMethodPeer(srv_con, "/p1/q2")); // unregistering just the parent shouldn't affect it con.unregisterObject("/p1/q2/r"); QVERIFY(callMethodPeer(srv_con, "/p1/q2")); // unregistering non-existing child shouldn't affect it either con.unregisterObject("/p1/q"); QVERIFY(callMethodPeer(srv_con, "/p1/q2")); // unregistering sibling (before) shouldn't affect con.unregisterObject("/p1/r"); QVERIFY(callMethodPeer(srv_con, "/p1/q2")); // unregistering sibling (after) shouldn't affect // now remove it: con.unregisterObject("/p1", QDBusConnection::UnregisterTree); QVERIFY(!callMethodPeer(srv_con, "/p1/q2")); // we removed the full tree } QDBusConnection::disconnectFromPeer("foo"); } void tst_QDBusConnection::registerQObjectChildren() { // make sure no one is there QDBusConnection con = QDBusConnection::sessionBus(); QVERIFY(!callMethod(con, "/p1")); { MyObject obj, *a, *b, *c, *cc; a = new MyObject(&obj); a->setObjectName("a"); b = new MyObject(&obj); b->setObjectName("b"); c = new MyObject(&obj); c->setObjectName("c"); cc = new MyObject(c); cc->setObjectName("cc"); con.registerObject("/p1", &obj, QDBusConnection::ExportAllSlots | QDBusConnection::ExportChildObjects); // make calls QVERIFY(callMethod(con, "/p1")); QCOMPARE(obj.callCount, 1); QVERIFY(callMethod(con, "/p1/a")); QCOMPARE(a->callCount, 1); QVERIFY(callMethod(con, "/p1/b")); QCOMPARE(b->callCount, 1); QVERIFY(callMethod(con, "/p1/c")); QCOMPARE(c->callCount, 1); QVERIFY(callMethod(con, "/p1/c/cc")); QCOMPARE(cc->callCount, 1); QVERIFY(!callMethod(con, "/p1/d")); QVERIFY(!callMethod(con, "/p1/c/abc")); // pull an object, see if it goes away: delete b; QVERIFY(!callMethod(con, "/p1/b")); delete c; QVERIFY(!callMethod(con, "/p1/c")); QVERIFY(!callMethod(con, "/p1/c/cc")); } QVERIFY(!callMethod(con, "/p1")); QVERIFY(!callMethod(con, "/p1/a")); QVERIFY(!callMethod(con, "/p1/b")); QVERIFY(!callMethod(con, "/p1/c")); QVERIFY(!callMethod(con, "/p1/c/cc")); } void tst_QDBusConnection::registerQObjectChildrenPeer() { MyServer2 server("unix:tmpdir=/tmp", 0); QDBusConnection con = QDBusConnection::connectToPeer(server.address(), "foo"); QCoreApplication::processEvents(); QVERIFY(con.isConnected()); QDBusConnection srv_con = server.connection(); QVERIFY(!callMethodPeer(srv_con, "/p1")); { MyObject obj, *a, *b, *c, *cc; a = new MyObject(&obj); a->setObjectName("a"); b = new MyObject(&obj); b->setObjectName("b"); c = new MyObject(&obj); c->setObjectName("c"); cc = new MyObject(c); cc->setObjectName("cc"); con.registerObject("/p1", &obj, QDBusConnection::ExportAllSlots | QDBusConnection::ExportChildObjects); // make calls QVERIFY(callMethodPeer(srv_con, "/p1")); QCOMPARE(obj.callCount, 1); QVERIFY(callMethodPeer(srv_con, "/p1/a")); QCOMPARE(a->callCount, 1); QVERIFY(callMethodPeer(srv_con, "/p1/b")); QCOMPARE(b->callCount, 1); QVERIFY(callMethodPeer(srv_con, "/p1/c")); QCOMPARE(c->callCount, 1); QVERIFY(callMethodPeer(srv_con, "/p1/c/cc")); QCOMPARE(cc->callCount, 1); QVERIFY(!callMethodPeer(srv_con, "/p1/d")); QVERIFY(!callMethodPeer(srv_con, "/p1/c/abc")); // pull an object, see if it goes away: delete b; QVERIFY(!callMethodPeer(srv_con, "/p1/b")); delete c; QVERIFY(!callMethodPeer(srv_con, "/p1/c")); QVERIFY(!callMethodPeer(srv_con, "/p1/c/cc")); } QVERIFY(!callMethodPeer(srv_con, "/p1")); QVERIFY(!callMethodPeer(srv_con, "/p1/a")); QVERIFY(!callMethodPeer(srv_con, "/p1/b")); QVERIFY(!callMethodPeer(srv_con, "/p1/c")); QVERIFY(!callMethodPeer(srv_con, "/p1/c/cc")); QDBusConnection::disconnectFromPeer("foo"); } bool tst_QDBusConnection::callMethod(const QDBusConnection &conn, const QString &path) { QDBusMessage msg = QDBusMessage::createMethodCall(conn.baseService(), path, "", "method"); QDBusMessage reply = conn.call(msg, QDBus::Block/*WithGui*/); if (reply.type() != QDBusMessage::ReplyMessage) return false; if (MyObject::path == path) { QTest::compare_helper(true, "COMPARE()", __FILE__, __LINE__); } else { QTest::compare_helper(false, "Compared values are not the same", QTest::toString(MyObject::path), QTest::toString(path), "MyObject::path", "path", __FILE__, __LINE__); return false; } return true; } bool tst_QDBusConnection::callMethodPeer(const QDBusConnection &conn, const QString &path) { QDBusMessage msg = QDBusMessage::createMethodCall("", path, "", "method"); QDBusMessage reply = conn.call(msg, QDBus::BlockWithGui); if (reply.type() != QDBusMessage::ReplyMessage) return false; if (MyObject::path == path) { QTest::compare_helper(true, "COMPARE()", __FILE__, __LINE__); } else { QTest::compare_helper(false, "Compared values are not the same", QTest::toString(MyObject::path), QTest::toString(path), "MyObject::path", "path", __FILE__, __LINE__); return false; } return true; } class TestObject : public QObject { Q_OBJECT public: TestObject(QObject *parent = 0) : QObject(parent) {} ~TestObject() {} QString func; public slots: void test0() { func = "test0"; } void test1(int i) { func = "test1 " + QString::number(i); } int test2() { func = "test2"; return 43; } int test3(int i) { func = "test2"; return i + 1; } }; void tst_QDBusConnection::callSelf() { TestObject testObject; QDBusConnection connection = QDBusConnection::sessionBus(); QVERIFY(connection.registerObject("/test", &testObject, QDBusConnection::ExportAllContents)); QCOMPARE(connection.objectRegisteredAt("/test"), static_cast(&testObject)); QVERIFY(connection.registerService(serviceName())); QDBusInterface interface(serviceName(), "/test"); QVERIFY(interface.isValid()); interface.call(QDBus::Block, "test0"); QCOMPARE(testObject.func, QString("test0")); interface.call(QDBus::Block, "test1", 42); QCOMPARE(testObject.func, QString("test1 42")); QDBusMessage reply = interface.call(QDBus::Block, "test2"); QCOMPARE(testObject.func, QString("test2")); QCOMPARE(reply.arguments().value(0).toInt(), 43); QDBusMessage msg = QDBusMessage::createMethodCall(serviceName(), "/test", QString(), "test3"); msg << 44; reply = connection.call(msg); QCOMPARE(reply.arguments().value(0).toInt(), 45); } void tst_QDBusConnection::callSelfByAnotherName_data() { QTest::addColumn("registerMethod"); QTest::newRow("connection") << 0; QTest::newRow("connection-interface") << 1; QTest::newRow("direct") << 2; } void tst_QDBusConnection::callSelfByAnotherName() { static int counter = 0; QString sname = serviceName() + QString::number(counter++); QDBusConnection con = QDBusConnection::sessionBus(); QVERIFY(con.isConnected()); TestObject testObject; QVERIFY(con.registerObject("/test", &testObject, QDBusConnection::ExportAllContents)); con.connect("org.freedesktop.DBus", "/org/freedesktop/DBus", "org.freedesktop.DBus", "NameOwnerChanged", QStringList() << sname << "", QString(), &QTestEventLoop::instance(), SLOT(exitLoop())); // register the name QFETCH(int, registerMethod); switch (registerMethod) { case 0: QVERIFY(con.registerService(sname)); break; case 1: QVERIFY(con.interface()->registerService(sname).value() == QDBusConnectionInterface::ServiceRegistered); break; case 2: { // flag is DBUS_NAME_FLAG_DO_NOT_QUEUE = 0x04 // reply is DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER = 1 QDBusReply reply = con.interface()->call("RequestName", sname, 4u); QVERIFY(reply.value() == 1); } } struct Deregisterer { QDBusConnection con; QString sname; Deregisterer(const QDBusConnection &con, const QString &sname) : con(con), sname(sname) {} ~Deregisterer() { con.interface()->unregisterService(sname); } } deregisterer(con, sname); // give the connection a chance to find out that we're good to go QTestEventLoop::instance().enterLoop(2); con.disconnect("org.freedesktop.DBus", "/org/freedesktop/DBus", "org.freedesktop.DBus", "NameOwnerChanged", QStringList() << sname << "", QString(), &QTestEventLoop::instance(), SLOT(exitLoop())); QVERIFY(!QTestEventLoop::instance().timeout()); // make the call QDBusMessage msg = QDBusMessage::createMethodCall(sname, "/test", QString(), "test0"); QDBusMessage reply = con.call(msg, QDBus::Block, 1000); QVERIFY(reply.type() == QDBusMessage::ReplyMessage); } void tst_QDBusConnection::multipleInterfacesInQObject() { QDBusConnection con = QDBusConnection::sessionBus(); QVERIFY(!callMethod(con, "/p1")); MyObject obj; con.registerObject("/p1", &obj, QDBusConnection::ExportAllSlots); // check if we can call the BaseObject's interface QDBusMessage msg = QDBusMessage::createMethodCall(con.baseService(), "/p1", "local.BaseObject", "anotherMethod"); QDBusMessage reply = con.call(msg, QDBus::Block); QCOMPARE(reply.type(), QDBusMessage::ReplyMessage); QVERIFY(reply.arguments().count() == 0); } void tst_QDBusConnection::slotsWithLessParameters() { QDBusConnection con = QDBusConnection::sessionBus(); QDBusMessage signal = QDBusMessage::createSignal("/", "com.trolltech.TestCase", "oneSignal"); signal << "one parameter"; signalsReceived = 0; QVERIFY(con.connect(con.baseService(), signal.path(), signal.interface(), signal.member(), this, SLOT(oneSlot()))); QVERIFY(con.send(signal)); QTest::qWait(100); QCOMPARE(signalsReceived, 1); // disconnect and try with a signature signalsReceived = 0; QVERIFY(con.disconnect(con.baseService(), signal.path(), signal.interface(), signal.member(), this, SLOT(oneSlot()))); QVERIFY(con.connect(con.baseService(), signal.path(), signal.interface(), signal.member(), "s", this, SLOT(oneSlot()))); QVERIFY(con.send(signal)); QTest::qWait(100); QCOMPARE(signalsReceived, 1); } void tst_QDBusConnection::secondCallWithCallback() { qDebug("Hello"); QDBusConnection con = QDBusConnection::sessionBus(); QDBusMessage msg = QDBusMessage::createMethodCall(con.baseService(), "/test", QString(), "test0"); con.callWithCallback(msg, this, SLOT(exitLoop()), SLOT(secondCallWithCallback())); } void tst_QDBusConnection::nestedCallWithCallback() { TestObject testObject; QDBusConnection connection = QDBusConnection::sessionBus(); QVERIFY(connection.registerObject("/test", &testObject, QDBusConnection::ExportAllContents)); QDBusMessage msg = QDBusMessage::createMethodCall(connection.baseService(), "/test", QString(), "ThisFunctionDoesntExist"); signalsReceived = 0; connection.callWithCallback(msg, this, SLOT(exitLoop()), SLOT(secondCallWithCallback()), 10); QTestEventLoop::instance().enterLoop(15); QVERIFY(!QTestEventLoop::instance().timeout()); QCOMPARE(signalsReceived, 1); } class RaceConditionSignalWaiter : public QObject { Q_OBJECT public: int count; RaceConditionSignalWaiter() : count (0) {} virtual ~RaceConditionSignalWaiter() {} public slots: void countUp() { ++count; emit done(); } signals: void done(); }; void tst_QDBusConnection::serviceRegistrationRaceCondition() { // There was a race condition in the updating of list of name owners in // QtDBus. When the user connects to a signal coming from a given // service, we must listen for NameOwnerChanged signals relevant to that // name and update when the owner changes. However, it's possible that we // receive in one chunk from the server both the NameOwnerChanged signal // about the service and the signal we're interested in. Since QtDBus // posts events in order to handle the incoming signals, the update // happens too late. const QString connectionName = "testConnectionName"; const QString serviceName = "org.example.SecondaryName"; QDBusConnection session = QDBusConnection::sessionBus(); QVERIFY(!session.interface()->isServiceRegistered(serviceName)); // connect to the signal: RaceConditionSignalWaiter recv; session.connect(serviceName, "/", "com.trolltech.TestCase", "oneSignal", &recv, SLOT(countUp())); // create a secondary connection and register a name QDBusConnection connection = QDBusConnection::connectToBus(QDBusConnection::SessionBus, connectionName); QDBusConnection::disconnectFromBus(connectionName); // disconnection happens when "connection" goes out of scope QVERIFY(connection.isConnected()); QVERIFY(connection.registerService(serviceName)); // send a signal QDBusMessage msg = QDBusMessage::createSignal("/", "com.trolltech.TestCase", "oneSignal"); connection.send(msg); // make a blocking call just to be sure that the buffer was flushed msg = QDBusMessage::createMethodCall("org.freedesktop.DBus", "/org/freedesktop/DBus", "org.freedesktop.DBus", "NameHasOwner"); msg << connectionName; connection.call(msg); // ignore result // Now here's the race condition (more info on task QTBUG-15651): // the bus has most likely queued three signals for us to work on: // 1) NameOwnerChanged for the connection we created above // 2) NameOwnerChanged for the service we registered above // 3) The "oneSignal" signal we sent // // We'll most likely receive all three in one go from the server. We must // update the owner of serviceName before we start processing the // "oneSignal" signal. QTestEventLoop::instance().connect(&recv, SIGNAL(done()), SLOT(exitLoop())); QTestEventLoop::instance().enterLoop(1); QVERIFY(!QTestEventLoop::instance().timeout()); QCOMPARE(recv.count, 1); } class VirtualObject: public QDBusVirtualObject { Q_OBJECT public: VirtualObject() :success(true) {} QString introspect(const QString &path) const { return QString(); } bool handleMessage(const QDBusMessage &message, const QDBusConnection &connection) { ++callCount; lastMessage = message; if (success) { QDBusMessage reply = message.createReply(replyArguments); connection.send(reply); } emit messageReceived(message); return success; } signals: void messageReceived(const QDBusMessage &message) const; public: mutable QDBusMessage lastMessage; QVariantList replyArguments; mutable int callCount; bool success; }; void tst_QDBusConnection::registerVirtualObject() { QDBusConnection con = QDBusConnection::sessionBus(); QVERIFY(con.isConnected()); QString path = "/tree/node"; QString childPath = "/tree/node/child"; QString childChildPath = "/tree/node/child/another"; { // Register VirtualObject that handles child paths. Unregister by going out of scope. VirtualObject obj; QVERIFY(con.registerVirtualObject(path, &obj, QDBusConnection::SubPath)); QCOMPARE(con.objectRegisteredAt(path), static_cast(&obj)); QCOMPARE(con.objectRegisteredAt(childPath), static_cast(&obj)); QCOMPARE(con.objectRegisteredAt(childChildPath), static_cast(&obj)); } QCOMPARE(con.objectRegisteredAt(path), static_cast(0)); QCOMPARE(con.objectRegisteredAt(childPath), static_cast(0)); { // Register VirtualObject that handles child paths. Unregister by calling unregister. VirtualObject obj; QVERIFY(con.registerVirtualObject(path, &obj, QDBusConnection::SubPath)); QCOMPARE(con.objectRegisteredAt(path), static_cast(&obj)); QCOMPARE(con.objectRegisteredAt(childPath), static_cast(&obj)); QCOMPARE(con.objectRegisteredAt(childChildPath), static_cast(&obj)); con.unregisterObject(path); QCOMPARE(con.objectRegisteredAt(path), static_cast(0)); QCOMPARE(con.objectRegisteredAt(childPath), static_cast(0)); } { // Single node has no sub path handling. VirtualObject obj; QVERIFY(con.registerVirtualObject(path, &obj, QDBusConnection::SingleNode)); QCOMPARE(con.objectRegisteredAt(path), static_cast(&obj)); QCOMPARE(con.objectRegisteredAt(childPath), static_cast(0)); } { // Register VirtualObject that handles child paths. Try to register an object on a child path of that. VirtualObject obj; QVERIFY(con.registerVirtualObject(path, &obj, QDBusConnection::SubPath)); QCOMPARE(con.objectRegisteredAt(path), static_cast(&obj)); QObject objectAtSubPath; QVERIFY(!con.registerObject(path, &objectAtSubPath)); QVERIFY(!con.registerObject(childPath, &objectAtSubPath)); QCOMPARE(con.objectRegisteredAt(childPath), static_cast(&obj)); } { // Register object, make sure no SubPath handling object can be registered on a parent path. QObject objectAtSubPath; QVERIFY(con.registerObject(childPath, &objectAtSubPath)); QCOMPARE(con.objectRegisteredAt(childPath), static_cast(&objectAtSubPath)); VirtualObject obj; QVERIFY(!con.registerVirtualObject(path, &obj, QDBusConnection::SubPath)); QCOMPARE(con.objectRegisteredAt(path), static_cast(0)); } QCOMPARE(con.objectRegisteredAt(path), static_cast(0)); QCOMPARE(con.objectRegisteredAt(childPath), static_cast(0)); QCOMPARE(con.objectRegisteredAt(childChildPath), static_cast(0)); } void tst_QDBusConnection::callVirtualObject() { QDBusConnection con = QDBusConnection::sessionBus(); QVERIFY(con.isConnected()); QDBusConnection con2 = QDBusConnection::connectToBus(QDBusConnection::SessionBus, "con2"); QString path = "/tree/node"; QString childPath = "/tree/node/child"; // register one object at root: VirtualObject obj; QVERIFY(con.registerVirtualObject(path, &obj, QDBusConnection::SubPath)); obj.callCount = 0; obj.replyArguments << 42 << 47u; QObject::connect(&obj, SIGNAL(messageReceived(QDBusMessage)), &QTestEventLoop::instance(), SLOT(exitLoop())); QDBusMessage message = QDBusMessage::createMethodCall(con.baseService(), path, QString(), "hello"); QDBusPendingCall reply = con2.asyncCall(message); QTestEventLoop::instance().enterLoop(5); QVERIFY(!QTestEventLoop::instance().timeout()); QCOMPARE(obj.callCount, 1); QCOMPARE(obj.lastMessage.service(), con2.baseService()); QCOMPARE(obj.lastMessage.interface(), QString()); QCOMPARE(obj.lastMessage.path(), path); reply.waitForFinished(); QVERIFY(reply.isValid()); QCOMPARE(reply.reply().arguments(), obj.replyArguments); // call sub path QDBusMessage childMessage = QDBusMessage::createMethodCall(con.baseService(), childPath, QString(), "helloChild"); obj.replyArguments.clear(); obj.replyArguments << 99; QDBusPendingCall childReply = con2.asyncCall(childMessage); QTestEventLoop::instance().enterLoop(5); QVERIFY(!QTestEventLoop::instance().timeout()); QCOMPARE(obj.callCount, 2); QCOMPARE(obj.lastMessage.service(), con2.baseService()); QCOMPARE(obj.lastMessage.interface(), QString()); QCOMPARE(obj.lastMessage.path(), childPath); childReply.waitForFinished(); QVERIFY(childReply.isValid()); QCOMPARE(childReply.reply().arguments(), obj.replyArguments); // let the call fail by having the virtual object return false obj.success = false; QDBusMessage errorMessage = QDBusMessage::createMethodCall(con.baseService(), childPath, QString(), "someFunc"); QDBusPendingCall errorReply = con2.asyncCall(errorMessage); QTestEventLoop::instance().enterLoop(5); QVERIFY(!QTestEventLoop::instance().timeout()); QTest::qWait(100); QVERIFY(errorReply.isError()); qDebug() << errorReply.reply().arguments(); QCOMPARE(errorReply.reply().errorName(), QString("org.freedesktop.DBus.Error.UnknownObject")); QDBusConnection::disconnectFromBus("con2"); } void tst_QDBusConnection::callVirtualObjectLocal() { QDBusConnection con = QDBusConnection::sessionBus(); QVERIFY(con.isConnected()); QString path = "/tree/node"; QString childPath = "/tree/node/child"; // register one object at root: VirtualObject obj; QVERIFY(con.registerVirtualObject(path, &obj, QDBusConnection::SubPath)); obj.callCount = 0; obj.replyArguments << 42 << 47u; QDBusMessage message = QDBusMessage::createMethodCall(con.baseService(), path, QString(), "hello"); QDBusMessage reply = con.call(message, QDBus::Block, 5000); QCOMPARE(obj.callCount, 1); QCOMPARE(obj.lastMessage.service(), con.baseService()); QCOMPARE(obj.lastMessage.interface(), QString()); QCOMPARE(obj.lastMessage.path(), path); QCOMPARE(obj.replyArguments, reply.arguments()); obj.replyArguments << QString("alien abduction"); QDBusMessage subPathMessage = QDBusMessage::createMethodCall(con.baseService(), childPath, QString(), "hello"); QDBusMessage subPathReply = con.call(subPathMessage , QDBus::Block, 5000); QCOMPARE(obj.callCount, 2); QCOMPARE(obj.lastMessage.service(), con.baseService()); QCOMPARE(obj.lastMessage.interface(), QString()); QCOMPARE(obj.lastMessage.path(), childPath); QCOMPARE(obj.replyArguments, subPathReply.arguments()); } QString MyObject::path; QTEST_MAIN(tst_QDBusConnection) #include "tst_qdbusconnection.moc"