aboutsummaryrefslogtreecommitdiff
path: root/tests
diff options
context:
space:
mode:
authorKitsune Ral <Kitsune-Ral@users.sf.net>2019-10-21 20:26:16 +0900
committerKitsune Ral <Kitsune-Ral@users.sf.net>2019-10-21 20:26:16 +0900
commitb822e2e116c38c99fd9f05fbe0f3c0fd20103b29 (patch)
treeb8687558e5352316ba38c17999f683a0b7431be3 /tests
parent9b588b30505a71da8004965364a42d24f6f2f16d (diff)
downloadlibquotient-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.cpp98
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;
});
}