diff options
author | Kitsune Ral <Kitsune-Ral@users.sf.net> | 2019-10-21 20:26:16 +0900 |
---|---|---|
committer | Kitsune Ral <Kitsune-Ral@users.sf.net> | 2019-10-21 20:26:16 +0900 |
commit | b822e2e116c38c99fd9f05fbe0f3c0fd20103b29 (patch) | |
tree | b8687558e5352316ba38c17999f683a0b7431be3 /tests | |
parent | 9b588b30505a71da8004965364a42d24f6f2f16d (diff) | |
download | libquotient-b822e2e116c38c99fd9f05fbe0f3c0fd20103b29.tar.gz libquotient-b822e2e116c38c99fd9f05fbe0f3c0fd20103b29.zip |
Quotest: more work on test invocation
Tests should be private slots, not public slots; and they
are invoked through a QueuedConnection so that both sync
and async tests could be processed uniformly with clear code.
Diffstat (limited to 'tests')
-rw-r--r-- | tests/quotest.cpp | 98 |
1 files changed, 53 insertions, 45 deletions
diff --git a/tests/quotest.cpp b/tests/quotest.cpp index 81f2e1e0..12e19a97 100644 --- a/tests/quotest.cpp +++ b/tests/quotest.cpp @@ -46,6 +46,7 @@ private: }; using TestToken = QByteArray; // return value of QMetaMethod::name +Q_DECLARE_METATYPE(TestToken); // For now, the token itself is the test name but that may change. const char* testName(const TestToken& token) { return token.constData(); } @@ -77,6 +78,7 @@ public: TestSuite(Room* testRoom, QString source, TestManager* parent) : QObject(parent), targetRoom(testRoom), origin(std::move(source)) { + qRegisterMetaType<TestToken>(); Q_ASSERT(testRoom && parent); } @@ -84,6 +86,9 @@ signals: void finishedItem(QByteArray /*name*/, bool /*condition*/); public slots: + void doTest(const QByteArray& testName); + +private slots: TEST_DECL(loadMembers) TEST_DECL(sendMessage) TEST_DECL(sendReaction) @@ -98,13 +103,12 @@ public: Room* room() const { return targetRoom; } Connection* connection() const { return targetRoom->connection(); } -private slots: +private: bool checkFileSendingOutcome(const TestToken& thisTest, const QString& txnId, const QString& fileName); bool checkRedactionOutcome(const QByteArray& thisTest, const QString& evtIdToRedact); -private: bool validatePendingEvent(const QString& txnId); bool checkDirectChat() const; void finishTest(const TestToken& token, bool condition, const char* file, @@ -125,6 +129,13 @@ private: #define FAIL_TEST() FINISH_TEST(false) +void TestSuite::doTest(const QByteArray& testName) +{ + clog << "Starting: " << testName.constData() << endl; + QMetaObject::invokeMethod(this, testName, Qt::DirectConnection, + Q_ARG(TestToken, testName)); +} + bool TestSuite::validatePendingEvent(const QString& txnId) { auto it = targetRoom->findPendingEvent(txnId); @@ -211,17 +222,14 @@ void TestManager::setupAndRun() // and filled with some events so that other tests could use that connect(joinJob, &BaseJob::success, this, [this, joinJob] { testSuite = new TestSuite(c->room(joinJob->roomId()), origin, this); - testSuite->room()->getPreviousContent(); - connectSingleShot(testSuite->room(), &Room::addedMessages, this, [this] { - connectSingleShot(c, &Connection::syncDone, this, - &TestManager::doTests); - connect(c, &Connection::syncDone, testSuite, [this] { - clog << "Sync complete" << endl; - if (auto* r = testSuite->room()) - clog << "Test room timeline size = " << r->timelineSize() - << ", pending size = " << r->pendingEvents().size() - << endl; - }); + connectSingleShot(c, &Connection::syncDone, this, [this] { + if (testSuite->room()->timelineSize() > 0) + doTests(); + else { + testSuite->room()->getPreviousContent(); + connectSingleShot(testSuite->room(), &Room::addedMessages, this, + &TestManager::doTests); + } }); }); connect(joinJob, &BaseJob::failure, this, [this] { @@ -244,16 +252,24 @@ void TestManager::onNewRoom(Room* r) void TestManager::doTests() { - if (testSuite->room()->memberJoinState(c->user()) != JoinState::Join) { - clog << "Test room sanity check failed; after joining the test " - "user is still not in Join state" - << endl; - finalize(); - return; + const auto* metaObj = testSuite->metaObject(); + for (auto i = metaObj->methodOffset(); i < metaObj->methodCount(); ++i) { + const auto metaMethod = metaObj->method(i); + if (metaMethod.access() != QMetaMethod::Private + || metaMethod.methodType() != QMetaMethod::Slot) + continue; + + const auto testName = metaMethod.name(); + running.push_back(testName); + // Some tests return the result immediately, so queue everything + // so that we could process all tests asynchronously. + QMetaObject::invokeMethod(testSuite, "doTest", Qt::QueuedConnection, + Q_ARG(QByteArray, testName)); } - clog << "Starting tests" << endl; - // Some tests below are synchronous; in order to analyse all test outcomes - // uniformly, connect to the signal in advance. + clog << "Tests to do:"; + for (const auto& test: qAsConst(running)) + clog << " " << testName(test); + clog << endl; connect(testSuite, &TestSuite::finishedItem, this, [this](const QByteArray& itemName, bool condition) { if (auto i = running.indexOf(itemName); i != -1) @@ -261,32 +277,24 @@ void TestManager::doTests() else Q_ASSERT_X(false, itemName, "Test item is not in running state"); + if (running.empty()) { + clog << "All tests finished" << endl; + conclude(); + } }); - const auto* metaObj = testSuite->metaObject(); - for (auto i = metaObj->methodOffset(); i < metaObj->methodCount(); - ++i) { - const auto metaMethod = metaObj->method(i); - if (metaMethod.access() != QMetaMethod::Public - || metaMethod.methodType() != QMetaMethod::Slot) - continue; - - clog << "Starting: " << metaMethod.name().constData() << endl; - running.push_back(metaMethod.name()); - metaMethod.invoke(testSuite, Qt::DirectConnection, - Q_ARG(QByteArray, metaMethod.name())); - } - // By now, sync tests have all completed; catch up with async ones - connect(testSuite, &TestSuite::finishedItem, this, [this] { - if (running.empty()) { - clog << "All tests finished" << endl; - conclude(); - return; + connect(c, &Connection::syncDone, this, [this] { + static int i = 0; + clog << "Sync " << ++i << " complete" << endl; + if (auto* r = testSuite->room()) + clog << "Test room timeline size = " << r->timelineSize() + << ", pending size = " << r->pendingEvents().size() << endl; + if (!running.empty()) { + clog << running.size() << " test(s) in the air:"; + for (const auto& test: qAsConst(running)) + clog << " " << testName(test); + clog << endl; } - clog << running.size() << " test(s) in the air:"; - for (const auto& test: qAsConst(running)) - clog << " " << testName(test); - clog << endl; }); } |