#include "connection.h" #include "room.h" #include "user.h" #include "jobs/sendeventjob.h" #include #include #include #include using namespace QMatrixClient; using std::cout; using std::endl; using namespace std::placeholders; class QMCTest : public QObject { public: QMCTest(Connection* conn, const QString& testRoomName, QString source); private slots: void onNewRoom(Room* r, const QString& testRoomName); void doTests(); void addAndRemoveTag(); void sendAndRedact(); void finalize(); private: QScopedPointer c; QString origin; Room* targetRoom = nullptr; int semaphor = 0; }; #define QMC_CHECK(description, condition) \ cout << (description) \ << (!!(condition) ? " successul" : " FAILED") << endl; \ targetRoom->postMessage(origin % ": " % QStringLiteral(description) % \ (!!(condition) ? QStringLiteral(" successful") : \ QStringLiteral(" FAILED")), MessageEventType::Notice) QMCTest::QMCTest(Connection* conn, const QString& testRoomName, QString source) : c(conn), origin(std::move(source)) { if (!origin.isEmpty()) cout << "Origin for the test message: " << origin.toStdString() << endl; if (!testRoomName.isEmpty()) cout << "Test room name: " << testRoomName.toStdString() << endl; connect(c.data(), &Connection::newRoom, this, [this,testRoomName] (Room* r) { onNewRoom(r, testRoomName); }); connect(c.data(), &Connection::syncDone, c.data(), [this] { cout << "Sync complete, " << semaphor << " tests in the air" << endl; if (semaphor) { // if (targetRoom) // targetRoom->postMessage( // QString("%1: sync done, %2 test(s) in the air") // .arg(origin).arg(semaphor), // MessageEventType::Notice); c->sync(10000); } else { if (targetRoom) { auto j = c->callApi(targetRoom->id(), RoomMessageEvent(origin % ": All tests finished")); connect(j, &BaseJob::finished, this, &QMCTest::finalize); } else finalize(); } }); // Big countdown watchdog QTimer::singleShot(180000, this, &QMCTest::finalize); } void QMCTest::onNewRoom(Room* r, const QString& testRoomName) { cout << "New room: " << r->id().toStdString() << endl; connect(r, &Room::namesChanged, this, [=] { cout << "Room " << r->id().toStdString() << ", name(s) changed:" << endl << " Name: " << r->name().toStdString() << endl << " Canonical alias: " << r->canonicalAlias().toStdString() << endl << endl << endl; if (!testRoomName.isEmpty() && (r->name() == testRoomName || r->canonicalAlias() == testRoomName)) { cout << "Found the target room, proceeding for tests" << endl; targetRoom = r; ++semaphor; auto j = targetRoom->connection()->callApi( targetRoom->id(), RoomMessageEvent(origin % ": connected to test room", MessageEventType::Notice)); connect(j, &BaseJob::success, this, [this] { doTests(); --semaphor; }); } }); connect(r, &Room::aboutToAddNewMessages, r, [r] (RoomEventsRange timeline) { cout << timeline.size() << " new event(s) in room " << r->id().toStdString() << endl; // for (const auto& item: timeline) // { // cout << "From: " // << r->roomMembername(item->senderId()).toStdString() // << endl << "Timestamp:" // << item->timestamp().toString().toStdString() << endl // << "JSON:" << endl << item->originalJson().toStdString() << endl; // } }); } void QMCTest::doTests() { addAndRemoveTag(); sendAndRedact(); } void QMCTest::addAndRemoveTag() { ++semaphor; static const auto TestTag = QStringLiteral("org.qmatrixclient.test"); // Pre-requisite if (targetRoom->tags().contains(TestTag)) targetRoom->removeTag(TestTag); QObject::connect(targetRoom, &Room::tagsChanged, targetRoom, [=] { cout << "Room " << targetRoom->id().toStdString() << ", tag(s) changed:" << endl << " " << targetRoom->tagNames().join(", ").toStdString() << endl; if (targetRoom->tags().contains(TestTag)) { cout << "Test tag set, removing it now" << endl; targetRoom->removeTag(TestTag); QMC_CHECK("Tagging test", !targetRoom->tags().contains(TestTag)); --semaphor; QObject::disconnect(targetRoom, &Room::tagsChanged, nullptr, nullptr); } }); // The reverse order because tagsChanged is emitted synchronously. cout << "Adding a tag" << endl; targetRoom->addTag(TestTag); } void QMCTest::sendAndRedact() { ++semaphor; cout << "Sending a message to redact" << endl; auto* job = targetRoom->connection()->callApi(targetRoom->id(), RoomMessageEvent(origin % ": Message to redact")); QObject::connect(job, &BaseJob::success, targetRoom, [job,this] { cout << "Message to redact has been succesfully sent, redacting" << endl; targetRoom->redactEvent(job->eventId(), "qmc-example"); }); QObject::connect(targetRoom, &Room::replacedEvent, targetRoom, [=] (const RoomEvent* newEvent) { QMC_CHECK("Redaction", newEvent->isRedacted() && newEvent->redactionReason() == "qmc-example"); --semaphor; QObject::disconnect(targetRoom, &Room::replacedEvent, nullptr, nullptr); }); } void QMCTest::finalize() { if (semaphor) cout << "One or more tests FAILED" << endl; cout << "Logging out" << endl; c->logout(); connect(c.data(), &Connection::loggedOut, QCoreApplication::instance(), [this] { c->deleteLater(); QCoreApplication::processEvents(); QCoreApplication::exit(semaphor); }); } int main(int argc, char* argv[]) { QCoreApplication app(argc, argv); if (argc < 4) { cout << "Usage: qmc-example [ [origin]]" << endl; return -1; } cout << "Connecting to the server as " << argv[1] << endl; auto conn = new Connection; conn->connectToServer(argv[1], argv[2], argv[3]); QObject::connect(conn, &Connection::connected, [=] { cout << "Connected, server: " << conn->homeserver().toDisplayString().toStdString() << endl; cout << "Access token: " << conn->accessToken().toStdString() << endl; conn->sync(); }); QMCTest test { conn, argc >= 5 ? argv[4] : nullptr, argc >= 6 ? argv[5] : nullptr }; return app.exec(); }