aboutsummaryrefslogtreecommitdiff
path: root/examples/qmc-example.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'examples/qmc-example.cpp')
-rw-r--r--examples/qmc-example.cpp227
1 files changed, 171 insertions, 56 deletions
diff --git a/examples/qmc-example.cpp b/examples/qmc-example.cpp
index 652c1f92..8fbf4824 100644
--- a/examples/qmc-example.cpp
+++ b/examples/qmc-example.cpp
@@ -9,6 +9,8 @@
#include <QtCore/QCoreApplication>
#include <QtCore/QStringBuilder>
#include <QtCore/QTimer>
+#include <QtCore/QTemporaryFile>
+#include <QtCore/QFileInfo>
#include <iostream>
#include <functional>
@@ -20,7 +22,7 @@ using namespace std::placeholders;
class QMCTest : public QObject
{
public:
- QMCTest(Connection* conn, QString targetRoomName, QString source);
+ QMCTest(Connection* conn, QString testRoomName, QString source);
private slots:
void setupAndRun();
@@ -29,10 +31,13 @@ class QMCTest : public QObject
void doTests();
void loadMembers();
void sendMessage();
+ void sendFile();
+ void checkFileSendingOutcome(const QString& txnId,
+ const QString& fileName);
void addAndRemoveTag();
void sendAndRedact();
void checkRedactionOutcome(const QString& evtIdToRedact,
- RoomEventsRange events);
+ const QMetaObject::Connection& sc);
void markDirectChat();
void checkDirectChatOutcome(
const Connection::DirectChatsMap& added);
@@ -47,6 +52,8 @@ class QMCTest : public QObject
QString origin;
QString targetRoomName;
Room* targetRoom = nullptr;
+
+ bool validatePendingEvent(const QString& txnId);
};
#define QMC_CHECK(description, condition) \
@@ -69,6 +76,14 @@ class QMCTest : public QObject
} \
}
+bool QMCTest::validatePendingEvent(const QString& txnId)
+{
+ auto it = targetRoom->findPendingEvent(txnId);
+ return it != targetRoom->pendingEvents().end() &&
+ it->deliveryStatus() == EventStatus::Submitted &&
+ (*it)->transactionId() == txnId;
+}
+
QMCTest::QMCTest(Connection* conn, QString testRoomName, QString source)
: c(conn), origin(std::move(source)), targetRoomName(std::move(testRoomName))
{
@@ -142,7 +157,8 @@ void QMCTest::run()
{
// TODO: Waiting for proper futures to come so that it could be:
// targetRoom->postPlainText(origin % ": All tests finished")
-// .then(this, &QMCTest::leave);
+// .then(this, &QMCTest::leave); // Qt-style
+// .then([this] { leave(); }); // STL-style
auto txnId =
targetRoom->postPlainText(origin % ": All tests finished");
connect(targetRoom, &Room::messageSent, this,
@@ -166,6 +182,7 @@ void QMCTest::doTests()
return;
sendMessage();
+ sendFile();
addAndRemoveTag();
sendAndRedact();
markDirectChat();
@@ -206,19 +223,133 @@ void QMCTest::sendMessage()
running.push_back("Message sending");
cout << "Sending a message" << endl;
auto txnId = targetRoom->postPlainText("Hello, " % origin % " is here");
- auto& pending = targetRoom->pendingEvents();
- if (pending.empty())
+ if (!validatePendingEvent(txnId))
{
+ cout << "Invalid pending event right after submitting" << endl;
QMC_CHECK("Message sending", false);
return;
}
- auto it = std::find_if(pending.begin(), pending.end(),
- [&txnId] (const auto& e) {
- return e->transactionId() == txnId;
+
+ QMetaObject::Connection sc;
+ sc = connect(targetRoom, &Room::pendingEventAboutToMerge, this,
+ [this,sc,txnId] (const RoomEvent* evt, int pendingIdx) {
+ const auto& pendingEvents = targetRoom->pendingEvents();
+ Q_ASSERT(pendingIdx >= 0 && pendingIdx < int(pendingEvents.size()));
+
+ if (evt->transactionId() != txnId)
+ return;
+
+ disconnect(sc);
+
+ QMC_CHECK("Message sending",
+ is<RoomMessageEvent>(*evt) && !evt->id().isEmpty() &&
+ pendingEvents[size_t(pendingIdx)]->transactionId()
+ == evt->transactionId());
+ });
+}
+
+void QMCTest::sendFile()
+{
+ running.push_back("File sending");
+ cout << "Sending a file" << endl;
+ auto* tf = new QTemporaryFile;
+ if (!tf->open())
+ {
+ cout << "Failed to create a temporary file" << endl;
+ QMC_CHECK("File sending", false);
+ return;
+ }
+ tf->write("Test");
+ tf->close();
+ // QFileInfo::fileName brings only the file name; QFile::fileName brings
+ // the full path
+ const auto tfName = QFileInfo(*tf).fileName();
+ cout << "Sending file" << tfName.toStdString() << endl;
+ const auto txnId = targetRoom->postFile("Test file",
+ QUrl::fromLocalFile(tf->fileName()));
+ if (!validatePendingEvent(txnId))
+ {
+ cout << "Invalid pending event right after submitting" << endl;
+ QMC_CHECK("File sending", false);
+ delete tf;
+ return;
+ }
+
+ QMetaObject::Connection scCompleted, scFailed;
+ scCompleted = connect(targetRoom, &Room::fileTransferCompleted, this,
+ [this,txnId,tf,tfName,scCompleted,scFailed] (const QString& id) {
+ auto fti = targetRoom->fileTransferInfo(id);
+ Q_ASSERT(fti.status == FileTransferInfo::Completed);
+
+ if (id != txnId)
+ return;
+
+ disconnect(scCompleted);
+ disconnect(scFailed);
+ delete tf;
+
+ checkFileSendingOutcome(txnId, tfName);
+ });
+ scFailed = connect(targetRoom, &Room::fileTransferFailed, this,
+ [this,txnId,tf,scCompleted,scFailed]
+ (const QString& id, const QString& error) {
+ if (id != txnId)
+ return;
+
+ targetRoom->postPlainText(origin % ": File upload failed: " % error);
+ disconnect(scCompleted);
+ disconnect(scFailed);
+ delete tf;
+
+ QMC_CHECK("File sending", false);
+ });
+}
+
+void QMCTest::checkFileSendingOutcome(const QString& txnId,
+ const QString& fileName)
+{
+ auto it = targetRoom->findPendingEvent(txnId);
+ if (it == targetRoom->pendingEvents().end())
+ {
+ cout << "Pending file event dropped before upload completion"
+ << endl;
+ QMC_CHECK("File sending", false);
+ return;
+ }
+ if (it->deliveryStatus() != EventStatus::FileUploaded)
+ {
+ cout << "Pending file event status upon upload completion is "
+ << it->deliveryStatus() << " != FileUploaded("
+ << EventStatus::FileUploaded << ')' << endl;
+ QMC_CHECK("File sending", false);
+ return;
+ }
+
+ QMetaObject::Connection sc;
+ sc = connect(targetRoom, &Room::pendingEventAboutToMerge, this,
+ [this,sc,txnId,fileName] (const RoomEvent* evt, int pendingIdx) {
+ const auto& pendingEvents = targetRoom->pendingEvents();
+ Q_ASSERT(pendingIdx >= 0 && pendingIdx < int(pendingEvents.size()));
+
+ if (evt->transactionId() != txnId)
+ return;
+
+ cout << "Event " << txnId.toStdString()
+ << " arrived in the timeline" << endl;
+ disconnect(sc);
+ visit(*evt,
+ [&] (const RoomMessageEvent& e) {
+ QMC_CHECK("File sending",
+ !e.id().isEmpty() &&
+ pendingEvents[size_t(pendingIdx)]
+ ->transactionId() == txnId &&
+ e.hasFileContent() &&
+ e.content()->fileInfo()->originalName == fileName);
+ },
+ [this] (const RoomEvent&) {
+ QMC_CHECK("File sending", false);
});
- QMC_CHECK("Message sending", it != pending.end());
- // TODO: Wait when it actually gets sent; check that it obtained an id
- // Independently, check when it shows up in the timeline.
+ });
}
void QMCTest::addAndRemoveTag()
@@ -250,70 +381,54 @@ void QMCTest::sendAndRedact()
{
running.push_back("Redaction");
cout << "Sending a message to redact" << endl;
- if (auto* job = targetRoom->connection()->sendMessage(targetRoom->id(),
- RoomMessageEvent(origin % ": message to redact")))
+ auto txnId = targetRoom->postPlainText(origin % ": message to redact");
+ if (txnId.isEmpty())
{
- connect(job, &BaseJob::success, targetRoom, [job,this] {
+ QMC_CHECK("Redaction", false);
+ return;
+ }
+ connect(targetRoom, &Room::messageSent, this,
+ [this,txnId] (const QString& tId, const QString& evtId) {
+ if (tId != txnId)
+ return;
+
cout << "Redacting the message" << endl;
- targetRoom->redactEvent(job->eventId(), origin);
- // Make sure to save the event id because the job is about to end.
- connect(targetRoom, &Room::aboutToAddNewMessages, this,
- std::bind(&QMCTest::checkRedactionOutcome,
- this, job->eventId(), _1));
+ targetRoom->redactEvent(evtId, origin);
+ QMetaObject::Connection sc;
+ sc = connect(targetRoom, &Room::addedMessages, this,
+ [this,sc,evtId] { checkRedactionOutcome(evtId, sc); });
});
- } else
- QMC_CHECK("Redaction", false);
}
void QMCTest::checkRedactionOutcome(const QString& evtIdToRedact,
- RoomEventsRange events)
+ const QMetaObject::Connection& sc)
{
- static bool checkSucceeded = false;
// There are two possible (correct) outcomes: either the event comes already
// redacted at the next sync, or the nearest sync completes with
// the unredacted event but the next one brings redaction.
- auto it = std::find_if(events.begin(), events.end(),
- [=] (const RoomEventPtr& e) {
- return e->id() == evtIdToRedact;
- });
- if (it == events.end())
+ auto it = targetRoom->findInTimeline(evtIdToRedact);
+ if (it == targetRoom->timelineEdge())
return; // Waiting for the next sync
if ((*it)->isRedacted())
{
- if (checkSucceeded)
- {
- const auto msg =
- "The redacted event came in with the sync again, ignoring";
- cout << msg << endl;
- targetRoom->postPlainText(msg);
- return;
- }
cout << "The sync brought already redacted message" << endl;
QMC_CHECK("Redaction", true);
- // Not disconnecting because there are other connections from this class
- // to aboutToAddNewMessages
- checkSucceeded = true;
+ disconnect(sc);
return;
}
- // The event is not redacted
- if (checkSucceeded)
- {
- const auto msg =
- "Warning: the redacted event came non-redacted with the sync!";
- cout << msg << endl;
- targetRoom->postPlainText(msg);
- }
- cout << "Message came non-redacted with the sync, waiting for redaction" << endl;
- connect(targetRoom, &Room::replacedEvent, targetRoom,
- [=] (const RoomEvent* newEvent, const RoomEvent* oldEvent) {
- QMC_CHECK("Redaction", oldEvent->id() == evtIdToRedact &&
- newEvent->isRedacted() &&
- newEvent->redactionReason() == origin);
- checkSucceeded = true;
- disconnect(targetRoom, &Room::replacedEvent, nullptr, nullptr);
+ cout << "Message came non-redacted with the sync, waiting for redaction"
+ << endl;
+ connect(targetRoom, &Room::replacedEvent, this,
+ [this,evtIdToRedact]
+ (const RoomEvent* newEvent, const RoomEvent* oldEvent) {
+ if (oldEvent->id() == evtIdToRedact)
+ {
+ QMC_CHECK("Redaction", newEvent->isRedacted() &&
+ newEvent->redactionReason() == origin);
+ disconnect(targetRoom, &Room::replacedEvent, nullptr, nullptr);
+ }
});
-
}
void QMCTest::markDirectChat()