// SPDX-FileCopyrightText: 2018 Kitsune Ral // SPDX-License-Identifier: LGPL-2.1-or-later #include "roomevent.h" #include "converters.h" #include "logging.h" #include "redactionevent.h" using namespace Quotient; [[maybe_unused]] static auto roomEventTypeInitialised = Event::factory_t::chainFactory(); RoomEvent::RoomEvent(Type type, event_mtype_t matrixType, const QJsonObject& contentJson) : Event(type, matrixType, contentJson) {} RoomEvent::RoomEvent(Type type, const QJsonObject& json) : Event(type, json) { if (const auto redaction = unsignedPart(RedactedCauseKeyL); redaction.isObject()) _redactedBecause = makeEvent(redaction.toObject()); } RoomEvent::~RoomEvent() = default; // Let the smart pointer do its job QString RoomEvent::id() const { return fullJson()[EventIdKeyL].toString(); } QDateTime RoomEvent::originTimestamp() const { return Quotient::fromJson(fullJson()["origin_server_ts"_ls]); } QString RoomEvent::roomId() const { return fullJson()[RoomIdKeyL].toString(); } QString RoomEvent::senderId() const { return fullJson()[SenderKeyL].toString(); } bool RoomEvent::isReplaced() const { return unsignedPart("m.relations"_ls).contains("m.replace"); } QString RoomEvent::replacedBy() const { // clang-format off return unsignedPart("m.relations"_ls) .value("m.replace"_ls).toObject() .value(EventIdKeyL).toString(); // clang-format on } QString RoomEvent::redactionReason() const { return isRedacted() ? _redactedBecause->reason() : QString {}; } QString RoomEvent::transactionId() const { return unsignedPart("transaction_id"_ls); } QString RoomEvent::stateKey() const { return fullJson()[StateKeyKeyL].toString(); } void RoomEvent::setRoomId(const QString& roomId) { editJson().insert(RoomIdKey, roomId); } void RoomEvent::setSender(const QString& senderId) { editJson().insert(SenderKey, senderId); } void RoomEvent::setTransactionId(const QString& txnId) { auto unsignedData = fullJson()[UnsignedKeyL].toObject(); unsignedData.insert(QStringLiteral("transaction_id"), txnId); editJson().insert(UnsignedKey, unsignedData); Q_ASSERT(transactionId() == txnId); } void RoomEvent::addId(const QString& newId) { Q_ASSERT(id().isEmpty()); Q_ASSERT(!newId.isEmpty()); editJson().insert(EventIdKey, newId); qCDebug(EVENTS) << "Event txnId -> id:" << transactionId() << "->" << id(); Q_ASSERT(id() == newId); } void RoomEvent::dumpTo(QDebug dbg) const { Event::dumpTo(dbg); dbg << " (made at " << originTimestamp().toString(Qt::ISODate) << ')'; } QJsonObject makeCallContentJson(const QString& callId, int version, QJsonObject content) { content.insert(QStringLiteral("call_id"), callId); content.insert(QStringLiteral("version"), version); return content; } CallEventBase::CallEventBase(Type type, event_mtype_t matrixType, const QString& callId, int version, const QJsonObject& contentJson) : RoomEvent(type, matrixType, makeCallContentJson(callId, version, contentJson)) {} CallEventBase::CallEventBase(Event::Type type, const QJsonObject& json) : RoomEvent(type, json) { if (callId().isEmpty()) qCWarning(EVENTS) << id() << "is a call event with an empty call id"; } void RoomEvent::setOriginalEvent(event_ptr_tt originalEvent) { _originalEvent = std::move(originalEvent); } const QJsonObject RoomEvent::encryptedJson() const { if(!_originalEvent) { return {}; } return _originalEvent->fullJson(); }