2021-03-05 02:35:15 +03:00
|
|
|
// SPDX-FileCopyrightText: 2017 Konstantinos Sideris <siderisk@auth.gr>
|
|
|
|
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2018-05-16 23:30:50 +03:00
|
|
|
#include <limits>
|
2017-07-29 11:49:00 +03:00
|
|
|
#include <stdexcept>
|
2019-12-14 19:08:36 +03:00
|
|
|
#include <variant>
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2017-12-22 01:00:48 +03:00
|
|
|
#include <QByteArray>
|
2019-06-27 21:53:44 +03:00
|
|
|
#include <QCoreApplication>
|
2020-11-10 05:28:41 +03:00
|
|
|
#include <QCryptographicHash>
|
2017-07-29 11:49:00 +03:00
|
|
|
#include <QFile>
|
2018-04-21 16:34:50 +03:00
|
|
|
#include <QHash>
|
2019-08-13 05:09:40 +03:00
|
|
|
#include <QMap>
|
2017-07-29 11:49:00 +03:00
|
|
|
#include <QStandardPaths>
|
|
|
|
|
2020-12-18 05:21:17 +03:00
|
|
|
#if __has_include(<keychain.h>)
|
|
|
|
#include <keychain.h>
|
|
|
|
#else
|
2020-12-17 00:10:09 +03:00
|
|
|
#include <qt5keychain/keychain.h>
|
2020-12-18 05:21:17 +03:00
|
|
|
#endif
|
2020-12-17 00:10:09 +03:00
|
|
|
|
2018-06-28 16:16:43 +03:00
|
|
|
#include <mtx/responses/common.hpp>
|
2018-03-26 20:41:16 +03:00
|
|
|
|
2017-07-29 11:49:00 +03:00
|
|
|
#include "Cache.h"
|
2019-12-15 04:56:04 +03:00
|
|
|
#include "Cache_p.h"
|
2020-07-06 19:02:21 +03:00
|
|
|
#include "ChatPage.h"
|
2020-05-30 17:37:51 +03:00
|
|
|
#include "EventAccessors.h"
|
2020-01-31 18:36:58 +03:00
|
|
|
#include "Logging.h"
|
2020-07-06 19:02:21 +03:00
|
|
|
#include "MatrixClient.h"
|
2020-08-18 00:30:36 +03:00
|
|
|
#include "Olm.h"
|
2020-11-10 05:28:41 +03:00
|
|
|
#include "UserSettingsPage.h"
|
2018-04-29 15:42:40 +03:00
|
|
|
#include "Utils.h"
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2018-04-21 16:34:50 +03:00
|
|
|
//! Should be changed when a breaking change occurs in the cache format.
|
|
|
|
//! This will reset client's data.
|
2020-10-20 14:46:05 +03:00
|
|
|
static const std::string CURRENT_CACHE_FORMAT_VERSION("2020.10.20");
|
2018-06-10 20:03:45 +03:00
|
|
|
static const std::string SECRET("secret");
|
2017-12-10 13:51:44 +03:00
|
|
|
|
2021-03-03 02:08:33 +03:00
|
|
|
//! Keys used for the DB
|
2021-03-03 01:15:12 +03:00
|
|
|
static const std::string_view NEXT_BATCH_KEY("next_batch");
|
|
|
|
static const std::string_view OLM_ACCOUNT_KEY("olm_account");
|
|
|
|
static const std::string_view CACHE_FORMAT_VERSION_KEY("cache_format_version");
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2020-02-24 03:07:25 +03:00
|
|
|
constexpr size_t MAX_RESTORED_MESSAGES = 30'000;
|
2018-06-28 16:16:43 +03:00
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
constexpr auto DB_SIZE = 32ULL * 1024ULL * 1024ULL * 1024ULL; // 32 GB
|
2020-11-26 02:19:09 +03:00
|
|
|
constexpr auto MAX_DBS = 32384UL;
|
2020-07-05 06:29:07 +03:00
|
|
|
constexpr auto BATCH_SIZE = 100;
|
2018-08-25 21:08:43 +03:00
|
|
|
|
2018-04-21 16:34:50 +03:00
|
|
|
//! Cache databases and their format.
|
|
|
|
//!
|
|
|
|
//! Contains UI information for the joined rooms. (i.e name, topic, avatar url etc).
|
|
|
|
//! Format: room_id -> RoomInfo
|
2018-06-10 20:03:45 +03:00
|
|
|
constexpr auto ROOMS_DB("rooms");
|
|
|
|
constexpr auto INVITES_DB("invites");
|
2018-04-21 16:34:50 +03:00
|
|
|
//! Information that must be kept between sync requests.
|
2018-06-10 20:03:45 +03:00
|
|
|
constexpr auto SYNC_STATE_DB("sync_state");
|
2018-04-21 16:34:50 +03:00
|
|
|
//! Read receipts per room/event.
|
2018-06-10 20:03:45 +03:00
|
|
|
constexpr auto READ_RECEIPTS_DB("read_receipts");
|
|
|
|
constexpr auto NOTIFICATIONS_DB("sent_notifications");
|
|
|
|
|
|
|
|
//! Encryption related databases.
|
|
|
|
|
|
|
|
//! user_id -> list of devices
|
|
|
|
constexpr auto DEVICES_DB("devices");
|
|
|
|
//! device_id -> device keys
|
|
|
|
constexpr auto DEVICE_KEYS_DB("device_keys");
|
|
|
|
//! room_ids that have encryption enabled.
|
2018-06-12 09:45:26 +03:00
|
|
|
constexpr auto ENCRYPTED_ROOMS_DB("encrypted_rooms");
|
2018-06-10 20:03:45 +03:00
|
|
|
|
2018-06-15 01:35:31 +03:00
|
|
|
//! room_id -> pickled OlmInboundGroupSession
|
2018-06-10 20:03:45 +03:00
|
|
|
constexpr auto INBOUND_MEGOLM_SESSIONS_DB("inbound_megolm_sessions");
|
|
|
|
//! MegolmSessionIndex -> pickled OlmOutboundGroupSession
|
|
|
|
constexpr auto OUTBOUND_MEGOLM_SESSIONS_DB("outbound_megolm_sessions");
|
2018-04-21 16:34:50 +03:00
|
|
|
|
2018-01-15 00:33:12 +03:00
|
|
|
using CachedReceipts = std::multimap<uint64_t, std::string, std::greater<uint64_t>>;
|
|
|
|
using Receipts = std::map<std::string, std::map<std::string, uint64_t>>;
|
|
|
|
|
2019-12-15 01:39:02 +03:00
|
|
|
Q_DECLARE_METATYPE(RoomMember)
|
|
|
|
Q_DECLARE_METATYPE(mtx::responses::Timeline)
|
|
|
|
Q_DECLARE_METATYPE(RoomSearchResult)
|
|
|
|
Q_DECLARE_METATYPE(RoomInfo)
|
2020-10-02 02:14:42 +03:00
|
|
|
Q_DECLARE_METATYPE(mtx::responses::QueryKeys)
|
2019-12-15 01:39:02 +03:00
|
|
|
|
2018-05-08 20:30:09 +03:00
|
|
|
namespace {
|
2018-05-11 13:41:46 +03:00
|
|
|
std::unique_ptr<Cache> instance_ = nullptr;
|
2018-05-08 20:30:09 +03:00
|
|
|
}
|
|
|
|
|
2021-03-15 19:11:02 +03:00
|
|
|
template<class T>
|
|
|
|
bool
|
|
|
|
containsStateUpdates(const T &e)
|
|
|
|
{
|
|
|
|
return std::visit([](const auto &ev) { return Cache::isStateEvent(ev); }, e);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
containsStateUpdates(const mtx::events::collections::StrippedEvents &e)
|
|
|
|
{
|
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
|
|
|
return std::holds_alternative<StrippedEvent<state::Avatar>>(e) ||
|
|
|
|
std::holds_alternative<StrippedEvent<CanonicalAlias>>(e) ||
|
|
|
|
std::holds_alternative<StrippedEvent<Name>>(e) ||
|
|
|
|
std::holds_alternative<StrippedEvent<Member>>(e) ||
|
|
|
|
std::holds_alternative<StrippedEvent<Topic>>(e);
|
|
|
|
}
|
|
|
|
|
2020-08-27 22:49:05 +03:00
|
|
|
bool
|
|
|
|
Cache::isHiddenEvent(lmdb::txn &txn,
|
|
|
|
mtx::events::collections::TimelineEvents e,
|
|
|
|
const std::string &room_id)
|
2020-08-18 00:30:36 +03:00
|
|
|
{
|
|
|
|
using namespace mtx::events;
|
2021-01-27 04:45:33 +03:00
|
|
|
|
|
|
|
// Always hide edits
|
|
|
|
if (mtx::accessors::relations(e).replaces())
|
|
|
|
return true;
|
|
|
|
|
2020-08-18 00:30:36 +03:00
|
|
|
if (auto encryptedEvent = std::get_if<EncryptedEvent<msg::Encrypted>>(&e)) {
|
|
|
|
MegolmSessionIndex index;
|
|
|
|
index.room_id = room_id;
|
|
|
|
index.session_id = encryptedEvent->content.session_id;
|
|
|
|
index.sender_key = encryptedEvent->content.sender_key;
|
|
|
|
|
|
|
|
auto result = olm::decryptEvent(index, *encryptedEvent);
|
|
|
|
if (!result.error)
|
|
|
|
e = result.event.value();
|
|
|
|
}
|
|
|
|
|
2020-08-27 22:49:05 +03:00
|
|
|
mtx::events::account_data::nheko_extensions::HiddenEvents hiddenEvents;
|
|
|
|
hiddenEvents.hidden_event_types = {
|
2020-08-18 00:30:36 +03:00
|
|
|
EventType::Reaction, EventType::CallCandidates, EventType::Unsupported};
|
|
|
|
|
2020-08-27 22:49:05 +03:00
|
|
|
if (auto temp = getAccountData(txn, mtx::events::EventType::NhekoHiddenEvents, ""))
|
2021-01-06 00:10:40 +03:00
|
|
|
hiddenEvents =
|
|
|
|
std::move(std::get<mtx::events::AccountDataEvent<
|
|
|
|
mtx::events::account_data::nheko_extensions::HiddenEvents>>(*temp)
|
|
|
|
.content);
|
2020-08-27 22:49:05 +03:00
|
|
|
if (auto temp = getAccountData(txn, mtx::events::EventType::NhekoHiddenEvents, room_id))
|
2021-01-06 00:10:40 +03:00
|
|
|
hiddenEvents =
|
|
|
|
std::move(std::get<mtx::events::AccountDataEvent<
|
|
|
|
mtx::events::account_data::nheko_extensions::HiddenEvents>>(*temp)
|
|
|
|
.content);
|
2020-08-27 22:49:05 +03:00
|
|
|
|
2020-08-18 00:30:36 +03:00
|
|
|
return std::visit(
|
2020-08-27 22:49:05 +03:00
|
|
|
[hiddenEvents](const auto &ev) {
|
|
|
|
return std::any_of(hiddenEvents.hidden_event_types.begin(),
|
|
|
|
hiddenEvents.hidden_event_types.end(),
|
2020-08-18 00:30:36 +03:00
|
|
|
[ev](EventType type) { return type == ev.type; });
|
|
|
|
},
|
|
|
|
e);
|
|
|
|
}
|
|
|
|
|
2018-01-21 22:43:21 +03:00
|
|
|
Cache::Cache(const QString &userId, QObject *parent)
|
|
|
|
: QObject{parent}
|
|
|
|
, env_{nullptr}
|
2018-04-21 16:34:50 +03:00
|
|
|
, localUserId_{userId}
|
2018-06-12 20:36:16 +03:00
|
|
|
{
|
|
|
|
setup();
|
2020-10-02 02:14:42 +03:00
|
|
|
connect(this, &Cache::userKeysUpdate, this, &Cache::updateUserKeys, Qt::QueuedConnection);
|
2018-06-12 20:36:16 +03:00
|
|
|
}
|
2017-10-03 21:16:31 +03:00
|
|
|
|
|
|
|
void
|
|
|
|
Cache::setup()
|
|
|
|
{
|
2020-12-28 00:56:43 +03:00
|
|
|
auto settings = UserSettings::instance();
|
2017-10-03 21:16:31 +03:00
|
|
|
|
2020-11-10 05:28:41 +03:00
|
|
|
nhlog::db()->debug("setting up cache");
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2021-01-19 21:44:22 +03:00
|
|
|
// Previous location of the cache directory
|
|
|
|
auto oldCache = QString("%1/%2%3")
|
|
|
|
.arg(QStandardPaths::writableLocation(QStandardPaths::CacheLocation))
|
|
|
|
.arg(QString::fromUtf8(localUserId_.toUtf8().toHex()))
|
|
|
|
.arg(QString::fromUtf8(settings->profile().toUtf8().toHex()));
|
|
|
|
|
2020-11-10 05:28:41 +03:00
|
|
|
cacheDirectory_ = QString("%1/%2%3")
|
2021-01-19 21:44:22 +03:00
|
|
|
.arg(QStandardPaths::writableLocation(QStandardPaths::AppDataLocation))
|
2020-12-25 21:20:15 +03:00
|
|
|
.arg(QString::fromUtf8(localUserId_.toUtf8().toHex()))
|
2020-12-28 00:56:43 +03:00
|
|
|
.arg(QString::fromUtf8(settings->profile().toUtf8().toHex()));
|
2017-10-03 21:16:31 +03:00
|
|
|
|
2020-11-10 05:28:41 +03:00
|
|
|
bool isInitial = !QFile::exists(cacheDirectory_);
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2021-01-19 21:44:22 +03:00
|
|
|
// NOTE: If both cache directories exist it's better to do nothing: it
|
|
|
|
// could mean a previous migration failed or was interrupted.
|
|
|
|
bool needsMigration = isInitial && QFile::exists(oldCache);
|
|
|
|
|
|
|
|
if (needsMigration) {
|
|
|
|
nhlog::db()->info("found old state directory, migrating");
|
|
|
|
if (!QDir().rename(oldCache, cacheDirectory_)) {
|
|
|
|
throw std::runtime_error(("Unable to migrate the old state directory (" +
|
|
|
|
oldCache + ") to the new location (" +
|
|
|
|
cacheDirectory_ + ")")
|
|
|
|
.toStdString()
|
|
|
|
.c_str());
|
|
|
|
}
|
|
|
|
nhlog::db()->info("completed state migration");
|
|
|
|
}
|
|
|
|
|
2017-08-26 13:49:16 +03:00
|
|
|
env_ = lmdb::env::create();
|
2018-08-25 21:08:43 +03:00
|
|
|
env_.set_mapsize(DB_SIZE);
|
|
|
|
env_.set_max_dbs(MAX_DBS);
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2017-08-26 13:49:16 +03:00
|
|
|
if (isInitial) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->info("initializing LMDB");
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2020-11-10 05:28:41 +03:00
|
|
|
if (!QDir().mkpath(cacheDirectory_)) {
|
2017-08-26 13:49:16 +03:00
|
|
|
throw std::runtime_error(
|
2020-12-25 21:20:15 +03:00
|
|
|
("Unable to create state directory:" + cacheDirectory_)
|
|
|
|
.toStdString()
|
|
|
|
.c_str());
|
2017-08-26 13:49:16 +03:00
|
|
|
}
|
|
|
|
}
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2017-08-26 13:49:16 +03:00
|
|
|
try {
|
2020-08-26 00:05:20 +03:00
|
|
|
// NOTE(Nico): We may want to use (MDB_MAPASYNC | MDB_WRITEMAP) in the future, but
|
|
|
|
// it can really mess up our database, so we shouldn't. For now, hopefully
|
|
|
|
// NOMETASYNC is fast enough.
|
2020-11-10 05:28:41 +03:00
|
|
|
env_.open(cacheDirectory_.toStdString().c_str(), MDB_NOMETASYNC | MDB_NOSYNC);
|
2017-08-26 13:49:16 +03:00
|
|
|
} catch (const lmdb::error &e) {
|
|
|
|
if (e.code() != MDB_VERSION_MISMATCH && e.code() != MDB_INVALID) {
|
|
|
|
throw std::runtime_error("LMDB initialization failed" +
|
|
|
|
std::string(e.what()));
|
|
|
|
}
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->warn("resetting cache due to LMDB version mismatch: {}", e.what());
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2020-11-10 05:28:41 +03:00
|
|
|
QDir stateDir(cacheDirectory_);
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2017-08-26 13:49:16 +03:00
|
|
|
for (const auto &file : stateDir.entryList(QDir::NoDotAndDotDot)) {
|
|
|
|
if (!stateDir.remove(file))
|
|
|
|
throw std::runtime_error(
|
|
|
|
("Unable to delete file " + file).toStdString().c_str());
|
|
|
|
}
|
2020-11-10 05:28:41 +03:00
|
|
|
env_.open(cacheDirectory_.toStdString().c_str());
|
2017-08-26 13:49:16 +03:00
|
|
|
}
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2018-05-05 16:38:41 +03:00
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
syncStateDb_ = lmdb::dbi::open(txn, SYNC_STATE_DB, MDB_CREATE);
|
|
|
|
roomsDb_ = lmdb::dbi::open(txn, ROOMS_DB, MDB_CREATE);
|
|
|
|
invitesDb_ = lmdb::dbi::open(txn, INVITES_DB, MDB_CREATE);
|
|
|
|
readReceiptsDb_ = lmdb::dbi::open(txn, READ_RECEIPTS_DB, MDB_CREATE);
|
|
|
|
notificationsDb_ = lmdb::dbi::open(txn, NOTIFICATIONS_DB, MDB_CREATE);
|
2018-06-10 20:03:45 +03:00
|
|
|
|
|
|
|
// Device management
|
|
|
|
devicesDb_ = lmdb::dbi::open(txn, DEVICES_DB, MDB_CREATE);
|
|
|
|
deviceKeysDb_ = lmdb::dbi::open(txn, DEVICE_KEYS_DB, MDB_CREATE);
|
|
|
|
|
|
|
|
// Session management
|
|
|
|
inboundMegolmSessionDb_ = lmdb::dbi::open(txn, INBOUND_MEGOLM_SESSIONS_DB, MDB_CREATE);
|
|
|
|
outboundMegolmSessionDb_ = lmdb::dbi::open(txn, OUTBOUND_MEGOLM_SESSIONS_DB, MDB_CREATE);
|
|
|
|
|
|
|
|
txn.commit();
|
2021-05-15 00:35:34 +03:00
|
|
|
|
|
|
|
databaseReady_ = true;
|
2018-06-10 20:03:45 +03:00
|
|
|
}
|
|
|
|
|
2018-06-12 09:45:26 +03:00
|
|
|
void
|
2018-06-18 18:36:19 +03:00
|
|
|
Cache::setEncryptedRoom(lmdb::txn &txn, const std::string &room_id)
|
2018-06-12 09:45:26 +03:00
|
|
|
{
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->info("mark room {} as encrypted", room_id);
|
2018-06-12 09:45:26 +03:00
|
|
|
|
2018-06-18 18:36:19 +03:00
|
|
|
auto db = lmdb::dbi::open(txn, ENCRYPTED_ROOMS_DB, MDB_CREATE);
|
2021-03-03 01:15:12 +03:00
|
|
|
db.put(txn, room_id, "0");
|
2018-06-12 09:45:26 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
Cache::isRoomEncrypted(const std::string &room_id)
|
|
|
|
{
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view unused;
|
2018-06-12 09:45:26 +03:00
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
auto db = lmdb::dbi::open(txn, ENCRYPTED_ROOMS_DB, MDB_CREATE);
|
2021-03-03 01:15:12 +03:00
|
|
|
auto res = db.get(txn, room_id, unused);
|
2018-06-12 09:45:26 +03:00
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2021-03-15 18:24:01 +03:00
|
|
|
std::optional<mtx::events::state::Encryption>
|
|
|
|
Cache::roomEncryptionSettings(const std::string &room_id)
|
|
|
|
{
|
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
|
|
|
try {
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
auto statesdb = getStatesDb(txn, room_id);
|
|
|
|
std::string_view event;
|
|
|
|
bool res =
|
|
|
|
statesdb.get(txn, to_string(mtx::events::EventType::RoomEncryption), event);
|
|
|
|
|
|
|
|
if (res) {
|
|
|
|
try {
|
|
|
|
StateEvent<Encryption> msg = json::parse(event);
|
|
|
|
|
|
|
|
return msg.content;
|
|
|
|
} catch (const json::exception &e) {
|
|
|
|
nhlog::db()->warn("failed to parse m.room.encryption event: {}",
|
|
|
|
e.what());
|
|
|
|
return Encryption{};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (lmdb::error &) {
|
|
|
|
}
|
|
|
|
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
2018-09-15 23:52:14 +03:00
|
|
|
mtx::crypto::ExportedSessionKeys
|
|
|
|
Cache::exportSessionKeys()
|
|
|
|
{
|
|
|
|
using namespace mtx::crypto;
|
|
|
|
|
|
|
|
ExportedSessionKeys keys;
|
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
auto cursor = lmdb::cursor::open(txn, inboundMegolmSessionDb_);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view key, value;
|
2018-09-15 23:52:14 +03:00
|
|
|
while (cursor.get(key, value, MDB_NEXT)) {
|
|
|
|
ExportedSession exported;
|
2018-09-16 11:19:18 +03:00
|
|
|
MegolmSessionIndex index;
|
2018-09-15 23:52:14 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
auto saved_session = unpickle<InboundSessionObject>(std::string(value), SECRET);
|
2018-09-16 11:19:18 +03:00
|
|
|
|
|
|
|
try {
|
|
|
|
index = nlohmann::json::parse(key).get<MegolmSessionIndex>();
|
|
|
|
} catch (const nlohmann::json::exception &e) {
|
|
|
|
nhlog::db()->critical("failed to export megolm session: {}", e.what());
|
|
|
|
continue;
|
|
|
|
}
|
2018-09-15 23:52:14 +03:00
|
|
|
|
|
|
|
exported.room_id = index.room_id;
|
|
|
|
exported.sender_key = index.sender_key;
|
|
|
|
exported.session_id = index.session_id;
|
2021-04-20 20:52:23 +03:00
|
|
|
exported.session_key = export_session(saved_session.get(), -1);
|
2018-09-15 23:52:14 +03:00
|
|
|
|
|
|
|
keys.sessions.push_back(exported);
|
|
|
|
}
|
|
|
|
|
|
|
|
cursor.close();
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return keys;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Cache::importSessionKeys(const mtx::crypto::ExportedSessionKeys &keys)
|
|
|
|
{
|
|
|
|
for (const auto &s : keys.sessions) {
|
|
|
|
MegolmSessionIndex index;
|
|
|
|
index.room_id = s.room_id;
|
|
|
|
index.session_id = s.session_id;
|
|
|
|
index.sender_key = s.sender_key;
|
|
|
|
|
|
|
|
auto exported_session = mtx::crypto::import_session(s.session_key);
|
|
|
|
|
|
|
|
saveInboundMegolmSession(index, std::move(exported_session));
|
2021-01-25 19:06:27 +03:00
|
|
|
ChatPage::instance()->receivedSessionKey(index.room_id, index.session_id);
|
2018-09-15 23:52:14 +03:00
|
|
|
}
|
|
|
|
}
|
2018-06-10 20:03:45 +03:00
|
|
|
|
|
|
|
//
|
|
|
|
// Session Management
|
|
|
|
//
|
|
|
|
|
|
|
|
void
|
|
|
|
Cache::saveInboundMegolmSession(const MegolmSessionIndex &index,
|
|
|
|
mtx::crypto::InboundGroupSessionPtr session)
|
|
|
|
{
|
|
|
|
using namespace mtx::crypto;
|
2018-09-15 23:52:14 +03:00
|
|
|
const auto key = json(index).dump();
|
2018-06-10 20:03:45 +03:00
|
|
|
const auto pickled = pickle<InboundSessionObject>(session.get(), SECRET);
|
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
2021-04-20 15:19:07 +03:00
|
|
|
|
|
|
|
std::string_view value;
|
|
|
|
if (inboundMegolmSessionDb_.get(txn, key, value)) {
|
|
|
|
auto oldSession = unpickle<InboundSessionObject>(std::string(value), SECRET);
|
|
|
|
if (olm_inbound_group_session_first_known_index(session.get()) >
|
|
|
|
olm_inbound_group_session_first_known_index(oldSession.get())) {
|
|
|
|
nhlog::crypto()->warn(
|
|
|
|
"Not storing inbound session with newer first known index");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
inboundMegolmSessionDb_.put(txn, key, pickled);
|
2018-06-10 20:03:45 +03:00
|
|
|
txn.commit();
|
|
|
|
}
|
|
|
|
|
2020-11-27 06:19:03 +03:00
|
|
|
mtx::crypto::InboundGroupSessionPtr
|
2018-06-10 20:03:45 +03:00
|
|
|
Cache::getInboundMegolmSession(const MegolmSessionIndex &index)
|
|
|
|
{
|
2020-11-27 06:19:03 +03:00
|
|
|
using namespace mtx::crypto;
|
|
|
|
|
|
|
|
try {
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
std::string key = json(index).dump();
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view value;
|
2020-11-27 06:19:03 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
if (inboundMegolmSessionDb_.get(txn, key, value)) {
|
|
|
|
auto session = unpickle<InboundSessionObject>(std::string(value), SECRET);
|
2020-11-27 06:19:03 +03:00
|
|
|
return session;
|
|
|
|
}
|
|
|
|
} catch (std::exception &e) {
|
|
|
|
nhlog::db()->error("Failed to get inbound megolm session {}", e.what());
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
2018-06-10 20:03:45 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2018-09-15 23:52:14 +03:00
|
|
|
Cache::inboundMegolmSessionExists(const MegolmSessionIndex &index)
|
2018-06-10 20:03:45 +03:00
|
|
|
{
|
2020-11-27 06:19:03 +03:00
|
|
|
using namespace mtx::crypto;
|
|
|
|
|
|
|
|
try {
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
std::string key = json(index).dump();
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view value;
|
2020-11-27 06:19:03 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
return inboundMegolmSessionDb_.get(txn, key, value);
|
2020-11-27 06:19:03 +03:00
|
|
|
} catch (std::exception &e) {
|
|
|
|
nhlog::db()->error("Failed to get inbound megolm session {}", e.what());
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2018-06-10 20:03:45 +03:00
|
|
|
}
|
|
|
|
|
2018-06-13 12:28:00 +03:00
|
|
|
void
|
2020-11-27 06:56:44 +03:00
|
|
|
Cache::updateOutboundMegolmSession(const std::string &room_id,
|
2020-11-30 02:26:27 +03:00
|
|
|
const OutboundGroupSessionData &data_,
|
2020-11-27 06:56:44 +03:00
|
|
|
mtx::crypto::OutboundGroupSessionPtr &ptr)
|
2018-06-13 12:28:00 +03:00
|
|
|
{
|
|
|
|
using namespace mtx::crypto;
|
|
|
|
|
|
|
|
if (!outboundMegolmSessionExists(room_id))
|
|
|
|
return;
|
|
|
|
|
2020-11-30 02:26:27 +03:00
|
|
|
OutboundGroupSessionData data = data_;
|
|
|
|
data.message_index = olm_outbound_group_session_message_index(ptr.get());
|
|
|
|
data.session_id = mtx::crypto::session_id(ptr.get());
|
|
|
|
data.session_key = mtx::crypto::session_key(ptr.get());
|
2018-06-13 12:28:00 +03:00
|
|
|
|
|
|
|
// Save the updated pickled data for the session.
|
|
|
|
json j;
|
|
|
|
j["data"] = data;
|
2020-11-27 06:56:44 +03:00
|
|
|
j["session"] = pickle<OutboundSessionObject>(ptr.get(), SECRET);
|
2018-06-13 12:28:00 +03:00
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
2021-03-03 01:15:12 +03:00
|
|
|
outboundMegolmSessionDb_.put(txn, room_id, j.dump());
|
2018-06-13 12:28:00 +03:00
|
|
|
txn.commit();
|
|
|
|
}
|
|
|
|
|
2020-10-03 19:38:28 +03:00
|
|
|
void
|
|
|
|
Cache::dropOutboundMegolmSession(const std::string &room_id)
|
|
|
|
{
|
|
|
|
using namespace mtx::crypto;
|
|
|
|
|
|
|
|
if (!outboundMegolmSessionExists(room_id))
|
|
|
|
return;
|
|
|
|
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
2021-03-03 01:15:12 +03:00
|
|
|
outboundMegolmSessionDb_.del(txn, room_id);
|
2020-10-03 19:38:28 +03:00
|
|
|
txn.commit();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-10 20:03:45 +03:00
|
|
|
void
|
2018-06-12 22:35:10 +03:00
|
|
|
Cache::saveOutboundMegolmSession(const std::string &room_id,
|
2018-06-10 20:03:45 +03:00
|
|
|
const OutboundGroupSessionData &data,
|
2020-11-30 02:26:27 +03:00
|
|
|
mtx::crypto::OutboundGroupSessionPtr &session)
|
2018-06-10 20:03:45 +03:00
|
|
|
{
|
|
|
|
using namespace mtx::crypto;
|
|
|
|
const auto pickled = pickle<OutboundSessionObject>(session.get(), SECRET);
|
|
|
|
|
|
|
|
json j;
|
|
|
|
j["data"] = data;
|
|
|
|
j["session"] = pickled;
|
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
2021-03-03 01:15:12 +03:00
|
|
|
outboundMegolmSessionDb_.put(txn, room_id, j.dump());
|
2018-06-10 20:03:45 +03:00
|
|
|
txn.commit();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2018-06-13 12:28:00 +03:00
|
|
|
Cache::outboundMegolmSessionExists(const std::string &room_id) noexcept
|
2018-06-10 20:03:45 +03:00
|
|
|
{
|
2020-11-27 06:56:44 +03:00
|
|
|
try {
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view value;
|
|
|
|
return outboundMegolmSessionDb_.get(txn, room_id, value);
|
2020-11-27 06:56:44 +03:00
|
|
|
} catch (std::exception &e) {
|
|
|
|
nhlog::db()->error("Failed to retrieve outbound Megolm Session: {}", e.what());
|
|
|
|
return false;
|
|
|
|
}
|
2018-06-10 20:03:45 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
OutboundGroupSessionDataRef
|
2018-06-13 12:28:00 +03:00
|
|
|
Cache::getOutboundMegolmSession(const std::string &room_id)
|
2018-06-10 20:03:45 +03:00
|
|
|
{
|
2020-11-27 06:56:44 +03:00
|
|
|
try {
|
|
|
|
using namespace mtx::crypto;
|
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view value;
|
|
|
|
outboundMegolmSessionDb_.get(txn, room_id, value);
|
|
|
|
auto obj = json::parse(value);
|
2020-11-27 06:56:44 +03:00
|
|
|
|
|
|
|
OutboundGroupSessionDataRef ref{};
|
|
|
|
ref.data = obj.at("data").get<OutboundGroupSessionData>();
|
|
|
|
ref.session = unpickle<OutboundSessionObject>(obj.at("session"), SECRET);
|
|
|
|
return ref;
|
|
|
|
} catch (std::exception &e) {
|
|
|
|
nhlog::db()->error("Failed to retrieve outbound Megolm Session: {}", e.what());
|
|
|
|
return {};
|
|
|
|
}
|
2018-06-10 20:03:45 +03:00
|
|
|
}
|
|
|
|
|
2018-06-15 01:35:31 +03:00
|
|
|
//
|
|
|
|
// OLM sessions.
|
|
|
|
//
|
|
|
|
|
2018-06-10 20:03:45 +03:00
|
|
|
void
|
2020-10-20 14:46:05 +03:00
|
|
|
Cache::saveOlmSession(const std::string &curve25519,
|
|
|
|
mtx::crypto::OlmSessionPtr session,
|
|
|
|
uint64_t timestamp)
|
2018-06-10 20:03:45 +03:00
|
|
|
{
|
|
|
|
using namespace mtx::crypto;
|
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
2018-06-15 01:35:31 +03:00
|
|
|
auto db = getOlmSessionsDb(txn, curve25519);
|
2018-06-10 20:03:45 +03:00
|
|
|
|
2018-06-15 01:35:31 +03:00
|
|
|
const auto pickled = pickle<SessionObject>(session.get(), SECRET);
|
|
|
|
const auto session_id = mtx::crypto::session_id(session.get());
|
|
|
|
|
2020-10-20 14:46:05 +03:00
|
|
|
StoredOlmSession stored_session;
|
|
|
|
stored_session.pickled_session = pickled;
|
|
|
|
stored_session.last_message_ts = timestamp;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
db.put(txn, session_id, json(stored_session).dump());
|
2018-06-15 01:35:31 +03:00
|
|
|
|
|
|
|
txn.commit();
|
2018-06-10 20:03:45 +03:00
|
|
|
}
|
|
|
|
|
2019-12-14 19:08:36 +03:00
|
|
|
std::optional<mtx::crypto::OlmSessionPtr>
|
2018-06-15 01:35:31 +03:00
|
|
|
Cache::getOlmSession(const std::string &curve25519, const std::string &session_id)
|
2018-06-10 20:03:45 +03:00
|
|
|
{
|
2018-06-15 01:35:31 +03:00
|
|
|
using namespace mtx::crypto;
|
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
auto db = getOlmSessionsDb(txn, curve25519);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view pickled;
|
|
|
|
bool found = db.get(txn, session_id, pickled);
|
2018-06-15 01:35:31 +03:00
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
if (found) {
|
2021-03-03 01:15:12 +03:00
|
|
|
auto data = json::parse(pickled).get<StoredOlmSession>();
|
2020-10-20 14:46:05 +03:00
|
|
|
return unpickle<SessionObject>(data.pickled_session, SECRET);
|
2018-06-15 01:35:31 +03:00
|
|
|
}
|
|
|
|
|
2019-12-14 19:08:36 +03:00
|
|
|
return std::nullopt;
|
2018-06-10 20:03:45 +03:00
|
|
|
}
|
|
|
|
|
2020-10-20 14:46:05 +03:00
|
|
|
std::optional<mtx::crypto::OlmSessionPtr>
|
|
|
|
Cache::getLatestOlmSession(const std::string &curve25519)
|
|
|
|
{
|
|
|
|
using namespace mtx::crypto;
|
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
auto db = getOlmSessionsDb(txn, curve25519);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view session_id, pickled_session;
|
2020-10-20 14:46:05 +03:00
|
|
|
|
|
|
|
std::optional<StoredOlmSession> currentNewest;
|
|
|
|
|
|
|
|
auto cursor = lmdb::cursor::open(txn, db);
|
|
|
|
while (cursor.get(session_id, pickled_session, MDB_NEXT)) {
|
2021-03-03 01:15:12 +03:00
|
|
|
auto data = json::parse(pickled_session).get<StoredOlmSession>();
|
2020-10-20 14:46:05 +03:00
|
|
|
if (!currentNewest || currentNewest->last_message_ts < data.last_message_ts)
|
|
|
|
currentNewest = data;
|
|
|
|
}
|
|
|
|
cursor.close();
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return currentNewest
|
|
|
|
? std::optional(unpickle<SessionObject>(currentNewest->pickled_session, SECRET))
|
|
|
|
: std::nullopt;
|
|
|
|
}
|
|
|
|
|
2018-06-15 01:35:31 +03:00
|
|
|
std::vector<std::string>
|
|
|
|
Cache::getOlmSessions(const std::string &curve25519)
|
2018-06-10 20:03:45 +03:00
|
|
|
{
|
2018-06-15 01:35:31 +03:00
|
|
|
using namespace mtx::crypto;
|
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
auto db = getOlmSessionsDb(txn, curve25519);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view session_id, unused;
|
2018-06-15 01:35:31 +03:00
|
|
|
std::vector<std::string> res;
|
|
|
|
|
|
|
|
auto cursor = lmdb::cursor::open(txn, db);
|
|
|
|
while (cursor.get(session_id, unused, MDB_NEXT))
|
|
|
|
res.emplace_back(session_id);
|
|
|
|
cursor.close();
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return res;
|
2018-06-10 20:03:45 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Cache::saveOlmAccount(const std::string &data)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
2021-03-03 01:15:12 +03:00
|
|
|
syncStateDb_.put(txn, OLM_ACCOUNT_KEY, data);
|
2018-06-10 20:03:45 +03:00
|
|
|
txn.commit();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string
|
|
|
|
Cache::restoreOlmAccount()
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view pickled;
|
|
|
|
syncStateDb_.get(txn, OLM_ACCOUNT_KEY, pickled);
|
2017-08-26 13:49:16 +03:00
|
|
|
txn.commit();
|
2018-06-10 20:03:45 +03:00
|
|
|
|
|
|
|
return std::string(pickled.data(), pickled.size());
|
2017-07-29 11:49:00 +03:00
|
|
|
}
|
|
|
|
|
2020-12-17 00:10:09 +03:00
|
|
|
void
|
|
|
|
Cache::storeSecret(const std::string &name, const std::string &secret)
|
|
|
|
{
|
2020-12-28 00:56:43 +03:00
|
|
|
auto settings = UserSettings::instance();
|
2020-12-17 00:10:09 +03:00
|
|
|
QKeychain::WritePasswordJob job(QCoreApplication::applicationName());
|
|
|
|
job.setAutoDelete(false);
|
|
|
|
job.setInsecureFallback(true);
|
2020-12-25 21:20:15 +03:00
|
|
|
job.setKey("matrix." +
|
2020-12-28 00:56:43 +03:00
|
|
|
QString(QCryptographicHash::hash(settings->profile().toUtf8(),
|
2020-12-25 21:20:15 +03:00
|
|
|
QCryptographicHash::Sha256)) +
|
|
|
|
"." + name.c_str());
|
2020-12-17 00:10:09 +03:00
|
|
|
job.setTextData(QString::fromStdString(secret));
|
|
|
|
QEventLoop loop;
|
|
|
|
job.connect(&job, &QKeychain::Job::finished, &loop, &QEventLoop::quit);
|
|
|
|
job.start();
|
|
|
|
loop.exec();
|
|
|
|
|
|
|
|
if (job.error()) {
|
|
|
|
nhlog::db()->warn(
|
|
|
|
"Storing secret '{}' failed: {}", name, job.errorString().toStdString());
|
|
|
|
} else {
|
|
|
|
emit secretChanged(name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Cache::deleteSecret(const std::string &name)
|
|
|
|
{
|
2020-12-28 00:56:43 +03:00
|
|
|
auto settings = UserSettings::instance();
|
2020-12-17 00:10:09 +03:00
|
|
|
QKeychain::DeletePasswordJob job(QCoreApplication::applicationName());
|
|
|
|
job.setAutoDelete(false);
|
|
|
|
job.setInsecureFallback(true);
|
2020-12-25 21:20:15 +03:00
|
|
|
job.setKey("matrix." +
|
2020-12-28 00:56:43 +03:00
|
|
|
QString(QCryptographicHash::hash(settings->profile().toUtf8(),
|
2020-12-25 21:20:15 +03:00
|
|
|
QCryptographicHash::Sha256)) +
|
|
|
|
"." + name.c_str());
|
2020-12-17 00:10:09 +03:00
|
|
|
QEventLoop loop;
|
|
|
|
job.connect(&job, &QKeychain::Job::finished, &loop, &QEventLoop::quit);
|
|
|
|
job.start();
|
|
|
|
loop.exec();
|
|
|
|
|
|
|
|
emit secretChanged(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<std::string>
|
|
|
|
Cache::secret(const std::string &name)
|
|
|
|
{
|
2020-12-28 00:56:43 +03:00
|
|
|
auto settings = UserSettings::instance();
|
2020-12-17 00:10:09 +03:00
|
|
|
QKeychain::ReadPasswordJob job(QCoreApplication::applicationName());
|
|
|
|
job.setAutoDelete(false);
|
|
|
|
job.setInsecureFallback(true);
|
2020-12-25 21:20:15 +03:00
|
|
|
job.setKey("matrix." +
|
2020-12-28 00:56:43 +03:00
|
|
|
QString(QCryptographicHash::hash(settings->profile().toUtf8(),
|
2020-12-25 21:20:15 +03:00
|
|
|
QCryptographicHash::Sha256)) +
|
|
|
|
"." + name.c_str());
|
2020-12-17 00:10:09 +03:00
|
|
|
QEventLoop loop;
|
|
|
|
job.connect(&job, &QKeychain::Job::finished, &loop, &QEventLoop::quit);
|
|
|
|
job.start();
|
|
|
|
loop.exec();
|
|
|
|
|
|
|
|
const QString secret = job.textData();
|
|
|
|
if (job.error()) {
|
|
|
|
nhlog::db()->debug(
|
|
|
|
"Restoring secret '{}' failed: {}", name, job.errorString().toStdString());
|
|
|
|
return std::nullopt;
|
2021-01-25 17:28:35 +03:00
|
|
|
}
|
|
|
|
if (secret.isEmpty()) {
|
|
|
|
nhlog::db()->debug("Restored empty secret '{}'.", name);
|
|
|
|
return std::nullopt;
|
2020-12-17 00:10:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return secret.toStdString();
|
|
|
|
}
|
|
|
|
|
2018-04-22 12:26:41 +03:00
|
|
|
void
|
|
|
|
Cache::removeInvite(lmdb::txn &txn, const std::string &room_id)
|
|
|
|
{
|
2021-03-03 01:15:12 +03:00
|
|
|
invitesDb_.del(txn, room_id);
|
|
|
|
getInviteStatesDb(txn, room_id).drop(txn, true);
|
|
|
|
getInviteMembersDb(txn, room_id).drop(txn, true);
|
2018-04-22 12:26:41 +03:00
|
|
|
}
|
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
void
|
2018-04-21 16:34:50 +03:00
|
|
|
Cache::removeInvite(const std::string &room_id)
|
2017-07-29 11:49:00 +03:00
|
|
|
{
|
2018-04-21 16:34:50 +03:00
|
|
|
auto txn = lmdb::txn::begin(env_);
|
2018-04-22 12:26:41 +03:00
|
|
|
removeInvite(txn, room_id);
|
2018-04-21 16:34:50 +03:00
|
|
|
txn.commit();
|
2017-07-29 11:49:00 +03:00
|
|
|
}
|
|
|
|
|
2017-10-01 19:49:36 +03:00
|
|
|
void
|
2018-04-21 16:34:50 +03:00
|
|
|
Cache::removeRoom(lmdb::txn &txn, const std::string &roomid)
|
2017-10-01 19:49:36 +03:00
|
|
|
{
|
2021-03-03 01:15:12 +03:00
|
|
|
roomsDb_.del(txn, roomid);
|
|
|
|
getStatesDb(txn, roomid).drop(txn, true);
|
|
|
|
getAccountDataDb(txn, roomid).drop(txn, true);
|
|
|
|
getMembersDb(txn, roomid).drop(txn, true);
|
2017-10-01 19:49:36 +03:00
|
|
|
}
|
|
|
|
|
2017-12-19 23:36:12 +03:00
|
|
|
void
|
2018-04-21 16:34:50 +03:00
|
|
|
Cache::removeRoom(const std::string &roomid)
|
2017-12-19 23:36:12 +03:00
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, 0);
|
2021-03-03 01:15:12 +03:00
|
|
|
roomsDb_.del(txn, roomid);
|
2017-12-19 23:36:12 +03:00
|
|
|
txn.commit();
|
|
|
|
}
|
|
|
|
|
2018-01-21 22:43:21 +03:00
|
|
|
void
|
2018-04-21 16:34:50 +03:00
|
|
|
Cache::setNextBatchToken(lmdb::txn &txn, const std::string &token)
|
2017-07-29 11:49:00 +03:00
|
|
|
{
|
2021-03-03 01:15:12 +03:00
|
|
|
syncStateDb_.put(txn, NEXT_BATCH_KEY, token);
|
2017-07-29 11:49:00 +03:00
|
|
|
}
|
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
void
|
2017-08-26 13:49:16 +03:00
|
|
|
Cache::setNextBatchToken(lmdb::txn &txn, const QString &token)
|
2017-07-29 11:49:00 +03:00
|
|
|
{
|
2018-04-21 16:34:50 +03:00
|
|
|
setNextBatchToken(txn, token.toStdString());
|
2017-07-29 11:49:00 +03:00
|
|
|
}
|
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
bool
|
2021-03-03 01:15:12 +03:00
|
|
|
Cache::isInitialized()
|
2017-07-29 11:49:00 +03:00
|
|
|
{
|
2017-08-26 13:49:16 +03:00
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view token;
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
bool res = syncStateDb_.get(txn, NEXT_BATCH_KEY, token);
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2017-08-26 13:49:16 +03:00
|
|
|
txn.commit();
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2017-08-26 13:49:16 +03:00
|
|
|
return res;
|
2017-07-29 11:49:00 +03:00
|
|
|
}
|
|
|
|
|
2018-06-09 16:03:14 +03:00
|
|
|
std::string
|
2021-03-03 01:15:12 +03:00
|
|
|
Cache::nextBatchToken()
|
2017-07-29 11:49:00 +03:00
|
|
|
{
|
2021-03-12 19:13:56 +03:00
|
|
|
if (!env_.handle())
|
|
|
|
throw lmdb::error("Env already closed", MDB_INVALID);
|
|
|
|
|
2017-08-26 13:49:16 +03:00
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view token;
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
bool result = syncStateDb_.get(txn, NEXT_BATCH_KEY, token);
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2017-08-26 13:49:16 +03:00
|
|
|
txn.commit();
|
2017-07-29 11:49:00 +03:00
|
|
|
|
2020-10-02 17:39:20 +03:00
|
|
|
if (result)
|
|
|
|
return std::string(token.data(), token.size());
|
|
|
|
else
|
|
|
|
return "";
|
2017-07-29 11:49:00 +03:00
|
|
|
}
|
2017-10-22 19:03:55 +03:00
|
|
|
|
|
|
|
void
|
|
|
|
Cache::deleteData()
|
|
|
|
{
|
2021-05-15 00:35:34 +03:00
|
|
|
this->databaseReady_ = false;
|
2018-06-12 20:36:16 +03:00
|
|
|
// TODO: We need to remove the env_ while not accepting new requests.
|
2020-12-17 00:10:09 +03:00
|
|
|
lmdb::dbi_close(env_, syncStateDb_);
|
|
|
|
lmdb::dbi_close(env_, roomsDb_);
|
|
|
|
lmdb::dbi_close(env_, invitesDb_);
|
|
|
|
lmdb::dbi_close(env_, readReceiptsDb_);
|
|
|
|
lmdb::dbi_close(env_, notificationsDb_);
|
|
|
|
|
|
|
|
lmdb::dbi_close(env_, devicesDb_);
|
|
|
|
lmdb::dbi_close(env_, deviceKeysDb_);
|
|
|
|
|
|
|
|
lmdb::dbi_close(env_, inboundMegolmSessionDb_);
|
|
|
|
lmdb::dbi_close(env_, outboundMegolmSessionDb_);
|
|
|
|
|
|
|
|
env_.close();
|
|
|
|
|
|
|
|
verification_storage.status.clear();
|
|
|
|
|
2018-06-12 20:36:16 +03:00
|
|
|
if (!cacheDirectory_.isEmpty()) {
|
2017-10-22 19:03:55 +03:00
|
|
|
QDir(cacheDirectory_).removeRecursively();
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->info("deleted cache files from disk");
|
2018-06-12 20:36:16 +03:00
|
|
|
}
|
2020-12-17 00:10:09 +03:00
|
|
|
|
|
|
|
deleteSecret(mtx::secret_storage::secrets::megolm_backup_v1);
|
|
|
|
deleteSecret(mtx::secret_storage::secrets::cross_signing_master);
|
|
|
|
deleteSecret(mtx::secret_storage::secrets::cross_signing_user_signing);
|
|
|
|
deleteSecret(mtx::secret_storage::secrets::cross_signing_self_signing);
|
2017-10-22 19:03:55 +03:00
|
|
|
}
|
2017-12-10 13:51:44 +03:00
|
|
|
|
2020-05-02 17:44:50 +03:00
|
|
|
//! migrates db to the current format
|
2017-12-10 13:51:44 +03:00
|
|
|
bool
|
2020-05-02 17:44:50 +03:00
|
|
|
Cache::runMigrations()
|
|
|
|
{
|
2020-05-02 18:24:45 +03:00
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view current_version;
|
|
|
|
bool res = syncStateDb_.get(txn, CACHE_FORMAT_VERSION_KEY, current_version);
|
2020-05-02 18:24:45 +03:00
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
if (!res)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
std::string stored_version(current_version.data(), current_version.size());
|
|
|
|
|
|
|
|
std::vector<std::pair<std::string, std::function<bool()>>> migrations{
|
|
|
|
{"2020.05.01",
|
|
|
|
[this]() {
|
|
|
|
try {
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr);
|
|
|
|
auto pending_receipts =
|
|
|
|
lmdb::dbi::open(txn, "pending_receipts", MDB_CREATE);
|
|
|
|
lmdb::dbi_drop(txn, pending_receipts, true);
|
|
|
|
txn.commit();
|
|
|
|
} catch (const lmdb::error &) {
|
|
|
|
nhlog::db()->critical(
|
|
|
|
"Failed to delete pending_receipts database in migration!");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
nhlog::db()->info("Successfully deleted pending receipts database.");
|
|
|
|
return true;
|
|
|
|
}},
|
|
|
|
{"2020.07.05",
|
|
|
|
[this]() {
|
|
|
|
try {
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr);
|
|
|
|
auto room_ids = getRoomIds(txn);
|
|
|
|
|
|
|
|
for (const auto &room_id : room_ids) {
|
2020-07-26 20:04:36 +03:00
|
|
|
try {
|
|
|
|
auto messagesDb = lmdb::dbi::open(
|
|
|
|
txn, std::string(room_id + "/messages").c_str());
|
|
|
|
|
|
|
|
// keep some old messages and batch token
|
|
|
|
{
|
|
|
|
auto roomsCursor =
|
|
|
|
lmdb::cursor::open(txn, messagesDb);
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view ts, stored_message;
|
2020-07-26 20:04:36 +03:00
|
|
|
bool start = true;
|
|
|
|
mtx::responses::Timeline oldMessages;
|
|
|
|
while (roomsCursor.get(ts,
|
|
|
|
stored_message,
|
|
|
|
start ? MDB_FIRST
|
|
|
|
: MDB_NEXT)) {
|
|
|
|
start = false;
|
|
|
|
|
|
|
|
auto j = json::parse(std::string_view(
|
|
|
|
stored_message.data(),
|
|
|
|
stored_message.size()));
|
|
|
|
|
|
|
|
if (oldMessages.prev_batch.empty())
|
|
|
|
oldMessages.prev_batch =
|
|
|
|
j["token"].get<std::string>();
|
|
|
|
else if (j["token"] !=
|
|
|
|
oldMessages.prev_batch)
|
|
|
|
break;
|
|
|
|
|
|
|
|
mtx::events::collections::TimelineEvent
|
|
|
|
te;
|
|
|
|
mtx::events::collections::from_json(
|
|
|
|
j["event"], te);
|
|
|
|
oldMessages.events.push_back(te.data);
|
|
|
|
}
|
|
|
|
// messages were stored in reverse order, so we
|
|
|
|
// need to reverse them
|
|
|
|
std::reverse(oldMessages.events.begin(),
|
|
|
|
oldMessages.events.end());
|
|
|
|
// save messages using the new method
|
|
|
|
saveTimelineMessages(txn, room_id, oldMessages);
|
|
|
|
}
|
|
|
|
|
|
|
|
// delete old messages db
|
|
|
|
lmdb::dbi_drop(txn, messagesDb, true);
|
|
|
|
} catch (std::exception &e) {
|
|
|
|
nhlog::db()->error(
|
|
|
|
"While migrating messages from {}, ignoring error {}",
|
|
|
|
room_id,
|
|
|
|
e.what());
|
|
|
|
}
|
2020-07-05 06:29:07 +03:00
|
|
|
}
|
|
|
|
txn.commit();
|
|
|
|
} catch (const lmdb::error &) {
|
|
|
|
nhlog::db()->critical(
|
|
|
|
"Failed to delete messages database in migration!");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-05-02 18:24:45 +03:00
|
|
|
nhlog::db()->info("Successfully deleted pending receipts database.");
|
|
|
|
return true;
|
|
|
|
}},
|
2020-10-20 14:46:05 +03:00
|
|
|
{"2020.10.20",
|
|
|
|
[this]() {
|
|
|
|
try {
|
|
|
|
using namespace mtx::crypto;
|
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
|
|
|
|
auto mainDb = lmdb::dbi::open(txn, nullptr);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view dbName, ignored;
|
2020-10-20 14:46:05 +03:00
|
|
|
auto olmDbCursor = lmdb::cursor::open(txn, mainDb);
|
|
|
|
while (olmDbCursor.get(dbName, ignored, MDB_NEXT)) {
|
|
|
|
// skip every db but olm session dbs
|
|
|
|
nhlog::db()->debug("Db {}", dbName);
|
|
|
|
if (dbName.find("olm_sessions/") != 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
nhlog::db()->debug("Migrating {}", dbName);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
auto olmDb = lmdb::dbi::open(txn, std::string(dbName).c_str());
|
2020-10-20 14:46:05 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view session_id, session_value;
|
2020-10-20 14:46:05 +03:00
|
|
|
|
|
|
|
std::vector<std::pair<std::string, StoredOlmSession>> sessions;
|
|
|
|
|
|
|
|
auto cursor = lmdb::cursor::open(txn, olmDb);
|
|
|
|
while (cursor.get(session_id, session_value, MDB_NEXT)) {
|
|
|
|
nhlog::db()->debug("session_id {}, session_value {}",
|
|
|
|
session_id,
|
|
|
|
session_value);
|
|
|
|
StoredOlmSession session;
|
|
|
|
bool invalid = false;
|
|
|
|
for (auto c : session_value)
|
|
|
|
if (!isprint(c)) {
|
|
|
|
invalid = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (invalid)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
nhlog::db()->debug("Not skipped");
|
|
|
|
|
|
|
|
session.pickled_session = session_value;
|
|
|
|
sessions.emplace_back(session_id, session);
|
|
|
|
}
|
|
|
|
cursor.close();
|
|
|
|
|
|
|
|
olmDb.drop(txn, true);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
auto newDbName = std::string(dbName);
|
2020-10-20 14:46:05 +03:00
|
|
|
newDbName.erase(0, sizeof("olm_sessions") - 1);
|
|
|
|
newDbName = "olm_sessions.v2" + newDbName;
|
|
|
|
|
|
|
|
auto newDb = lmdb::dbi::open(txn, newDbName.c_str(), MDB_CREATE);
|
|
|
|
|
|
|
|
for (const auto &[key, value] : sessions) {
|
2021-03-03 01:15:12 +03:00
|
|
|
// nhlog::db()->debug("{}\n{}", key, json(value).dump());
|
|
|
|
newDb.put(txn, key, json(value).dump());
|
2020-10-20 14:46:05 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
olmDbCursor.close();
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
} catch (const lmdb::error &) {
|
|
|
|
nhlog::db()->critical("Failed to migrate olm sessions,");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
nhlog::db()->info("Successfully migrated olm sessions.");
|
|
|
|
return true;
|
|
|
|
}},
|
2020-05-02 18:24:45 +03:00
|
|
|
};
|
|
|
|
|
2020-08-26 00:12:01 +03:00
|
|
|
nhlog::db()->info("Running migrations, this may take a while!");
|
2020-05-02 18:24:45 +03:00
|
|
|
for (const auto &[target_version, migration] : migrations) {
|
|
|
|
if (target_version > stored_version)
|
|
|
|
if (!migration()) {
|
|
|
|
nhlog::db()->critical("migration failure!");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2020-08-26 00:12:01 +03:00
|
|
|
nhlog::db()->info("Migrations finished.");
|
2020-05-02 18:24:45 +03:00
|
|
|
|
|
|
|
setCurrentFormat();
|
2020-05-02 17:44:50 +03:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
cache::CacheVersion
|
|
|
|
Cache::formatVersion()
|
2017-12-10 13:51:44 +03:00
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view current_version;
|
|
|
|
bool res = syncStateDb_.get(txn, CACHE_FORMAT_VERSION_KEY, current_version);
|
2017-12-10 13:51:44 +03:00
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
if (!res)
|
2020-05-02 17:44:50 +03:00
|
|
|
return cache::CacheVersion::Older;
|
2017-12-10 13:51:44 +03:00
|
|
|
|
|
|
|
std::string stored_version(current_version.data(), current_version.size());
|
|
|
|
|
2020-05-02 17:44:50 +03:00
|
|
|
if (stored_version < CURRENT_CACHE_FORMAT_VERSION)
|
|
|
|
return cache::CacheVersion::Older;
|
|
|
|
else if (stored_version > CURRENT_CACHE_FORMAT_VERSION)
|
|
|
|
return cache::CacheVersion::Older;
|
|
|
|
else
|
|
|
|
return cache::CacheVersion::Current;
|
2017-12-10 13:51:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Cache::setCurrentFormat()
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
syncStateDb_.put(txn, CACHE_FORMAT_VERSION_KEY, CURRENT_CACHE_FORMAT_VERSION);
|
2017-12-10 13:51:44 +03:00
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
}
|
2017-12-19 23:36:12 +03:00
|
|
|
|
2018-01-15 00:33:12 +03:00
|
|
|
CachedReceipts
|
2018-01-03 19:05:49 +03:00
|
|
|
Cache::readReceipts(const QString &event_id, const QString &room_id)
|
|
|
|
{
|
2018-01-15 00:33:12 +03:00
|
|
|
CachedReceipts receipts;
|
2018-01-03 19:05:49 +03:00
|
|
|
|
|
|
|
ReadReceiptKey receipt_key{event_id.toStdString(), room_id.toStdString()};
|
|
|
|
nlohmann::json json_key = receipt_key;
|
|
|
|
|
|
|
|
try {
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
auto key = json_key.dump();
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view value;
|
2018-01-03 19:05:49 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
bool res = readReceiptsDb_.get(txn, key, value);
|
2018-01-03 19:05:49 +03:00
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
if (res) {
|
2020-07-06 04:43:14 +03:00
|
|
|
auto json_response =
|
|
|
|
json::parse(std::string_view(value.data(), value.size()));
|
|
|
|
auto values = json_response.get<std::map<std::string, uint64_t>>();
|
2018-01-03 19:05:49 +03:00
|
|
|
|
2018-02-28 13:12:07 +03:00
|
|
|
for (const auto &v : values)
|
2018-01-15 00:33:12 +03:00
|
|
|
// timestamp, user_id
|
|
|
|
receipts.emplace(v.second, v.first);
|
2018-01-03 19:05:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
} catch (const lmdb::error &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->critical("readReceipts: {}", e.what());
|
2018-01-03 19:05:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return receipts;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2018-04-21 17:14:16 +03:00
|
|
|
Cache::updateReadReceipt(lmdb::txn &txn, const std::string &room_id, const Receipts &receipts)
|
2018-01-03 19:05:49 +03:00
|
|
|
{
|
2020-04-09 21:52:50 +03:00
|
|
|
auto user_id = this->localUserId_.toStdString();
|
2018-02-28 13:12:07 +03:00
|
|
|
for (const auto &receipt : receipts) {
|
2018-01-03 19:05:49 +03:00
|
|
|
const auto event_id = receipt.first;
|
|
|
|
auto event_receipts = receipt.second;
|
|
|
|
|
|
|
|
ReadReceiptKey receipt_key{event_id, room_id};
|
|
|
|
nlohmann::json json_key = receipt_key;
|
|
|
|
|
|
|
|
try {
|
|
|
|
const auto key = json_key.dump();
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view prev_value;
|
2018-01-03 19:05:49 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
bool exists = readReceiptsDb_.get(txn, key, prev_value);
|
2018-01-03 19:05:49 +03:00
|
|
|
|
2018-01-15 00:33:12 +03:00
|
|
|
std::map<std::string, uint64_t> saved_receipts;
|
2018-01-03 19:05:49 +03:00
|
|
|
|
|
|
|
// If an entry for the event id already exists, we would
|
|
|
|
// merge the existing receipts with the new ones.
|
|
|
|
if (exists) {
|
2020-07-06 04:43:14 +03:00
|
|
|
auto json_value = json::parse(
|
|
|
|
std::string_view(prev_value.data(), prev_value.size()));
|
2018-01-03 19:05:49 +03:00
|
|
|
|
|
|
|
// Retrieve the saved receipts.
|
2018-01-15 00:33:12 +03:00
|
|
|
saved_receipts = json_value.get<std::map<std::string, uint64_t>>();
|
2018-01-03 19:05:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Append the new ones.
|
2020-04-09 21:52:50 +03:00
|
|
|
for (const auto &[read_by, timestamp] : event_receipts) {
|
|
|
|
if (read_by == user_id) {
|
|
|
|
emit removeNotification(QString::fromStdString(room_id),
|
|
|
|
QString::fromStdString(event_id));
|
|
|
|
}
|
|
|
|
saved_receipts.emplace(read_by, timestamp);
|
|
|
|
}
|
2018-01-03 19:05:49 +03:00
|
|
|
|
|
|
|
// Save back the merged (or only the new) receipts.
|
|
|
|
nlohmann::json json_updated_value = saved_receipts;
|
|
|
|
std::string merged_receipts = json_updated_value.dump();
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
readReceiptsDb_.put(txn, key, merged_receipts);
|
2018-01-03 19:05:49 +03:00
|
|
|
|
|
|
|
} catch (const lmdb::error &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->critical("updateReadReceipts: {}", e.what());
|
2018-01-03 19:05:49 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-04-21 16:34:50 +03:00
|
|
|
|
2018-09-13 19:15:58 +03:00
|
|
|
void
|
|
|
|
Cache::calculateRoomReadStatus()
|
|
|
|
{
|
|
|
|
const auto joined_rooms = joinedRooms();
|
|
|
|
|
|
|
|
std::map<QString, bool> readStatus;
|
|
|
|
|
|
|
|
for (const auto &room : joined_rooms)
|
|
|
|
readStatus.emplace(QString::fromStdString(room), calculateRoomReadStatus(room));
|
|
|
|
|
|
|
|
emit roomReadStatus(readStatus);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
Cache::calculateRoomReadStatus(const std::string &room_id)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
|
|
|
|
// Get last event id on the room.
|
2020-02-24 03:07:25 +03:00
|
|
|
const auto last_event_id = getLastEventId(txn, room_id);
|
2018-09-13 19:15:58 +03:00
|
|
|
const auto localUser = utils::localUser().toStdString();
|
|
|
|
|
2021-02-10 04:37:47 +03:00
|
|
|
std::string fullyReadEventId;
|
|
|
|
if (auto ev = getAccountData(txn, mtx::events::EventType::FullyRead, room_id)) {
|
|
|
|
if (auto fr = std::get_if<
|
|
|
|
mtx::events::AccountDataEvent<mtx::events::account_data::FullyRead>>(
|
|
|
|
&ev.value())) {
|
|
|
|
fullyReadEventId = fr->content.event_id;
|
|
|
|
}
|
|
|
|
}
|
2018-09-13 19:15:58 +03:00
|
|
|
txn.commit();
|
|
|
|
|
2021-02-10 04:37:47 +03:00
|
|
|
if (last_event_id.empty() || fullyReadEventId.empty())
|
2018-09-13 19:15:58 +03:00
|
|
|
return true;
|
|
|
|
|
2021-02-10 04:37:47 +03:00
|
|
|
if (last_event_id == fullyReadEventId)
|
|
|
|
return false;
|
2018-09-13 19:15:58 +03:00
|
|
|
|
2021-02-10 04:37:47 +03:00
|
|
|
// Retrieve all read receipts for that event.
|
|
|
|
return getEventIndex(room_id, last_event_id) > getEventIndex(room_id, fullyReadEventId);
|
2018-07-17 23:50:18 +03:00
|
|
|
}
|
|
|
|
|
2018-04-21 16:34:50 +03:00
|
|
|
void
|
|
|
|
Cache::saveState(const mtx::responses::Sync &res)
|
|
|
|
{
|
2018-09-28 15:40:51 +03:00
|
|
|
using namespace mtx::events;
|
2021-01-06 00:10:40 +03:00
|
|
|
auto local_user_id = this->localUserId_.toStdString();
|
2018-09-28 15:40:51 +03:00
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
auto currentBatchToken = nextBatchToken();
|
|
|
|
|
2018-04-21 16:34:50 +03:00
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
|
|
|
|
setNextBatchToken(txn, res.next_batch);
|
|
|
|
|
2020-08-27 22:49:05 +03:00
|
|
|
if (!res.account_data.events.empty()) {
|
|
|
|
auto accountDataDb = getAccountDataDb(txn, "");
|
|
|
|
for (const auto &ev : res.account_data.events)
|
|
|
|
std::visit(
|
|
|
|
[&txn, &accountDataDb](const auto &event) {
|
2021-01-12 02:02:18 +03:00
|
|
|
auto j = json(event);
|
2021-03-03 01:15:12 +03:00
|
|
|
accountDataDb.put(txn, j["type"].get<std::string>(), j.dump());
|
2020-08-27 22:49:05 +03:00
|
|
|
},
|
|
|
|
ev);
|
|
|
|
}
|
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
auto userKeyCacheDb = getUserKeysDb(txn);
|
|
|
|
|
2018-04-21 16:34:50 +03:00
|
|
|
// Save joined rooms
|
|
|
|
for (const auto &room : res.rooms.join) {
|
2021-04-03 14:15:35 +03:00
|
|
|
auto statesdb = getStatesDb(txn, room.first);
|
|
|
|
auto stateskeydb = getStatesKeyDb(txn, room.first);
|
|
|
|
auto membersdb = getMembersDb(txn, room.first);
|
|
|
|
|
|
|
|
saveStateEvents(
|
|
|
|
txn, statesdb, stateskeydb, membersdb, room.first, room.second.state.events);
|
|
|
|
saveStateEvents(
|
|
|
|
txn, statesdb, stateskeydb, membersdb, room.first, room.second.timeline.events);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
2018-06-28 16:16:43 +03:00
|
|
|
saveTimelineMessages(txn, room.first, room.second.timeline);
|
|
|
|
|
2018-04-21 16:34:50 +03:00
|
|
|
RoomInfo updatedInfo;
|
2020-11-26 01:43:31 +03:00
|
|
|
updatedInfo.name = getRoomName(txn, statesdb, membersdb).toStdString();
|
|
|
|
updatedInfo.topic = getRoomTopic(txn, statesdb).toStdString();
|
2020-11-26 00:45:33 +03:00
|
|
|
updatedInfo.avatar_url = getRoomAvatarUrl(txn, statesdb, membersdb).toStdString();
|
2020-11-26 01:43:31 +03:00
|
|
|
updatedInfo.version = getRoomVersion(txn, statesdb).toStdString();
|
2018-04-21 16:34:50 +03:00
|
|
|
|
2021-01-22 05:18:36 +03:00
|
|
|
bool has_new_tags = false;
|
2018-09-28 15:40:51 +03:00
|
|
|
// Process the account_data associated with this room
|
2020-09-07 20:03:54 +03:00
|
|
|
if (!room.second.account_data.events.empty()) {
|
2020-08-27 22:49:05 +03:00
|
|
|
auto accountDataDb = getAccountDataDb(txn, room.first);
|
|
|
|
|
|
|
|
for (const auto &evt : room.second.account_data.events) {
|
|
|
|
std::visit(
|
|
|
|
[&txn, &accountDataDb](const auto &event) {
|
2021-01-12 02:02:18 +03:00
|
|
|
auto j = json(event);
|
2021-03-03 01:15:12 +03:00
|
|
|
accountDataDb.put(
|
|
|
|
txn, j["type"].get<std::string>(), j.dump());
|
2020-08-27 22:49:05 +03:00
|
|
|
},
|
|
|
|
evt);
|
|
|
|
|
|
|
|
// for tag events
|
2021-01-06 00:10:40 +03:00
|
|
|
if (std::holds_alternative<AccountDataEvent<account_data::Tags>>(
|
|
|
|
evt)) {
|
|
|
|
auto tags_evt =
|
|
|
|
std::get<AccountDataEvent<account_data::Tags>>(evt);
|
|
|
|
has_new_tags = true;
|
2020-08-27 22:49:05 +03:00
|
|
|
for (const auto &tag : tags_evt.content.tags) {
|
|
|
|
updatedInfo.tags.push_back(tag.first);
|
|
|
|
}
|
2018-09-28 15:40:51 +03:00
|
|
|
}
|
2021-01-06 00:10:40 +03:00
|
|
|
if (auto fr = std::get_if<mtx::events::AccountDataEvent<
|
|
|
|
mtx::events::account_data::FullyRead>>(&evt)) {
|
|
|
|
nhlog::db()->debug("Fully read: {}", fr->content.event_id);
|
|
|
|
}
|
2018-09-28 15:40:51 +03:00
|
|
|
}
|
2021-01-22 05:18:36 +03:00
|
|
|
}
|
|
|
|
if (!has_new_tags) {
|
|
|
|
// retrieve the old tags, they haven't changed
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view data;
|
|
|
|
if (roomsDb_.get(txn, room.first, data)) {
|
2021-01-22 05:18:36 +03:00
|
|
|
try {
|
|
|
|
RoomInfo tmp =
|
|
|
|
json::parse(std::string_view(data.data(), data.size()));
|
|
|
|
updatedInfo.tags = tmp.tags;
|
|
|
|
} catch (const json::exception &e) {
|
|
|
|
nhlog::db()->warn(
|
|
|
|
"failed to parse room info: room_id ({}), {}: {}",
|
|
|
|
room.first,
|
|
|
|
std::string(data.data(), data.size()),
|
|
|
|
e.what());
|
2018-09-28 15:40:51 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
roomsDb_.put(txn, room.first, json(updatedInfo).dump());
|
2018-04-21 17:14:16 +03:00
|
|
|
|
2021-01-06 00:10:40 +03:00
|
|
|
for (const auto &e : room.second.ephemeral.events) {
|
|
|
|
if (auto receiptsEv = std::get_if<
|
|
|
|
mtx::events::EphemeralEvent<mtx::events::ephemeral::Receipt>>(&e)) {
|
|
|
|
Receipts receipts;
|
|
|
|
|
|
|
|
for (const auto &[event_id, userReceipts] :
|
|
|
|
receiptsEv->content.receipts) {
|
|
|
|
for (const auto &[user_id, receipt] : userReceipts.users) {
|
|
|
|
receipts[event_id][user_id] = receipt.ts;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
updateReadReceipt(txn, room.first, receipts);
|
|
|
|
}
|
|
|
|
}
|
2018-04-22 12:26:41 +03:00
|
|
|
|
|
|
|
// Clean up non-valid invites.
|
|
|
|
removeInvite(txn, room.first);
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
saveInvites(txn, res.rooms.invite);
|
|
|
|
|
2020-06-08 02:45:24 +03:00
|
|
|
savePresence(txn, res.presence);
|
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
markUserKeysOutOfDate(txn, userKeyCacheDb, res.device_lists.changed, currentBatchToken);
|
|
|
|
deleteUserKeys(txn, userKeyCacheDb, res.device_lists.left);
|
2020-07-06 19:02:21 +03:00
|
|
|
|
2018-04-21 16:34:50 +03:00
|
|
|
removeLeftRooms(txn, res.rooms.leave);
|
|
|
|
|
|
|
|
txn.commit();
|
2018-07-17 23:50:18 +03:00
|
|
|
|
2018-09-13 19:15:58 +03:00
|
|
|
std::map<QString, bool> readStatus;
|
|
|
|
|
2018-07-17 23:50:18 +03:00
|
|
|
for (const auto &room : res.rooms.join) {
|
2021-01-06 00:10:40 +03:00
|
|
|
for (const auto &e : room.second.ephemeral.events) {
|
|
|
|
if (auto receiptsEv = std::get_if<
|
|
|
|
mtx::events::EphemeralEvent<mtx::events::ephemeral::Receipt>>(&e)) {
|
|
|
|
std::vector<QString> receipts;
|
|
|
|
|
|
|
|
for (const auto &[event_id, userReceipts] :
|
|
|
|
receiptsEv->content.receipts) {
|
|
|
|
for (const auto &[user_id, receipt] : userReceipts.users) {
|
|
|
|
(void)receipt;
|
|
|
|
|
|
|
|
if (user_id != local_user_id) {
|
|
|
|
receipts.push_back(
|
|
|
|
QString::fromStdString(event_id));
|
|
|
|
break;
|
|
|
|
}
|
2020-05-06 20:21:31 +03:00
|
|
|
}
|
|
|
|
}
|
2021-01-06 00:10:40 +03:00
|
|
|
if (!receipts.empty())
|
|
|
|
emit newReadReceipts(QString::fromStdString(room.first),
|
|
|
|
receipts);
|
2020-05-06 20:21:31 +03:00
|
|
|
}
|
2020-04-30 23:42:56 +03:00
|
|
|
}
|
2018-09-13 19:15:58 +03:00
|
|
|
readStatus.emplace(QString::fromStdString(room.first),
|
|
|
|
calculateRoomReadStatus(room.first));
|
2018-07-17 23:50:18 +03:00
|
|
|
}
|
2018-09-13 19:15:58 +03:00
|
|
|
|
|
|
|
emit roomReadStatus(readStatus);
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Cache::saveInvites(lmdb::txn &txn, const std::map<std::string, mtx::responses::InvitedRoom> &rooms)
|
|
|
|
{
|
|
|
|
for (const auto &room : rooms) {
|
|
|
|
auto statesdb = getInviteStatesDb(txn, room.first);
|
|
|
|
auto membersdb = getInviteMembersDb(txn, room.first);
|
|
|
|
|
|
|
|
saveInvite(txn, statesdb, membersdb, room.second);
|
|
|
|
|
|
|
|
RoomInfo updatedInfo;
|
|
|
|
updatedInfo.name = getInviteRoomName(txn, statesdb, membersdb).toStdString();
|
|
|
|
updatedInfo.topic = getInviteRoomTopic(txn, statesdb).toStdString();
|
|
|
|
updatedInfo.avatar_url =
|
|
|
|
getInviteRoomAvatarUrl(txn, statesdb, membersdb).toStdString();
|
|
|
|
updatedInfo.is_invite = true;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
invitesDb_.put(txn, room.first, json(updatedInfo).dump());
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Cache::saveInvite(lmdb::txn &txn,
|
|
|
|
lmdb::dbi &statesdb,
|
|
|
|
lmdb::dbi &membersdb,
|
|
|
|
const mtx::responses::InvitedRoom &room)
|
|
|
|
{
|
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
|
|
|
for (const auto &e : room.invite_state) {
|
2019-12-14 19:08:36 +03:00
|
|
|
if (auto msg = std::get_if<StrippedEvent<Member>>(&e)) {
|
|
|
|
auto display_name = msg->content.display_name.empty()
|
|
|
|
? msg->state_key
|
|
|
|
: msg->content.display_name;
|
2018-04-21 16:34:50 +03:00
|
|
|
|
2019-12-14 19:08:36 +03:00
|
|
|
MemberInfo tmp{display_name, msg->content.avatar_url};
|
2018-04-21 16:34:50 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
membersdb.put(txn, msg->state_key, json(tmp).dump());
|
2018-04-21 16:34:50 +03:00
|
|
|
} else {
|
2019-12-14 19:08:36 +03:00
|
|
|
std::visit(
|
2018-04-21 16:34:50 +03:00
|
|
|
[&txn, &statesdb](auto msg) {
|
2021-03-03 01:15:12 +03:00
|
|
|
auto j = json(msg);
|
|
|
|
bool res =
|
|
|
|
statesdb.put(txn, j["type"].get<std::string>(), j.dump());
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
if (!res)
|
2019-12-23 07:22:03 +03:00
|
|
|
nhlog::db()->warn("couldn't save data: {}",
|
|
|
|
json(msg).dump());
|
2018-04-21 16:34:50 +03:00
|
|
|
},
|
|
|
|
e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-08 02:45:24 +03:00
|
|
|
void
|
|
|
|
Cache::savePresence(
|
|
|
|
lmdb::txn &txn,
|
|
|
|
const std::vector<mtx::events::Event<mtx::events::presence::Presence>> &presenceUpdates)
|
|
|
|
{
|
|
|
|
for (const auto &update : presenceUpdates) {
|
|
|
|
auto presenceDb = getPresenceDb(txn);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
presenceDb.put(txn, update.sender, json(update.content).dump());
|
2020-06-08 02:45:24 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-21 16:34:50 +03:00
|
|
|
std::vector<std::string>
|
|
|
|
Cache::roomsWithStateUpdates(const mtx::responses::Sync &res)
|
|
|
|
{
|
|
|
|
std::vector<std::string> rooms;
|
|
|
|
for (const auto &room : res.rooms.join) {
|
|
|
|
bool hasUpdates = false;
|
|
|
|
for (const auto &s : room.second.state.events) {
|
|
|
|
if (containsStateUpdates(s)) {
|
|
|
|
hasUpdates = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const auto &s : room.second.timeline.events) {
|
|
|
|
if (containsStateUpdates(s)) {
|
|
|
|
hasUpdates = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hasUpdates)
|
|
|
|
rooms.emplace_back(room.first);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const auto &room : res.rooms.invite) {
|
|
|
|
for (const auto &s : room.second.invite_state) {
|
|
|
|
if (containsStateUpdates(s)) {
|
|
|
|
rooms.emplace_back(room.first);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return rooms;
|
|
|
|
}
|
|
|
|
|
2018-09-28 15:40:51 +03:00
|
|
|
std::vector<std::string>
|
|
|
|
Cache::roomsWithTagUpdates(const mtx::responses::Sync &res)
|
|
|
|
{
|
|
|
|
using namespace mtx::events;
|
|
|
|
|
|
|
|
std::vector<std::string> rooms;
|
|
|
|
for (const auto &room : res.rooms.join) {
|
|
|
|
bool hasUpdates = false;
|
|
|
|
for (const auto &evt : room.second.account_data.events) {
|
2021-01-06 00:10:40 +03:00
|
|
|
if (std::holds_alternative<AccountDataEvent<account_data::Tags>>(evt)) {
|
2018-09-28 15:40:51 +03:00
|
|
|
hasUpdates = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hasUpdates)
|
|
|
|
rooms.emplace_back(room.first);
|
|
|
|
}
|
|
|
|
|
|
|
|
return rooms;
|
|
|
|
}
|
|
|
|
|
2018-05-05 22:40:24 +03:00
|
|
|
RoomInfo
|
|
|
|
Cache::singleRoomInfo(const std::string &room_id)
|
|
|
|
{
|
2018-05-13 01:31:58 +03:00
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
auto statesdb = getStatesDb(txn, room_id);
|
2018-05-05 22:40:24 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view data;
|
2018-05-05 22:40:24 +03:00
|
|
|
|
|
|
|
// Check if the room is joined.
|
2021-03-03 01:15:12 +03:00
|
|
|
if (roomsDb_.get(txn, room_id, data)) {
|
2018-05-05 22:40:24 +03:00
|
|
|
try {
|
2021-03-03 01:15:12 +03:00
|
|
|
RoomInfo tmp = json::parse(data);
|
2018-05-05 22:40:24 +03:00
|
|
|
tmp.member_count = getMembersDb(txn, room_id).size(txn);
|
2018-05-13 01:31:58 +03:00
|
|
|
tmp.join_rule = getRoomJoinRule(txn, statesdb);
|
|
|
|
tmp.guest_access = getRoomGuestAccess(txn, statesdb);
|
2018-05-05 22:40:24 +03:00
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return tmp;
|
|
|
|
} catch (const json::exception &e) {
|
2020-12-25 03:08:06 +03:00
|
|
|
nhlog::db()->warn("failed to parse room info: room_id ({}), {}: {}",
|
2018-06-14 02:28:35 +03:00
|
|
|
room_id,
|
2020-12-25 06:11:47 +03:00
|
|
|
std::string(data.data(), data.size()),
|
|
|
|
e.what());
|
2018-05-05 22:40:24 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return RoomInfo();
|
|
|
|
}
|
|
|
|
|
2018-04-21 16:34:50 +03:00
|
|
|
std::map<QString, RoomInfo>
|
|
|
|
Cache::getRoomInfo(const std::vector<std::string> &rooms)
|
|
|
|
{
|
|
|
|
std::map<QString, RoomInfo> room_info;
|
|
|
|
|
2018-06-17 18:44:04 +03:00
|
|
|
// TODO This should be read only.
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
for (const auto &room : rooms) {
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view data;
|
2018-05-13 01:31:58 +03:00
|
|
|
auto statesdb = getStatesDb(txn, room);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
// Check if the room is joined.
|
2021-03-03 01:15:12 +03:00
|
|
|
if (roomsDb_.get(txn, room, data)) {
|
2018-04-21 16:34:50 +03:00
|
|
|
try {
|
2021-03-03 01:15:12 +03:00
|
|
|
RoomInfo tmp = json::parse(data);
|
2018-04-30 21:41:47 +03:00
|
|
|
tmp.member_count = getMembersDb(txn, room).size(txn);
|
2018-05-13 01:31:58 +03:00
|
|
|
tmp.join_rule = getRoomJoinRule(txn, statesdb);
|
|
|
|
tmp.guest_access = getRoomGuestAccess(txn, statesdb);
|
2018-04-30 21:41:47 +03:00
|
|
|
|
|
|
|
room_info.emplace(QString::fromStdString(room), std::move(tmp));
|
2018-04-21 16:34:50 +03:00
|
|
|
} catch (const json::exception &e) {
|
2020-12-25 03:08:06 +03:00
|
|
|
nhlog::db()->warn("failed to parse room info: room_id ({}), {}: {}",
|
2018-06-14 02:28:35 +03:00
|
|
|
room,
|
2020-12-25 06:11:47 +03:00
|
|
|
std::string(data.data(), data.size()),
|
|
|
|
e.what());
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Check if the room is an invite.
|
2021-03-03 01:15:12 +03:00
|
|
|
if (invitesDb_.get(txn, room, data)) {
|
2018-04-21 16:34:50 +03:00
|
|
|
try {
|
2021-03-03 01:15:12 +03:00
|
|
|
RoomInfo tmp = json::parse(std::string_view(data));
|
2018-04-30 21:41:47 +03:00
|
|
|
tmp.member_count = getInviteMembersDb(txn, room).size(txn);
|
|
|
|
|
|
|
|
room_info.emplace(QString::fromStdString(room),
|
|
|
|
std::move(tmp));
|
2018-04-21 16:34:50 +03:00
|
|
|
} catch (const json::exception &e) {
|
2020-12-25 06:11:47 +03:00
|
|
|
nhlog::db()->warn("failed to parse room info for invite: "
|
|
|
|
"room_id ({}), {}: {}",
|
|
|
|
room,
|
|
|
|
std::string(data.data(), data.size()),
|
|
|
|
e.what());
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return room_info;
|
|
|
|
}
|
|
|
|
|
2020-10-28 15:06:28 +03:00
|
|
|
std::vector<QString>
|
|
|
|
Cache::roomIds()
|
2018-06-28 16:16:43 +03:00
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
|
2020-10-28 15:06:28 +03:00
|
|
|
std::vector<QString> rooms;
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view room_id, unused;
|
2018-06-28 16:16:43 +03:00
|
|
|
|
|
|
|
auto roomsCursor = lmdb::cursor::open(txn, roomsDb_);
|
|
|
|
while (roomsCursor.get(room_id, unused, MDB_NEXT))
|
2021-03-03 01:15:12 +03:00
|
|
|
rooms.push_back(QString::fromStdString(std::string(room_id)));
|
2018-06-28 16:16:43 +03:00
|
|
|
|
|
|
|
roomsCursor.close();
|
|
|
|
txn.commit();
|
|
|
|
|
2020-10-28 15:06:28 +03:00
|
|
|
return rooms;
|
2018-06-28 16:16:43 +03:00
|
|
|
}
|
|
|
|
|
2019-08-13 05:09:40 +03:00
|
|
|
QMap<QString, mtx::responses::Notifications>
|
2019-08-10 06:34:44 +03:00
|
|
|
Cache::getTimelineMentions()
|
|
|
|
{
|
|
|
|
// TODO: Should be read-only, but getMentionsDb will attempt to create a DB
|
|
|
|
// if it doesn't exist, throwing an error.
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr);
|
|
|
|
|
2019-08-13 05:09:40 +03:00
|
|
|
QMap<QString, mtx::responses::Notifications> notifs;
|
2019-08-10 06:34:44 +03:00
|
|
|
|
|
|
|
auto room_ids = getRoomIds(txn);
|
|
|
|
|
|
|
|
for (const auto &room_id : room_ids) {
|
2019-08-13 05:09:40 +03:00
|
|
|
auto roomNotifs = getTimelineMentionsForRoom(txn, room_id);
|
|
|
|
notifs[QString::fromStdString(room_id)] = roomNotifs;
|
2019-08-10 06:34:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return notifs;
|
|
|
|
}
|
|
|
|
|
2020-07-13 01:08:58 +03:00
|
|
|
std::string
|
|
|
|
Cache::previousBatchToken(const std::string &room_id)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr);
|
|
|
|
auto orderDb = getEventOrderDb(txn, room_id);
|
|
|
|
|
|
|
|
auto cursor = lmdb::cursor::open(txn, orderDb);
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view indexVal, val;
|
2020-07-13 01:08:58 +03:00
|
|
|
if (!cursor.get(indexVal, val, MDB_FIRST)) {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
auto j = json::parse(val);
|
2020-07-13 01:08:58 +03:00
|
|
|
|
|
|
|
return j.value("prev_batch", "");
|
|
|
|
}
|
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
Cache::Messages
|
2020-07-13 01:08:58 +03:00
|
|
|
Cache::getTimelineMessages(lmdb::txn &txn, const std::string &room_id, uint64_t index, bool forward)
|
2018-06-28 16:16:43 +03:00
|
|
|
{
|
2020-02-24 03:07:25 +03:00
|
|
|
// TODO(nico): Limit the messages returned by this maybe?
|
2020-07-08 03:02:14 +03:00
|
|
|
auto orderDb = getOrderToMessageDb(txn, room_id);
|
2020-07-05 06:29:07 +03:00
|
|
|
auto eventsDb = getEventsDb(txn, room_id);
|
2018-06-28 16:16:43 +03:00
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
Messages messages{};
|
2018-06-28 16:16:43 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view indexVal, event_id;
|
2020-07-05 06:29:07 +03:00
|
|
|
|
|
|
|
auto cursor = lmdb::cursor::open(txn, orderDb);
|
2020-07-13 01:08:58 +03:00
|
|
|
if (index == std::numeric_limits<uint64_t>::max()) {
|
2020-07-08 03:02:14 +03:00
|
|
|
if (cursor.get(indexVal, event_id, forward ? MDB_FIRST : MDB_LAST)) {
|
2021-03-03 01:15:12 +03:00
|
|
|
index = lmdb::from_sv<uint64_t>(indexVal);
|
2020-07-05 06:29:07 +03:00
|
|
|
} else {
|
|
|
|
messages.end_of_cache = true;
|
|
|
|
return messages;
|
|
|
|
}
|
|
|
|
} else {
|
2020-07-08 03:02:14 +03:00
|
|
|
if (cursor.get(indexVal, event_id, MDB_SET)) {
|
2021-03-03 01:15:12 +03:00
|
|
|
index = lmdb::from_sv<uint64_t>(indexVal);
|
2020-07-05 06:29:07 +03:00
|
|
|
} else {
|
|
|
|
messages.end_of_cache = true;
|
|
|
|
return messages;
|
|
|
|
}
|
|
|
|
}
|
2018-06-28 16:16:43 +03:00
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
int counter = 0;
|
2018-06-28 16:16:43 +03:00
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
bool ret;
|
2020-07-10 00:15:22 +03:00
|
|
|
while ((ret = cursor.get(indexVal,
|
|
|
|
event_id,
|
|
|
|
counter == 0 ? (forward ? MDB_FIRST : MDB_LAST)
|
|
|
|
: (forward ? MDB_NEXT : MDB_PREV))) &&
|
2020-07-05 06:29:07 +03:00
|
|
|
counter++ < BATCH_SIZE) {
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event;
|
|
|
|
bool success = eventsDb.get(txn, event_id, event);
|
2020-07-05 06:29:07 +03:00
|
|
|
if (!success)
|
|
|
|
continue;
|
2018-06-28 16:16:43 +03:00
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
mtx::events::collections::TimelineEvent te;
|
2020-07-10 00:15:22 +03:00
|
|
|
try {
|
2021-03-03 01:15:12 +03:00
|
|
|
mtx::events::collections::from_json(json::parse(event), te);
|
2020-07-10 00:15:22 +03:00
|
|
|
} catch (std::exception &e) {
|
|
|
|
nhlog::db()->error("Failed to parse message from cache {}", e.what());
|
|
|
|
continue;
|
|
|
|
}
|
2018-06-28 16:16:43 +03:00
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
messages.timeline.events.push_back(std::move(te.data));
|
2018-06-28 16:16:43 +03:00
|
|
|
}
|
|
|
|
cursor.close();
|
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
// std::reverse(timeline.events.begin(), timeline.events.end());
|
2021-03-03 01:15:12 +03:00
|
|
|
messages.next_index = lmdb::from_sv<uint64_t>(indexVal);
|
2020-07-05 06:29:07 +03:00
|
|
|
messages.end_of_cache = !ret;
|
2018-06-28 16:16:43 +03:00
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
return messages;
|
2018-06-28 16:16:43 +03:00
|
|
|
}
|
|
|
|
|
2020-07-06 04:43:14 +03:00
|
|
|
std::optional<mtx::events::collections::TimelineEvent>
|
|
|
|
Cache::getEvent(const std::string &room_id, const std::string &event_id)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
auto eventsDb = getEventsDb(txn, room_id);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event{};
|
|
|
|
bool success = eventsDb.get(txn, event_id, event);
|
2020-07-06 04:43:14 +03:00
|
|
|
if (!success)
|
|
|
|
return {};
|
|
|
|
|
|
|
|
mtx::events::collections::TimelineEvent te;
|
2020-07-10 00:15:22 +03:00
|
|
|
try {
|
2021-03-03 01:15:12 +03:00
|
|
|
mtx::events::collections::from_json(json::parse(event), te);
|
2020-07-10 00:15:22 +03:00
|
|
|
} catch (std::exception &e) {
|
|
|
|
nhlog::db()->error("Failed to parse message from cache {}", e.what());
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
2020-07-06 04:43:14 +03:00
|
|
|
|
|
|
|
return te;
|
|
|
|
}
|
2020-07-10 02:37:55 +03:00
|
|
|
void
|
|
|
|
Cache::storeEvent(const std::string &room_id,
|
|
|
|
const std::string &event_id,
|
|
|
|
const mtx::events::collections::TimelineEvent &event)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
auto eventsDb = getEventsDb(txn, room_id);
|
|
|
|
auto event_json = mtx::accessors::serialize_event(event.data);
|
2021-03-03 01:15:12 +03:00
|
|
|
eventsDb.put(txn, event_id, event_json.dump());
|
2020-07-10 02:37:55 +03:00
|
|
|
txn.commit();
|
|
|
|
}
|
2020-07-06 04:43:14 +03:00
|
|
|
|
2020-07-19 13:22:54 +03:00
|
|
|
std::vector<std::string>
|
|
|
|
Cache::relatedEvents(const std::string &room_id, const std::string &event_id)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
auto relationsDb = getRelationsDb(txn, room_id);
|
|
|
|
|
|
|
|
std::vector<std::string> related_ids;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
auto related_cursor = lmdb::cursor::open(txn, relationsDb);
|
|
|
|
std::string_view related_to = event_id, related_event;
|
|
|
|
bool first = true;
|
2020-07-19 13:22:54 +03:00
|
|
|
|
|
|
|
try {
|
|
|
|
if (!related_cursor.get(related_to, related_event, MDB_SET))
|
|
|
|
return {};
|
|
|
|
|
|
|
|
while (related_cursor.get(
|
|
|
|
related_to, related_event, first ? MDB_FIRST_DUP : MDB_NEXT_DUP)) {
|
|
|
|
first = false;
|
|
|
|
if (event_id != std::string_view(related_to.data(), related_to.size()))
|
|
|
|
break;
|
|
|
|
|
|
|
|
related_ids.emplace_back(related_event.data(), related_event.size());
|
|
|
|
}
|
|
|
|
} catch (const lmdb::error &e) {
|
|
|
|
nhlog::db()->error("related events error: {}", e.what());
|
|
|
|
}
|
|
|
|
|
|
|
|
return related_ids;
|
|
|
|
}
|
|
|
|
|
2018-04-21 16:34:50 +03:00
|
|
|
QMap<QString, RoomInfo>
|
2018-04-22 11:27:00 +03:00
|
|
|
Cache::roomInfo(bool withInvites)
|
2018-04-21 16:34:50 +03:00
|
|
|
{
|
|
|
|
QMap<QString, RoomInfo> result;
|
|
|
|
|
2018-04-22 11:27:00 +03:00
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view room_id;
|
|
|
|
std::string_view room_data;
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
// Gather info about the joined rooms.
|
2018-04-22 11:27:00 +03:00
|
|
|
auto roomsCursor = lmdb::cursor::open(txn, roomsDb_);
|
2018-04-21 16:34:50 +03:00
|
|
|
while (roomsCursor.get(room_id, room_data, MDB_NEXT)) {
|
2018-04-30 21:41:47 +03:00
|
|
|
RoomInfo tmp = json::parse(std::move(room_data));
|
2021-03-03 01:15:12 +03:00
|
|
|
tmp.member_count = getMembersDb(txn, std::string(room_id)).size(txn);
|
|
|
|
tmp.msgInfo = getLastMessageInfo(txn, std::string(room_id));
|
2018-06-28 16:16:43 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
result.insert(QString::fromStdString(std::string(room_id)), std::move(tmp));
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
2018-04-22 11:27:00 +03:00
|
|
|
roomsCursor.close();
|
2018-04-21 16:34:50 +03:00
|
|
|
|
2018-04-22 11:27:00 +03:00
|
|
|
if (withInvites) {
|
|
|
|
// Gather info about the invites.
|
|
|
|
auto invitesCursor = lmdb::cursor::open(txn, invitesDb_);
|
|
|
|
while (invitesCursor.get(room_id, room_data, MDB_NEXT)) {
|
2018-04-30 21:41:47 +03:00
|
|
|
RoomInfo tmp = json::parse(room_data);
|
2021-03-03 01:15:12 +03:00
|
|
|
tmp.member_count = getInviteMembersDb(txn, std::string(room_id)).size(txn);
|
|
|
|
result.insert(QString::fromStdString(std::string(room_id)), std::move(tmp));
|
2018-04-22 11:27:00 +03:00
|
|
|
}
|
|
|
|
invitesCursor.close();
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2020-02-24 03:07:25 +03:00
|
|
|
std::string
|
|
|
|
Cache::getLastEventId(lmdb::txn &txn, const std::string &room_id)
|
|
|
|
{
|
2021-03-03 01:15:12 +03:00
|
|
|
lmdb::dbi orderDb;
|
2020-10-09 19:32:21 +03:00
|
|
|
try {
|
|
|
|
orderDb = getOrderToMessageDb(txn, room_id);
|
|
|
|
} catch (lmdb::runtime_error &e) {
|
|
|
|
nhlog::db()->error("Can't open db for room '{}', probably doesn't exist yet. ({})",
|
|
|
|
room_id,
|
|
|
|
e.what());
|
|
|
|
return {};
|
|
|
|
}
|
2020-02-24 03:07:25 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view indexVal, val;
|
2020-02-24 03:07:25 +03:00
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
auto cursor = lmdb::cursor::open(txn, orderDb);
|
|
|
|
if (!cursor.get(indexVal, val, MDB_LAST)) {
|
|
|
|
return {};
|
2020-02-24 03:07:25 +03:00
|
|
|
}
|
|
|
|
|
2020-07-08 03:02:14 +03:00
|
|
|
return std::string(val.data(), val.size());
|
2020-02-24 03:07:25 +03:00
|
|
|
}
|
|
|
|
|
2020-07-10 00:15:22 +03:00
|
|
|
std::optional<Cache::TimelineRange>
|
|
|
|
Cache::getTimelineRange(const std::string &room_id)
|
|
|
|
{
|
2020-07-20 19:25:22 +03:00
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
2021-03-03 01:15:12 +03:00
|
|
|
lmdb::dbi orderDb;
|
2020-07-20 19:25:22 +03:00
|
|
|
try {
|
|
|
|
orderDb = getOrderToMessageDb(txn, room_id);
|
|
|
|
} catch (lmdb::runtime_error &e) {
|
|
|
|
nhlog::db()->error("Can't open db for room '{}', probably doesn't exist yet. ({})",
|
|
|
|
room_id,
|
|
|
|
e.what());
|
|
|
|
return {};
|
|
|
|
}
|
2020-07-10 00:15:22 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view indexVal, val;
|
2020-07-10 00:15:22 +03:00
|
|
|
|
|
|
|
auto cursor = lmdb::cursor::open(txn, orderDb);
|
|
|
|
if (!cursor.get(indexVal, val, MDB_LAST)) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
TimelineRange range{};
|
2021-03-03 01:15:12 +03:00
|
|
|
range.last = lmdb::from_sv<uint64_t>(indexVal);
|
2020-07-10 00:15:22 +03:00
|
|
|
|
|
|
|
if (!cursor.get(indexVal, val, MDB_FIRST)) {
|
|
|
|
return {};
|
|
|
|
}
|
2021-03-03 01:15:12 +03:00
|
|
|
range.first = lmdb::from_sv<uint64_t>(indexVal);
|
2020-07-10 00:15:22 +03:00
|
|
|
|
|
|
|
return range;
|
|
|
|
}
|
2020-07-13 01:08:58 +03:00
|
|
|
std::optional<uint64_t>
|
2020-07-10 00:15:22 +03:00
|
|
|
Cache::getTimelineIndex(const std::string &room_id, std::string_view event_id)
|
|
|
|
{
|
2021-03-12 19:13:56 +03:00
|
|
|
if (event_id.empty() || room_id.empty())
|
|
|
|
return {};
|
|
|
|
|
2020-10-09 19:32:21 +03:00
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
lmdb::dbi orderDb;
|
2020-10-09 19:32:21 +03:00
|
|
|
try {
|
2020-10-13 01:14:28 +03:00
|
|
|
orderDb = getMessageToOrderDb(txn, room_id);
|
2020-10-09 19:32:21 +03:00
|
|
|
} catch (lmdb::runtime_error &e) {
|
|
|
|
nhlog::db()->error("Can't open db for room '{}', probably doesn't exist yet. ({})",
|
|
|
|
room_id,
|
|
|
|
e.what());
|
|
|
|
return {};
|
|
|
|
}
|
2020-07-10 00:15:22 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view indexVal{event_id.data(), event_id.size()}, val;
|
2020-07-10 00:15:22 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
bool success = orderDb.get(txn, indexVal, val);
|
2020-07-10 00:15:22 +03:00
|
|
|
if (!success) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
return lmdb::from_sv<uint64_t>(val);
|
2020-07-10 00:15:22 +03:00
|
|
|
}
|
|
|
|
|
2021-02-10 03:03:20 +03:00
|
|
|
std::optional<uint64_t>
|
|
|
|
Cache::getEventIndex(const std::string &room_id, std::string_view event_id)
|
|
|
|
{
|
|
|
|
if (event_id.empty())
|
|
|
|
return {};
|
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
lmdb::dbi orderDb;
|
2021-02-10 03:03:20 +03:00
|
|
|
try {
|
|
|
|
orderDb = getEventToOrderDb(txn, room_id);
|
|
|
|
} catch (lmdb::runtime_error &e) {
|
|
|
|
nhlog::db()->error("Can't open db for room '{}', probably doesn't exist yet. ({})",
|
|
|
|
room_id,
|
|
|
|
e.what());
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view val;
|
2021-02-10 03:03:20 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
bool success = orderDb.get(txn, event_id, val);
|
2021-02-10 03:03:20 +03:00
|
|
|
if (!success) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
return lmdb::from_sv<uint64_t>(val);
|
2021-02-10 03:03:20 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<std::pair<uint64_t, std::string>>
|
|
|
|
Cache::lastInvisibleEventAfter(const std::string &room_id, std::string_view event_id)
|
|
|
|
{
|
|
|
|
if (event_id.empty())
|
|
|
|
return {};
|
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
lmdb::dbi orderDb;
|
|
|
|
lmdb::dbi eventOrderDb;
|
|
|
|
lmdb::dbi timelineDb;
|
2021-02-10 03:03:20 +03:00
|
|
|
try {
|
|
|
|
orderDb = getEventToOrderDb(txn, room_id);
|
|
|
|
eventOrderDb = getEventOrderDb(txn, room_id);
|
|
|
|
timelineDb = getMessageToOrderDb(txn, room_id);
|
|
|
|
} catch (lmdb::runtime_error &e) {
|
|
|
|
nhlog::db()->error("Can't open db for room '{}', probably doesn't exist yet. ({})",
|
|
|
|
room_id,
|
|
|
|
e.what());
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view indexVal;
|
2021-02-10 03:03:20 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
bool success = orderDb.get(txn, event_id, indexVal);
|
2021-02-10 03:03:20 +03:00
|
|
|
if (!success) {
|
|
|
|
return {};
|
|
|
|
}
|
2021-03-03 01:15:12 +03:00
|
|
|
uint64_t prevIdx = lmdb::from_sv<uint64_t>(indexVal);
|
|
|
|
std::string prevId{event_id};
|
2021-02-10 03:03:20 +03:00
|
|
|
|
|
|
|
auto cursor = lmdb::cursor::open(txn, eventOrderDb);
|
|
|
|
cursor.get(indexVal, MDB_SET);
|
2021-03-03 01:15:12 +03:00
|
|
|
while (cursor.get(indexVal, event_id, MDB_NEXT)) {
|
|
|
|
std::string evId = json::parse(event_id)["event_id"].get<std::string>();
|
|
|
|
std::string_view temp;
|
|
|
|
if (timelineDb.get(txn, evId, temp)) {
|
2021-02-10 03:03:20 +03:00
|
|
|
return std::pair{prevIdx, std::string(prevId)};
|
|
|
|
} else {
|
2021-03-03 01:15:12 +03:00
|
|
|
prevIdx = lmdb::from_sv<uint64_t>(indexVal);
|
2021-02-10 03:03:20 +03:00
|
|
|
prevId = std::move(evId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return std::pair{prevIdx, std::string(prevId)};
|
|
|
|
}
|
|
|
|
|
2021-01-27 04:45:33 +03:00
|
|
|
std::optional<uint64_t>
|
|
|
|
Cache::getArrivalIndex(const std::string &room_id, std::string_view event_id)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
lmdb::dbi orderDb;
|
2021-01-27 04:45:33 +03:00
|
|
|
try {
|
|
|
|
orderDb = getEventToOrderDb(txn, room_id);
|
|
|
|
} catch (lmdb::runtime_error &e) {
|
|
|
|
nhlog::db()->error("Can't open db for room '{}', probably doesn't exist yet. ({})",
|
|
|
|
room_id,
|
|
|
|
e.what());
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view val;
|
2021-01-27 04:45:33 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
bool success = orderDb.get(txn, event_id, val);
|
2021-01-27 04:45:33 +03:00
|
|
|
if (!success) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
return lmdb::from_sv<uint64_t>(val);
|
2021-01-27 04:45:33 +03:00
|
|
|
}
|
|
|
|
|
2020-07-10 00:15:22 +03:00
|
|
|
std::optional<std::string>
|
2020-07-13 01:08:58 +03:00
|
|
|
Cache::getTimelineEventId(const std::string &room_id, uint64_t index)
|
2020-07-10 00:15:22 +03:00
|
|
|
{
|
2020-10-09 19:32:21 +03:00
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
2021-03-03 01:15:12 +03:00
|
|
|
lmdb::dbi orderDb;
|
2020-10-09 19:32:21 +03:00
|
|
|
try {
|
|
|
|
orderDb = getOrderToMessageDb(txn, room_id);
|
|
|
|
} catch (lmdb::runtime_error &e) {
|
|
|
|
nhlog::db()->error("Can't open db for room '{}', probably doesn't exist yet. ({})",
|
|
|
|
room_id,
|
|
|
|
e.what());
|
|
|
|
return {};
|
|
|
|
}
|
2020-07-10 00:15:22 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view val;
|
2020-07-10 00:15:22 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
bool success = orderDb.get(txn, lmdb::to_sv(index), val);
|
2020-07-10 00:15:22 +03:00
|
|
|
if (!success) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
return std::string(val);
|
2020-07-10 00:15:22 +03:00
|
|
|
}
|
|
|
|
|
2018-06-28 16:16:43 +03:00
|
|
|
DescInfo
|
|
|
|
Cache::getLastMessageInfo(lmdb::txn &txn, const std::string &room_id)
|
|
|
|
{
|
2021-03-03 01:15:12 +03:00
|
|
|
lmdb::dbi orderDb;
|
2020-10-09 19:32:21 +03:00
|
|
|
try {
|
|
|
|
orderDb = getOrderToMessageDb(txn, room_id);
|
|
|
|
} catch (lmdb::runtime_error &e) {
|
|
|
|
nhlog::db()->error("Can't open db for room '{}', probably doesn't exist yet. ({})",
|
|
|
|
room_id,
|
|
|
|
e.what());
|
|
|
|
return {};
|
|
|
|
}
|
2021-03-03 01:15:12 +03:00
|
|
|
|
|
|
|
lmdb::dbi eventsDb;
|
2020-10-09 19:32:21 +03:00
|
|
|
try {
|
|
|
|
eventsDb = getEventsDb(txn, room_id);
|
|
|
|
} catch (lmdb::runtime_error &e) {
|
|
|
|
nhlog::db()->error("Can't open db for room '{}', probably doesn't exist yet. ({})",
|
|
|
|
room_id,
|
|
|
|
e.what());
|
|
|
|
return {};
|
|
|
|
}
|
2020-11-26 01:43:31 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
lmdb::dbi membersdb;
|
2020-11-26 01:43:31 +03:00
|
|
|
try {
|
|
|
|
membersdb = getMembersDb(txn, room_id);
|
|
|
|
} catch (lmdb::runtime_error &e) {
|
|
|
|
nhlog::db()->error("Can't open db for room '{}', probably doesn't exist yet. ({})",
|
|
|
|
room_id,
|
|
|
|
e.what());
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
if (orderDb.size(txn) == 0)
|
2018-06-28 16:16:43 +03:00
|
|
|
return DescInfo{};
|
|
|
|
|
2020-02-24 03:07:25 +03:00
|
|
|
const auto local_user = utils::localUser();
|
2018-06-28 16:16:43 +03:00
|
|
|
|
2020-05-04 21:17:57 +03:00
|
|
|
DescInfo fallbackDesc{};
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view indexVal, event_id;
|
2018-06-28 16:16:43 +03:00
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
auto cursor = lmdb::cursor::open(txn, orderDb);
|
2020-07-10 00:15:22 +03:00
|
|
|
bool first = true;
|
|
|
|
while (cursor.get(indexVal, event_id, first ? MDB_LAST : MDB_PREV)) {
|
|
|
|
first = false;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event;
|
|
|
|
bool success = eventsDb.get(txn, event_id, event);
|
2020-07-05 06:29:07 +03:00
|
|
|
if (!success)
|
2020-05-04 21:17:57 +03:00
|
|
|
continue;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
auto obj = json::parse(event);
|
2020-07-05 06:29:07 +03:00
|
|
|
|
|
|
|
if (fallbackDesc.event_id.isEmpty() && obj["type"] == "m.room.member" &&
|
|
|
|
obj["state_key"] == local_user.toStdString() &&
|
|
|
|
obj["content"]["membership"] == "join") {
|
|
|
|
uint64_t ts = obj["origin_server_ts"];
|
2020-05-04 21:17:57 +03:00
|
|
|
auto time = QDateTime::fromMSecsSinceEpoch(ts);
|
2020-07-05 06:29:07 +03:00
|
|
|
fallbackDesc = DescInfo{QString::fromStdString(obj["event_id"]),
|
2020-05-04 21:17:57 +03:00
|
|
|
local_user,
|
2020-05-10 02:38:40 +03:00
|
|
|
tr("You joined this room."),
|
2020-05-04 21:17:57 +03:00
|
|
|
utils::descriptiveTime(time),
|
|
|
|
ts,
|
|
|
|
time};
|
|
|
|
}
|
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
if (!(obj["type"] == "m.room.message" || obj["type"] == "m.sticker" ||
|
2020-08-17 21:40:33 +03:00
|
|
|
obj["type"] == "m.call.invite" || obj["type"] == "m.call.answer" ||
|
|
|
|
obj["type"] == "m.call.hangup" || obj["type"] == "m.room.encrypted"))
|
2018-06-28 16:16:43 +03:00
|
|
|
continue;
|
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
mtx::events::collections::TimelineEvent te;
|
|
|
|
mtx::events::collections::from_json(obj, te);
|
2018-06-28 16:16:43 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view info;
|
2020-11-26 01:43:31 +03:00
|
|
|
MemberInfo m;
|
2021-03-03 01:15:12 +03:00
|
|
|
if (membersdb.get(txn, obj["sender"].get<std::string>(), info)) {
|
2020-11-26 01:43:31 +03:00
|
|
|
m = json::parse(std::string_view(info.data(), info.size()));
|
|
|
|
}
|
|
|
|
|
2018-06-28 16:16:43 +03:00
|
|
|
cursor.close();
|
|
|
|
return utils::getMessageDescription(
|
2020-11-26 01:43:31 +03:00
|
|
|
te.data, local_user, QString::fromStdString(m.name));
|
2018-06-28 16:16:43 +03:00
|
|
|
}
|
|
|
|
cursor.close();
|
|
|
|
|
2020-05-04 21:17:57 +03:00
|
|
|
return fallbackDesc;
|
2018-06-28 16:16:43 +03:00
|
|
|
}
|
|
|
|
|
2021-05-24 15:04:07 +03:00
|
|
|
QHash<QString, RoomInfo>
|
2018-04-22 14:19:05 +03:00
|
|
|
Cache::invites()
|
|
|
|
{
|
2021-05-24 15:04:07 +03:00
|
|
|
QHash<QString, RoomInfo> result;
|
2018-04-22 14:19:05 +03:00
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
auto cursor = lmdb::cursor::open(txn, invitesDb_);
|
|
|
|
|
2021-05-24 15:04:07 +03:00
|
|
|
std::string_view room_id, room_data;
|
2018-04-22 14:19:05 +03:00
|
|
|
|
2021-05-24 15:04:07 +03:00
|
|
|
while (cursor.get(room_id, room_data, MDB_NEXT)) {
|
|
|
|
try {
|
|
|
|
RoomInfo tmp = json::parse(room_data);
|
|
|
|
tmp.member_count = getInviteMembersDb(txn, std::string(room_id)).size(txn);
|
|
|
|
result.insert(QString::fromStdString(std::string(room_id)), std::move(tmp));
|
|
|
|
} catch (const json::exception &e) {
|
|
|
|
nhlog::db()->warn("failed to parse room info for invite: "
|
|
|
|
"room_id ({}), {}: {}",
|
|
|
|
room_id,
|
|
|
|
std::string(room_data),
|
|
|
|
e.what());
|
|
|
|
}
|
|
|
|
}
|
2018-04-22 14:19:05 +03:00
|
|
|
|
|
|
|
cursor.close();
|
2021-05-24 15:04:07 +03:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<RoomInfo>
|
|
|
|
Cache::invite(std::string_view roomid)
|
|
|
|
{
|
|
|
|
std::optional<RoomInfo> result;
|
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
|
|
|
|
std::string_view room_data;
|
|
|
|
|
|
|
|
if (invitesDb_.get(txn, roomid, room_data)) {
|
|
|
|
try {
|
|
|
|
RoomInfo tmp = json::parse(room_data);
|
|
|
|
tmp.member_count = getInviteMembersDb(txn, std::string(roomid)).size(txn);
|
|
|
|
result = std::move(tmp);
|
|
|
|
} catch (const json::exception &e) {
|
|
|
|
nhlog::db()->warn("failed to parse room info for invite: "
|
|
|
|
"room_id ({}), {}: {}",
|
|
|
|
roomid,
|
|
|
|
std::string(room_data),
|
|
|
|
e.what());
|
|
|
|
}
|
|
|
|
}
|
2018-04-22 14:19:05 +03:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2018-04-21 16:34:50 +03:00
|
|
|
QString
|
2020-11-26 00:45:33 +03:00
|
|
|
Cache::getRoomAvatarUrl(lmdb::txn &txn, lmdb::dbi &statesdb, lmdb::dbi &membersdb)
|
2018-04-21 16:34:50 +03:00
|
|
|
{
|
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event;
|
|
|
|
bool res = statesdb.get(txn, to_string(mtx::events::EventType::RoomAvatar), event);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
if (res) {
|
|
|
|
try {
|
|
|
|
StateEvent<Avatar> msg =
|
2020-07-06 04:43:14 +03:00
|
|
|
json::parse(std::string_view(event.data(), event.size()));
|
2018-04-21 16:34:50 +03:00
|
|
|
|
2020-03-22 22:43:45 +03:00
|
|
|
if (!msg.content.url.empty())
|
|
|
|
return QString::fromStdString(msg.content.url);
|
2018-04-21 16:34:50 +03:00
|
|
|
} catch (const json::exception &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->warn("failed to parse m.room.avatar event: {}", e.what());
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We don't use an avatar for group chats.
|
|
|
|
if (membersdb.size(txn) > 2)
|
|
|
|
return QString();
|
|
|
|
|
|
|
|
auto cursor = lmdb::cursor::open(txn, membersdb);
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view user_id;
|
|
|
|
std::string_view member_data;
|
2020-11-26 00:45:33 +03:00
|
|
|
std::string fallback_url;
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
// Resolve avatar for 1-1 chats.
|
|
|
|
while (cursor.get(user_id, member_data, MDB_NEXT)) {
|
|
|
|
try {
|
|
|
|
MemberInfo m = json::parse(member_data);
|
2020-11-26 00:45:33 +03:00
|
|
|
if (user_id == localUserId_.toStdString()) {
|
|
|
|
fallback_url = m.avatar_url;
|
|
|
|
continue;
|
|
|
|
}
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
cursor.close();
|
|
|
|
return QString::fromStdString(m.avatar_url);
|
|
|
|
} catch (const json::exception &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->warn("failed to parse member info: {}", e.what());
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cursor.close();
|
|
|
|
|
|
|
|
// Default case when there is only one member.
|
2020-11-26 00:45:33 +03:00
|
|
|
return QString::fromStdString(fallback_url);
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
QString
|
|
|
|
Cache::getRoomName(lmdb::txn &txn, lmdb::dbi &statesdb, lmdb::dbi &membersdb)
|
|
|
|
{
|
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event;
|
|
|
|
bool res = statesdb.get(txn, to_string(mtx::events::EventType::RoomName), event);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
if (res) {
|
|
|
|
try {
|
2020-07-06 04:43:14 +03:00
|
|
|
StateEvent<Name> msg =
|
|
|
|
json::parse(std::string_view(event.data(), event.size()));
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
if (!msg.content.name.empty())
|
|
|
|
return QString::fromStdString(msg.content.name);
|
|
|
|
} catch (const json::exception &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->warn("failed to parse m.room.name event: {}", e.what());
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
res = statesdb.get(txn, to_string(mtx::events::EventType::RoomCanonicalAlias), event);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
if (res) {
|
|
|
|
try {
|
|
|
|
StateEvent<CanonicalAlias> msg =
|
2020-07-06 04:43:14 +03:00
|
|
|
json::parse(std::string_view(event.data(), event.size()));
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
if (!msg.content.alias.empty())
|
|
|
|
return QString::fromStdString(msg.content.alias);
|
|
|
|
} catch (const json::exception &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->warn("failed to parse m.room.canonical_alias event: {}",
|
|
|
|
e.what());
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-25 06:11:47 +03:00
|
|
|
auto cursor = lmdb::cursor::open(txn, membersdb);
|
2020-12-25 03:08:06 +03:00
|
|
|
const auto total = membersdb.size(txn);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
std::size_t ii = 0;
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view user_id;
|
|
|
|
std::string_view member_data;
|
2018-04-21 16:34:50 +03:00
|
|
|
std::map<std::string, MemberInfo> members;
|
|
|
|
|
|
|
|
while (cursor.get(user_id, member_data, MDB_NEXT) && ii < 3) {
|
|
|
|
try {
|
|
|
|
members.emplace(user_id, json::parse(member_data));
|
|
|
|
} catch (const json::exception &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->warn("failed to parse member info: {}", e.what());
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
ii++;
|
|
|
|
}
|
|
|
|
|
|
|
|
cursor.close();
|
|
|
|
|
|
|
|
if (total == 1 && !members.empty())
|
|
|
|
return QString::fromStdString(members.begin()->second.name);
|
|
|
|
|
|
|
|
auto first_member = [&members, this]() {
|
|
|
|
for (const auto &m : members) {
|
|
|
|
if (m.first != localUserId_.toStdString())
|
|
|
|
return QString::fromStdString(m.second.name);
|
|
|
|
}
|
|
|
|
|
|
|
|
return localUserId_;
|
|
|
|
}();
|
|
|
|
|
|
|
|
if (total == 2)
|
|
|
|
return first_member;
|
|
|
|
else if (total > 2)
|
2020-03-03 04:30:19 +03:00
|
|
|
return QString("%1 and %2 others").arg(first_member).arg(total - 1);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
return "Empty Room";
|
|
|
|
}
|
|
|
|
|
2019-12-15 01:39:02 +03:00
|
|
|
mtx::events::state::JoinRule
|
2018-05-13 01:31:58 +03:00
|
|
|
Cache::getRoomJoinRule(lmdb::txn &txn, lmdb::dbi &statesdb)
|
|
|
|
{
|
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event;
|
|
|
|
bool res = statesdb.get(txn, to_string(mtx::events::EventType::RoomJoinRules), event);
|
2018-05-13 01:31:58 +03:00
|
|
|
|
|
|
|
if (res) {
|
|
|
|
try {
|
2021-03-03 01:15:12 +03:00
|
|
|
StateEvent<state::JoinRules> msg = json::parse(event);
|
2018-05-13 01:31:58 +03:00
|
|
|
return msg.content.join_rule;
|
|
|
|
} catch (const json::exception &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->warn("failed to parse m.room.join_rule event: {}", e.what());
|
2018-05-13 01:31:58 +03:00
|
|
|
}
|
|
|
|
}
|
2019-12-15 01:39:02 +03:00
|
|
|
return state::JoinRule::Knock;
|
2018-05-13 01:31:58 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
Cache::getRoomGuestAccess(lmdb::txn &txn, lmdb::dbi &statesdb)
|
|
|
|
{
|
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event;
|
|
|
|
bool res = statesdb.get(txn, to_string(mtx::events::EventType::RoomGuestAccess), event);
|
2018-05-13 01:31:58 +03:00
|
|
|
|
|
|
|
if (res) {
|
|
|
|
try {
|
2021-03-03 01:15:12 +03:00
|
|
|
StateEvent<GuestAccess> msg = json::parse(event);
|
2018-05-13 01:31:58 +03:00
|
|
|
return msg.content.guest_access == AccessState::CanJoin;
|
|
|
|
} catch (const json::exception &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->warn("failed to parse m.room.guest_access event: {}",
|
|
|
|
e.what());
|
2018-05-13 01:31:58 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-04-21 16:34:50 +03:00
|
|
|
QString
|
|
|
|
Cache::getRoomTopic(lmdb::txn &txn, lmdb::dbi &statesdb)
|
|
|
|
{
|
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event;
|
|
|
|
bool res = statesdb.get(txn, to_string(mtx::events::EventType::RoomTopic), event);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
if (res) {
|
|
|
|
try {
|
2021-03-03 01:15:12 +03:00
|
|
|
StateEvent<Topic> msg = json::parse(event);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
if (!msg.content.topic.empty())
|
|
|
|
return QString::fromStdString(msg.content.topic);
|
|
|
|
} catch (const json::exception &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->warn("failed to parse m.room.topic event: {}", e.what());
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
2019-06-27 21:53:44 +03:00
|
|
|
QString
|
|
|
|
Cache::getRoomVersion(lmdb::txn &txn, lmdb::dbi &statesdb)
|
|
|
|
{
|
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event;
|
|
|
|
bool res = statesdb.get(txn, to_string(mtx::events::EventType::RoomCreate), event);
|
2019-06-27 21:53:44 +03:00
|
|
|
|
|
|
|
if (res) {
|
|
|
|
try {
|
2021-03-03 01:15:12 +03:00
|
|
|
StateEvent<Create> msg = json::parse(event);
|
2019-06-27 21:53:44 +03:00
|
|
|
|
|
|
|
if (!msg.content.room_version.empty())
|
|
|
|
return QString::fromStdString(msg.content.room_version);
|
|
|
|
} catch (const json::exception &e) {
|
|
|
|
nhlog::db()->warn("failed to parse m.room.create event: {}", e.what());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-04 20:18:32 +03:00
|
|
|
nhlog::db()->warn("m.room.create event is missing room version, assuming version \"1\"");
|
2019-06-27 21:53:44 +03:00
|
|
|
return QString("1");
|
|
|
|
}
|
|
|
|
|
2021-01-10 20:36:06 +03:00
|
|
|
std::optional<mtx::events::state::CanonicalAlias>
|
|
|
|
Cache::getRoomAliases(const std::string &roomid)
|
|
|
|
{
|
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
auto statesdb = getStatesDb(txn, roomid);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event;
|
|
|
|
bool res = statesdb.get(txn, to_string(mtx::events::EventType::RoomCanonicalAlias), event);
|
2021-01-10 20:36:06 +03:00
|
|
|
|
|
|
|
if (res) {
|
|
|
|
try {
|
2021-03-03 01:15:12 +03:00
|
|
|
StateEvent<CanonicalAlias> msg = json::parse(event);
|
2021-01-10 20:36:06 +03:00
|
|
|
|
|
|
|
return msg.content;
|
|
|
|
} catch (const json::exception &e) {
|
|
|
|
nhlog::db()->warn("failed to parse m.room.canonical_alias event: {}",
|
|
|
|
e.what());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
2018-04-21 16:34:50 +03:00
|
|
|
QString
|
|
|
|
Cache::getInviteRoomName(lmdb::txn &txn, lmdb::dbi &statesdb, lmdb::dbi &membersdb)
|
|
|
|
{
|
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event;
|
|
|
|
bool res = statesdb.get(txn, to_string(mtx::events::EventType::RoomName), event);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
if (res) {
|
|
|
|
try {
|
2021-03-03 01:15:12 +03:00
|
|
|
StrippedEvent<state::Name> msg = json::parse(event);
|
2018-04-21 16:34:50 +03:00
|
|
|
return QString::fromStdString(msg.content.name);
|
|
|
|
} catch (const json::exception &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->warn("failed to parse m.room.name event: {}", e.what());
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
auto cursor = lmdb::cursor::open(txn, membersdb);
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view user_id, member_data;
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
while (cursor.get(user_id, member_data, MDB_NEXT)) {
|
|
|
|
if (user_id == localUserId_.toStdString())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
try {
|
|
|
|
MemberInfo tmp = json::parse(member_data);
|
|
|
|
cursor.close();
|
|
|
|
|
|
|
|
return QString::fromStdString(tmp.name);
|
|
|
|
} catch (const json::exception &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->warn("failed to parse member info: {}", e.what());
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cursor.close();
|
|
|
|
|
|
|
|
return QString("Empty Room");
|
|
|
|
}
|
|
|
|
|
|
|
|
QString
|
|
|
|
Cache::getInviteRoomAvatarUrl(lmdb::txn &txn, lmdb::dbi &statesdb, lmdb::dbi &membersdb)
|
|
|
|
{
|
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event;
|
|
|
|
bool res = statesdb.get(txn, to_string(mtx::events::EventType::RoomAvatar), event);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
if (res) {
|
|
|
|
try {
|
2021-03-03 01:15:12 +03:00
|
|
|
StrippedEvent<state::Avatar> msg = json::parse(event);
|
2018-04-21 16:34:50 +03:00
|
|
|
return QString::fromStdString(msg.content.url);
|
|
|
|
} catch (const json::exception &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->warn("failed to parse m.room.avatar event: {}", e.what());
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
auto cursor = lmdb::cursor::open(txn, membersdb);
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view user_id, member_data;
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
while (cursor.get(user_id, member_data, MDB_NEXT)) {
|
|
|
|
if (user_id == localUserId_.toStdString())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
try {
|
|
|
|
MemberInfo tmp = json::parse(member_data);
|
|
|
|
cursor.close();
|
|
|
|
|
|
|
|
return QString::fromStdString(tmp.avatar_url);
|
|
|
|
} catch (const json::exception &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->warn("failed to parse member info: {}", e.what());
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cursor.close();
|
|
|
|
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
|
|
|
QString
|
|
|
|
Cache::getInviteRoomTopic(lmdb::txn &txn, lmdb::dbi &db)
|
|
|
|
{
|
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event;
|
|
|
|
bool res = db.get(txn, to_string(mtx::events::EventType::RoomTopic), event);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
if (res) {
|
|
|
|
try {
|
2021-03-03 01:15:12 +03:00
|
|
|
StrippedEvent<Topic> msg = json::parse(event);
|
2018-04-21 16:34:50 +03:00
|
|
|
return QString::fromStdString(msg.content.topic);
|
|
|
|
} catch (const json::exception &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->warn("failed to parse m.room.topic event: {}", e.what());
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::string>
|
|
|
|
Cache::joinedRooms()
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
auto roomsCursor = lmdb::cursor::open(txn, roomsDb_);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view id, data;
|
2018-04-21 16:34:50 +03:00
|
|
|
std::vector<std::string> room_ids;
|
|
|
|
|
|
|
|
// Gather the room ids for the joined rooms.
|
|
|
|
while (roomsCursor.get(id, data, MDB_NEXT))
|
|
|
|
room_ids.emplace_back(id);
|
|
|
|
|
|
|
|
roomsCursor.close();
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return room_ids;
|
|
|
|
}
|
|
|
|
|
2020-10-22 21:48:28 +03:00
|
|
|
std::optional<MemberInfo>
|
|
|
|
Cache::getMember(const std::string &room_id, const std::string &user_id)
|
2018-04-21 16:34:50 +03:00
|
|
|
{
|
2021-05-15 00:35:34 +03:00
|
|
|
if (user_id.empty() || !env_.handle())
|
2021-04-16 18:09:38 +03:00
|
|
|
return std::nullopt;
|
|
|
|
|
2020-10-22 21:48:28 +03:00
|
|
|
try {
|
2020-10-24 19:07:14 +03:00
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
2020-10-22 21:48:28 +03:00
|
|
|
auto membersdb = getMembersDb(txn, room_id);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view info;
|
|
|
|
if (membersdb.get(txn, user_id, info)) {
|
|
|
|
MemberInfo m = json::parse(info);
|
2020-10-22 21:48:28 +03:00
|
|
|
return m;
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
2020-11-26 00:45:33 +03:00
|
|
|
} catch (std::exception &e) {
|
|
|
|
nhlog::db()->warn("Failed to read member ({}): {}", user_id, e.what());
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
2020-10-22 21:48:28 +03:00
|
|
|
return std::nullopt;
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
|
|
|
|
2018-04-27 01:57:46 +03:00
|
|
|
std::vector<RoomSearchResult>
|
|
|
|
Cache::searchRooms(const std::string &query, std::uint8_t max_items)
|
|
|
|
{
|
|
|
|
std::multimap<int, std::pair<std::string, RoomInfo>> items;
|
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
auto cursor = lmdb::cursor::open(txn, roomsDb_);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view room_id, room_data;
|
2018-04-27 01:57:46 +03:00
|
|
|
while (cursor.get(room_id, room_data, MDB_NEXT)) {
|
2021-03-03 01:15:12 +03:00
|
|
|
RoomInfo tmp = json::parse(room_data);
|
2018-04-27 01:57:46 +03:00
|
|
|
|
|
|
|
const int score = utils::levenshtein_distance(
|
|
|
|
query, QString::fromStdString(tmp.name).toLower().toStdString());
|
|
|
|
items.emplace(score, std::make_pair(room_id, tmp));
|
|
|
|
}
|
|
|
|
|
|
|
|
cursor.close();
|
|
|
|
|
|
|
|
auto end = items.begin();
|
|
|
|
|
|
|
|
if (items.size() >= max_items)
|
|
|
|
std::advance(end, max_items);
|
|
|
|
else if (items.size() > 0)
|
|
|
|
std::advance(end, items.size());
|
|
|
|
|
|
|
|
std::vector<RoomSearchResult> results;
|
|
|
|
for (auto it = items.begin(); it != end; it++) {
|
2019-08-26 02:24:56 +03:00
|
|
|
results.push_back(RoomSearchResult{it->second.first, it->second.second});
|
2018-04-27 01:57:46 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return results;
|
|
|
|
}
|
|
|
|
|
2018-05-01 19:35:28 +03:00
|
|
|
std::vector<RoomMember>
|
|
|
|
Cache::getMembers(const std::string &room_id, std::size_t startIndex, std::size_t len)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
auto db = getMembersDb(txn, room_id);
|
|
|
|
auto cursor = lmdb::cursor::open(txn, db);
|
|
|
|
|
|
|
|
std::size_t currentIndex = 0;
|
|
|
|
|
|
|
|
const auto endIndex = std::min(startIndex + len, db.size(txn));
|
|
|
|
|
|
|
|
std::vector<RoomMember> members;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view user_id, user_data;
|
2018-05-01 19:35:28 +03:00
|
|
|
while (cursor.get(user_id, user_data, MDB_NEXT)) {
|
|
|
|
if (currentIndex < startIndex) {
|
|
|
|
currentIndex += 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (currentIndex >= endIndex)
|
|
|
|
break;
|
|
|
|
|
|
|
|
try {
|
|
|
|
MemberInfo tmp = json::parse(user_data);
|
|
|
|
members.emplace_back(
|
2021-03-03 01:15:12 +03:00
|
|
|
RoomMember{QString::fromStdString(std::string(user_id)),
|
2021-03-17 21:08:17 +03:00
|
|
|
QString::fromStdString(tmp.name)});
|
2018-05-01 19:35:28 +03:00
|
|
|
} catch (const json::exception &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->warn("{}", e.what());
|
2018-05-01 19:35:28 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
currentIndex += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
cursor.close();
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return members;
|
|
|
|
}
|
|
|
|
|
2018-07-21 21:40:11 +03:00
|
|
|
bool
|
|
|
|
Cache::isRoomMember(const std::string &user_id, const std::string &room_id)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
auto db = getMembersDb(txn, room_id);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view value;
|
|
|
|
bool res = db.get(txn, user_id, value);
|
2018-07-21 21:40:11 +03:00
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2020-07-18 18:43:49 +03:00
|
|
|
void
|
|
|
|
Cache::savePendingMessage(const std::string &room_id,
|
|
|
|
const mtx::events::collections::TimelineEvent &message)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
|
|
|
|
mtx::responses::Timeline timeline;
|
|
|
|
timeline.events.push_back(message.data);
|
|
|
|
saveTimelineMessages(txn, room_id, timeline);
|
|
|
|
|
|
|
|
auto pending = getPendingMessagesDb(txn, room_id);
|
|
|
|
|
|
|
|
int64_t now = QDateTime::currentMSecsSinceEpoch();
|
2021-03-03 01:15:12 +03:00
|
|
|
pending.put(txn, lmdb::to_sv(now), mtx::accessors::event_id(message.data));
|
2020-07-18 18:43:49 +03:00
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<mtx::events::collections::TimelineEvent>
|
|
|
|
Cache::firstPendingMessage(const std::string &room_id)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
auto pending = getPendingMessagesDb(txn, room_id);
|
|
|
|
|
2020-07-30 19:13:19 +03:00
|
|
|
{
|
|
|
|
auto pendingCursor = lmdb::cursor::open(txn, pending);
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view tsIgnored, pendingTxn;
|
2020-07-30 19:13:19 +03:00
|
|
|
while (pendingCursor.get(tsIgnored, pendingTxn, MDB_NEXT)) {
|
|
|
|
auto eventsDb = getEventsDb(txn, room_id);
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event;
|
|
|
|
if (!eventsDb.get(txn, pendingTxn, event)) {
|
|
|
|
pending.del(txn, tsIgnored, pendingTxn);
|
2020-07-30 19:13:19 +03:00
|
|
|
continue;
|
|
|
|
}
|
2020-07-18 18:43:49 +03:00
|
|
|
|
2020-07-30 19:13:19 +03:00
|
|
|
try {
|
|
|
|
mtx::events::collections::TimelineEvent te;
|
2021-03-03 01:15:12 +03:00
|
|
|
mtx::events::collections::from_json(json::parse(event), te);
|
2020-07-30 19:13:19 +03:00
|
|
|
|
|
|
|
pendingCursor.close();
|
|
|
|
txn.commit();
|
|
|
|
return te;
|
|
|
|
} catch (std::exception &e) {
|
|
|
|
nhlog::db()->error("Failed to parse message from cache {}",
|
|
|
|
e.what());
|
2021-03-03 01:15:12 +03:00
|
|
|
pending.del(txn, tsIgnored, pendingTxn);
|
2020-07-30 19:13:19 +03:00
|
|
|
continue;
|
|
|
|
}
|
2020-07-18 18:43:49 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Cache::removePendingStatus(const std::string &room_id, const std::string &txn_id)
|
|
|
|
{
|
2020-07-30 19:13:19 +03:00
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
auto pending = getPendingMessagesDb(txn, room_id);
|
|
|
|
|
|
|
|
{
|
|
|
|
auto pendingCursor = lmdb::cursor::open(txn, pending);
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view tsIgnored, pendingTxn;
|
2020-07-30 19:13:19 +03:00
|
|
|
while (pendingCursor.get(tsIgnored, pendingTxn, MDB_NEXT)) {
|
|
|
|
if (std::string_view(pendingTxn.data(), pendingTxn.size()) == txn_id)
|
|
|
|
lmdb::cursor_del(pendingCursor);
|
|
|
|
}
|
2020-07-18 18:43:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
}
|
|
|
|
|
2018-06-28 16:16:43 +03:00
|
|
|
void
|
|
|
|
Cache::saveTimelineMessages(lmdb::txn &txn,
|
|
|
|
const std::string &room_id,
|
|
|
|
const mtx::responses::Timeline &res)
|
|
|
|
{
|
2020-07-13 01:08:58 +03:00
|
|
|
if (res.events.empty())
|
|
|
|
return;
|
|
|
|
|
2020-07-08 03:02:14 +03:00
|
|
|
auto eventsDb = getEventsDb(txn, room_id);
|
|
|
|
auto relationsDb = getRelationsDb(txn, room_id);
|
2018-06-28 16:16:43 +03:00
|
|
|
|
2020-07-08 03:02:14 +03:00
|
|
|
auto orderDb = getEventOrderDb(txn, room_id);
|
2020-07-18 18:43:49 +03:00
|
|
|
auto evToOrderDb = getEventToOrderDb(txn, room_id);
|
2020-07-08 03:02:14 +03:00
|
|
|
auto msg2orderDb = getMessageToOrderDb(txn, room_id);
|
|
|
|
auto order2msgDb = getOrderToMessageDb(txn, room_id);
|
2020-07-18 18:43:49 +03:00
|
|
|
auto pending = getPendingMessagesDb(txn, room_id);
|
|
|
|
|
2020-07-08 03:02:14 +03:00
|
|
|
if (res.limited) {
|
2020-07-04 03:09:12 +03:00
|
|
|
lmdb::dbi_drop(txn, orderDb, false);
|
2020-07-18 18:43:49 +03:00
|
|
|
lmdb::dbi_drop(txn, evToOrderDb, false);
|
2020-07-08 03:02:14 +03:00
|
|
|
lmdb::dbi_drop(txn, msg2orderDb, false);
|
|
|
|
lmdb::dbi_drop(txn, order2msgDb, false);
|
2020-07-18 18:43:49 +03:00
|
|
|
lmdb::dbi_drop(txn, pending, true);
|
2020-07-08 03:02:14 +03:00
|
|
|
}
|
2020-07-04 03:09:12 +03:00
|
|
|
|
2018-06-28 16:16:43 +03:00
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view indexVal, val;
|
2020-07-13 01:08:58 +03:00
|
|
|
uint64_t index = std::numeric_limits<uint64_t>::max() / 2;
|
|
|
|
auto cursor = lmdb::cursor::open(txn, orderDb);
|
2020-07-04 03:09:12 +03:00
|
|
|
if (cursor.get(indexVal, val, MDB_LAST)) {
|
2021-03-03 01:15:12 +03:00
|
|
|
index = lmdb::from_sv<uint64_t>(indexVal);
|
2020-07-04 03:09:12 +03:00
|
|
|
}
|
|
|
|
|
2020-07-13 01:08:58 +03:00
|
|
|
uint64_t msgIndex = std::numeric_limits<uint64_t>::max() / 2;
|
|
|
|
auto msgCursor = lmdb::cursor::open(txn, order2msgDb);
|
2020-07-08 03:02:14 +03:00
|
|
|
if (msgCursor.get(indexVal, val, MDB_LAST)) {
|
2021-03-03 01:15:12 +03:00
|
|
|
msgIndex = lmdb::from_sv<uint64_t>(indexVal);
|
2020-07-08 03:02:14 +03:00
|
|
|
}
|
|
|
|
|
2020-07-04 03:09:12 +03:00
|
|
|
bool first = true;
|
2018-06-28 16:16:43 +03:00
|
|
|
for (const auto &e : res.events) {
|
2020-07-18 18:43:49 +03:00
|
|
|
auto event = mtx::accessors::serialize_event(e);
|
|
|
|
auto txn_id = mtx::accessors::transaction_id(e);
|
|
|
|
|
2020-07-25 15:08:13 +03:00
|
|
|
std::string event_id_val = event.value("event_id", "");
|
|
|
|
if (event_id_val.empty()) {
|
|
|
|
nhlog::db()->error("Event without id!");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event_id = event_id_val;
|
2020-07-25 15:08:13 +03:00
|
|
|
|
2020-08-10 00:36:47 +03:00
|
|
|
json orderEntry = json::object();
|
|
|
|
orderEntry["event_id"] = event_id_val;
|
|
|
|
if (first && !res.prev_batch.empty())
|
|
|
|
orderEntry["prev_batch"] = res.prev_batch;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view txn_order;
|
|
|
|
if (!txn_id.empty() && evToOrderDb.get(txn, txn_id, txn_order)) {
|
|
|
|
eventsDb.put(txn, event_id, event.dump());
|
|
|
|
eventsDb.del(txn, txn_id);
|
|
|
|
|
|
|
|
std::string_view msg_txn_order;
|
|
|
|
if (msg2orderDb.get(txn, txn_id, msg_txn_order)) {
|
|
|
|
order2msgDb.put(txn, msg_txn_order, event_id);
|
|
|
|
msg2orderDb.put(txn, event_id, msg_txn_order);
|
|
|
|
msg2orderDb.del(txn, txn_id);
|
2020-07-18 18:43:49 +03:00
|
|
|
}
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
orderDb.put(txn, txn_order, orderEntry.dump());
|
|
|
|
evToOrderDb.put(txn, event_id, txn_order);
|
|
|
|
evToOrderDb.del(txn, txn_id);
|
2020-07-18 18:43:49 +03:00
|
|
|
|
2021-01-27 00:36:35 +03:00
|
|
|
auto relations = mtx::accessors::relations(e);
|
|
|
|
if (!relations.relations.empty()) {
|
|
|
|
for (const auto &r : relations.relations) {
|
|
|
|
if (!r.event_id.empty()) {
|
2021-03-03 01:15:12 +03:00
|
|
|
relationsDb.del(txn, r.event_id, txn_id);
|
|
|
|
relationsDb.put(txn, r.event_id, event_id);
|
2021-01-27 00:36:35 +03:00
|
|
|
}
|
2020-07-18 18:43:49 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
auto pendingCursor = lmdb::cursor::open(txn, pending);
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view tsIgnored, pendingTxn;
|
2020-07-18 18:43:49 +03:00
|
|
|
while (pendingCursor.get(tsIgnored, pendingTxn, MDB_NEXT)) {
|
|
|
|
if (std::string_view(pendingTxn.data(), pendingTxn.size()) ==
|
|
|
|
txn_id)
|
|
|
|
lmdb::cursor_del(pendingCursor);
|
|
|
|
}
|
|
|
|
} else if (auto redaction =
|
|
|
|
std::get_if<mtx::events::RedactionEvent<mtx::events::msg::Redaction>>(
|
|
|
|
&e)) {
|
2020-07-10 00:15:22 +03:00
|
|
|
if (redaction->redacts.empty())
|
|
|
|
continue;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view oldEvent;
|
|
|
|
bool success = eventsDb.get(txn, redaction->redacts, oldEvent);
|
2020-07-25 15:08:13 +03:00
|
|
|
if (!success)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
mtx::events::collections::TimelineEvent te;
|
|
|
|
try {
|
|
|
|
mtx::events::collections::from_json(
|
|
|
|
json::parse(std::string_view(oldEvent.data(), oldEvent.size())),
|
|
|
|
te);
|
|
|
|
// overwrite the content and add redation data
|
|
|
|
std::visit(
|
|
|
|
[redaction](auto &ev) {
|
|
|
|
ev.unsigned_data.redacted_because = *redaction;
|
|
|
|
ev.unsigned_data.redacted_by = redaction->event_id;
|
|
|
|
},
|
|
|
|
te.data);
|
|
|
|
event = mtx::accessors::serialize_event(te.data);
|
|
|
|
event["content"].clear();
|
|
|
|
|
|
|
|
} catch (std::exception &e) {
|
|
|
|
nhlog::db()->error("Failed to parse message from cache {}",
|
|
|
|
e.what());
|
2020-07-18 18:43:49 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
eventsDb.put(txn, redaction->redacts, event.dump());
|
|
|
|
eventsDb.put(txn, redaction->event_id, json(*redaction).dump());
|
2020-07-25 15:08:13 +03:00
|
|
|
} else {
|
2021-03-03 01:15:12 +03:00
|
|
|
eventsDb.put(txn, event_id, event.dump());
|
2020-07-04 03:09:12 +03:00
|
|
|
|
|
|
|
++index;
|
|
|
|
|
2020-07-08 03:02:14 +03:00
|
|
|
first = false;
|
2020-07-05 06:29:07 +03:00
|
|
|
|
|
|
|
nhlog::db()->debug("saving '{}'", orderEntry.dump());
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
cursor.put(lmdb::to_sv(index), orderEntry.dump(), MDB_APPEND);
|
|
|
|
evToOrderDb.put(txn, event_id, lmdb::to_sv(index));
|
2020-07-08 03:02:14 +03:00
|
|
|
|
|
|
|
// TODO(Nico): Allow blacklisting more event types in UI
|
2020-08-27 22:49:05 +03:00
|
|
|
if (!isHiddenEvent(txn, e, room_id)) {
|
2020-07-08 03:02:14 +03:00
|
|
|
++msgIndex;
|
2021-03-03 01:15:12 +03:00
|
|
|
msgCursor.put(lmdb::to_sv(msgIndex), event_id, MDB_APPEND);
|
|
|
|
|
|
|
|
msg2orderDb.put(txn, event_id, lmdb::to_sv(msgIndex));
|
2020-07-08 03:02:14 +03:00
|
|
|
}
|
|
|
|
|
2021-01-27 00:36:35 +03:00
|
|
|
auto relations = mtx::accessors::relations(e);
|
|
|
|
if (!relations.relations.empty()) {
|
|
|
|
for (const auto &r : relations.relations) {
|
|
|
|
if (!r.event_id.empty()) {
|
2021-03-03 01:15:12 +03:00
|
|
|
relationsDb.put(txn, r.event_id, event_id);
|
2021-01-27 00:36:35 +03:00
|
|
|
}
|
|
|
|
}
|
2020-07-08 03:02:14 +03:00
|
|
|
}
|
2020-07-01 21:15:39 +03:00
|
|
|
}
|
2018-06-28 16:16:43 +03:00
|
|
|
}
|
|
|
|
}
|
2019-08-10 06:34:44 +03:00
|
|
|
|
2020-07-13 01:08:58 +03:00
|
|
|
uint64_t
|
|
|
|
Cache::saveOldMessages(const std::string &room_id, const mtx::responses::Messages &res)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
auto eventsDb = getEventsDb(txn, room_id);
|
|
|
|
auto relationsDb = getRelationsDb(txn, room_id);
|
|
|
|
|
|
|
|
auto orderDb = getEventOrderDb(txn, room_id);
|
2020-08-10 00:36:47 +03:00
|
|
|
auto evToOrderDb = getEventToOrderDb(txn, room_id);
|
2020-07-13 01:08:58 +03:00
|
|
|
auto msg2orderDb = getMessageToOrderDb(txn, room_id);
|
|
|
|
auto order2msgDb = getOrderToMessageDb(txn, room_id);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view indexVal, val;
|
2020-07-13 01:08:58 +03:00
|
|
|
uint64_t index = std::numeric_limits<uint64_t>::max() / 2;
|
2020-07-18 23:59:03 +03:00
|
|
|
{
|
|
|
|
auto cursor = lmdb::cursor::open(txn, orderDb);
|
|
|
|
if (cursor.get(indexVal, val, MDB_FIRST)) {
|
2021-03-03 01:15:12 +03:00
|
|
|
index = lmdb::from_sv<uint64_t>(indexVal);
|
2020-07-18 23:59:03 +03:00
|
|
|
}
|
2020-07-13 01:08:58 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t msgIndex = std::numeric_limits<uint64_t>::max() / 2;
|
2020-07-18 23:59:03 +03:00
|
|
|
{
|
|
|
|
auto msgCursor = lmdb::cursor::open(txn, order2msgDb);
|
|
|
|
if (msgCursor.get(indexVal, val, MDB_FIRST)) {
|
2021-03-03 01:15:12 +03:00
|
|
|
msgIndex = lmdb::from_sv<uint64_t>(indexVal);
|
2020-07-18 23:59:03 +03:00
|
|
|
}
|
2020-07-13 01:08:58 +03:00
|
|
|
}
|
|
|
|
|
2020-10-23 20:42:12 +03:00
|
|
|
if (res.chunk.empty()) {
|
2021-03-03 01:15:12 +03:00
|
|
|
if (orderDb.get(txn, lmdb::to_sv(index), val)) {
|
|
|
|
auto orderEntry = json::parse(val);
|
2020-10-23 20:42:12 +03:00
|
|
|
orderEntry["prev_batch"] = res.end;
|
2021-03-03 01:15:12 +03:00
|
|
|
orderDb.put(txn, lmdb::to_sv(index), orderEntry.dump());
|
2020-10-23 20:42:12 +03:00
|
|
|
txn.commit();
|
|
|
|
}
|
2020-07-13 01:08:58 +03:00
|
|
|
return index;
|
2020-10-23 20:42:12 +03:00
|
|
|
}
|
2020-07-13 01:08:58 +03:00
|
|
|
|
|
|
|
std::string event_id_val;
|
|
|
|
for (const auto &e : res.chunk) {
|
2020-07-19 13:22:54 +03:00
|
|
|
if (std::holds_alternative<
|
|
|
|
mtx::events::RedactionEvent<mtx::events::msg::Redaction>>(e))
|
|
|
|
continue;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
auto event = mtx::accessors::serialize_event(e);
|
|
|
|
event_id_val = event["event_id"].get<std::string>();
|
|
|
|
std::string_view event_id = event_id_val;
|
|
|
|
eventsDb.put(txn, event_id, event.dump());
|
2020-07-13 01:08:58 +03:00
|
|
|
|
|
|
|
--index;
|
|
|
|
|
|
|
|
json orderEntry = json::object();
|
|
|
|
orderEntry["event_id"] = event_id_val;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
orderDb.put(txn, lmdb::to_sv(index), orderEntry.dump());
|
|
|
|
evToOrderDb.put(txn, event_id, lmdb::to_sv(index));
|
2020-07-13 01:08:58 +03:00
|
|
|
|
|
|
|
// TODO(Nico): Allow blacklisting more event types in UI
|
2020-08-27 22:49:05 +03:00
|
|
|
if (!isHiddenEvent(txn, e, room_id)) {
|
2020-07-13 01:08:58 +03:00
|
|
|
--msgIndex;
|
2021-03-03 01:15:12 +03:00
|
|
|
order2msgDb.put(txn, lmdb::to_sv(msgIndex), event_id);
|
2020-07-13 01:08:58 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
msg2orderDb.put(txn, event_id, lmdb::to_sv(msgIndex));
|
2020-07-13 01:08:58 +03:00
|
|
|
}
|
|
|
|
|
2021-01-27 00:36:35 +03:00
|
|
|
auto relations = mtx::accessors::relations(e);
|
|
|
|
if (!relations.relations.empty()) {
|
|
|
|
for (const auto &r : relations.relations) {
|
|
|
|
if (!r.event_id.empty()) {
|
2021-03-03 01:15:12 +03:00
|
|
|
relationsDb.put(txn, r.event_id, event_id);
|
2021-01-27 00:36:35 +03:00
|
|
|
}
|
|
|
|
}
|
2020-07-13 01:08:58 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
json orderEntry = json::object();
|
|
|
|
orderEntry["event_id"] = event_id_val;
|
|
|
|
orderEntry["prev_batch"] = res.end;
|
2021-03-03 01:15:12 +03:00
|
|
|
orderDb.put(txn, lmdb::to_sv(index), orderEntry.dump());
|
2020-07-13 01:08:58 +03:00
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return msgIndex;
|
|
|
|
}
|
|
|
|
|
2020-08-10 00:36:47 +03:00
|
|
|
void
|
|
|
|
Cache::clearTimeline(const std::string &room_id)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
auto eventsDb = getEventsDb(txn, room_id);
|
|
|
|
auto relationsDb = getRelationsDb(txn, room_id);
|
|
|
|
|
|
|
|
auto orderDb = getEventOrderDb(txn, room_id);
|
|
|
|
auto evToOrderDb = getEventToOrderDb(txn, room_id);
|
|
|
|
auto msg2orderDb = getMessageToOrderDb(txn, room_id);
|
|
|
|
auto order2msgDb = getOrderToMessageDb(txn, room_id);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view indexVal, val;
|
2020-08-10 00:36:47 +03:00
|
|
|
auto cursor = lmdb::cursor::open(txn, orderDb);
|
|
|
|
|
|
|
|
bool start = true;
|
|
|
|
bool passed_pagination_token = false;
|
|
|
|
while (cursor.get(indexVal, val, start ? MDB_LAST : MDB_PREV)) {
|
|
|
|
start = false;
|
|
|
|
json obj;
|
|
|
|
|
|
|
|
try {
|
|
|
|
obj = json::parse(std::string_view(val.data(), val.size()));
|
|
|
|
} catch (std::exception &) {
|
|
|
|
// workaround bug in the initial db format, where we sometimes didn't store
|
|
|
|
// json...
|
|
|
|
obj = {{"event_id", std::string(val.data(), val.size())}};
|
|
|
|
}
|
|
|
|
|
|
|
|
if (passed_pagination_token) {
|
|
|
|
if (obj.count("event_id") != 0) {
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string event_id = obj["event_id"].get<std::string>();
|
2021-04-08 13:56:31 +03:00
|
|
|
|
|
|
|
if (!event_id.empty()) {
|
|
|
|
evToOrderDb.del(txn, event_id);
|
|
|
|
eventsDb.del(txn, event_id);
|
|
|
|
relationsDb.del(txn, event_id);
|
|
|
|
|
|
|
|
std::string_view order{};
|
|
|
|
bool exists = msg2orderDb.get(txn, event_id, order);
|
|
|
|
if (exists) {
|
|
|
|
order2msgDb.del(txn, order);
|
|
|
|
msg2orderDb.del(txn, event_id);
|
|
|
|
}
|
2020-08-10 00:36:47 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
lmdb::cursor_del(cursor);
|
|
|
|
} else {
|
|
|
|
if (obj.count("prev_batch") != 0)
|
|
|
|
passed_pagination_token = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
auto msgCursor = lmdb::cursor::open(txn, order2msgDb);
|
|
|
|
start = true;
|
|
|
|
while (msgCursor.get(indexVal, val, start ? MDB_LAST : MDB_PREV)) {
|
|
|
|
start = false;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view eventId;
|
2020-08-10 00:36:47 +03:00
|
|
|
bool innerStart = true;
|
|
|
|
bool found = false;
|
|
|
|
while (cursor.get(indexVal, eventId, innerStart ? MDB_LAST : MDB_PREV)) {
|
|
|
|
innerStart = false;
|
|
|
|
|
|
|
|
json obj;
|
|
|
|
try {
|
|
|
|
obj = json::parse(std::string_view(eventId.data(), eventId.size()));
|
|
|
|
} catch (std::exception &) {
|
|
|
|
obj = {{"event_id", std::string(eventId.data(), eventId.size())}};
|
|
|
|
}
|
|
|
|
|
|
|
|
if (obj["event_id"] == std::string(val.data(), val.size())) {
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!found)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
do {
|
|
|
|
lmdb::cursor_del(msgCursor);
|
|
|
|
} while (msgCursor.get(indexVal, val, MDB_PREV));
|
|
|
|
|
|
|
|
cursor.close();
|
|
|
|
msgCursor.close();
|
|
|
|
txn.commit();
|
|
|
|
}
|
|
|
|
|
2019-08-06 06:00:07 +03:00
|
|
|
mtx::responses::Notifications
|
2019-08-10 06:34:44 +03:00
|
|
|
Cache::getTimelineMentionsForRoom(lmdb::txn &txn, const std::string &room_id)
|
2019-08-06 06:00:07 +03:00
|
|
|
{
|
|
|
|
auto db = getMentionsDb(txn, room_id);
|
|
|
|
|
2019-08-10 06:34:44 +03:00
|
|
|
if (db.size(txn) == 0) {
|
|
|
|
return mtx::responses::Notifications{};
|
|
|
|
}
|
|
|
|
|
2019-08-06 06:00:07 +03:00
|
|
|
mtx::responses::Notifications notif;
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event_id, msg;
|
2019-08-06 06:00:07 +03:00
|
|
|
|
|
|
|
auto cursor = lmdb::cursor::open(txn, db);
|
|
|
|
|
|
|
|
while (cursor.get(event_id, msg, MDB_NEXT)) {
|
|
|
|
auto obj = json::parse(msg);
|
|
|
|
|
2019-08-13 05:09:40 +03:00
|
|
|
if (obj.count("event") == 0)
|
2019-08-06 06:00:07 +03:00
|
|
|
continue;
|
|
|
|
|
|
|
|
mtx::responses::Notification notification;
|
2019-08-13 05:09:40 +03:00
|
|
|
mtx::responses::from_json(obj, notification);
|
2019-08-06 06:00:07 +03:00
|
|
|
|
|
|
|
notif.notifications.push_back(notification);
|
|
|
|
}
|
|
|
|
cursor.close();
|
|
|
|
|
|
|
|
std::reverse(notif.notifications.begin(), notif.notifications.end());
|
|
|
|
|
|
|
|
return notif;
|
|
|
|
}
|
2019-08-10 06:34:44 +03:00
|
|
|
|
|
|
|
//! Add all notifications containing a user mention to the db.
|
|
|
|
void
|
|
|
|
Cache::saveTimelineMentions(const mtx::responses::Notifications &res)
|
|
|
|
{
|
|
|
|
QMap<std::string, QList<mtx::responses::Notification>> notifsByRoom;
|
|
|
|
|
|
|
|
// Sort into room-specific 'buckets'
|
|
|
|
for (const auto ¬if : res.notifications) {
|
2019-08-13 05:09:40 +03:00
|
|
|
json val = notif;
|
2019-08-10 06:34:44 +03:00
|
|
|
notifsByRoom[notif.room_id].push_back(notif);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
// Insert the entire set of mentions for each room at a time.
|
2019-08-13 05:09:40 +03:00
|
|
|
QMap<std::string, QList<mtx::responses::Notification>>::const_iterator it =
|
|
|
|
notifsByRoom.constBegin();
|
|
|
|
auto end = notifsByRoom.constEnd();
|
|
|
|
while (it != end) {
|
|
|
|
nhlog::db()->debug("Storing notifications for " + it.key());
|
|
|
|
saveTimelineMentions(txn, it.key(), std::move(it.value()));
|
|
|
|
++it;
|
2019-08-10 06:34:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
}
|
|
|
|
|
2019-08-06 06:00:07 +03:00
|
|
|
void
|
|
|
|
Cache::saveTimelineMentions(lmdb::txn &txn,
|
|
|
|
const std::string &room_id,
|
2019-08-10 06:34:44 +03:00
|
|
|
const QList<mtx::responses::Notification> &res)
|
2019-08-06 06:00:07 +03:00
|
|
|
{
|
|
|
|
auto db = getMentionsDb(txn, room_id);
|
|
|
|
|
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
2019-08-10 06:34:44 +03:00
|
|
|
for (const auto ¬if : res) {
|
2020-05-30 17:37:51 +03:00
|
|
|
const auto event_id = mtx::accessors::event_id(notif.event);
|
2019-08-06 06:00:07 +03:00
|
|
|
|
|
|
|
// double check that we have the correct room_id...
|
2019-08-10 06:34:44 +03:00
|
|
|
if (room_id.compare(notif.room_id) != 0) {
|
|
|
|
return;
|
|
|
|
}
|
2019-08-06 06:00:07 +03:00
|
|
|
|
2019-08-13 05:09:40 +03:00
|
|
|
json obj = notif;
|
2019-08-06 06:00:07 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
db.put(txn, event_id, obj.dump());
|
2019-08-06 06:00:07 +03:00
|
|
|
}
|
|
|
|
}
|
2018-06-28 16:16:43 +03:00
|
|
|
|
2018-05-05 16:38:41 +03:00
|
|
|
void
|
|
|
|
Cache::markSentNotification(const std::string &event_id)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
2021-03-03 01:15:12 +03:00
|
|
|
notificationsDb_.put(txn, event_id, "");
|
2018-05-05 16:38:41 +03:00
|
|
|
txn.commit();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Cache::removeReadNotification(const std::string &event_id)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
notificationsDb_.del(txn, event_id);
|
2018-05-05 16:38:41 +03:00
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
Cache::isNotificationSent(const std::string &event_id)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view value;
|
|
|
|
bool res = notificationsDb_.get(txn, event_id, value);
|
2018-05-05 16:38:41 +03:00
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2018-08-25 21:08:43 +03:00
|
|
|
std::vector<std::string>
|
|
|
|
Cache::getRoomIds(lmdb::txn &txn)
|
|
|
|
{
|
|
|
|
auto db = lmdb::dbi::open(txn, ROOMS_DB, MDB_CREATE);
|
|
|
|
auto cursor = lmdb::cursor::open(txn, db);
|
|
|
|
|
|
|
|
std::vector<std::string> rooms;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view room_id, _unused;
|
2018-08-25 21:08:43 +03:00
|
|
|
while (cursor.get(room_id, _unused, MDB_NEXT))
|
|
|
|
rooms.emplace_back(room_id);
|
|
|
|
|
|
|
|
cursor.close();
|
|
|
|
|
|
|
|
return rooms;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Cache::deleteOldMessages()
|
|
|
|
{
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view indexVal, val;
|
2020-07-05 06:29:07 +03:00
|
|
|
|
2018-08-25 21:08:43 +03:00
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
auto room_ids = getRoomIds(txn);
|
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
for (const auto &room_id : room_ids) {
|
2020-08-10 00:36:47 +03:00
|
|
|
auto orderDb = getEventOrderDb(txn, room_id);
|
|
|
|
auto evToOrderDb = getEventToOrderDb(txn, room_id);
|
|
|
|
auto o2m = getOrderToMessageDb(txn, room_id);
|
|
|
|
auto m2o = getMessageToOrderDb(txn, room_id);
|
|
|
|
auto eventsDb = getEventsDb(txn, room_id);
|
|
|
|
auto relationsDb = getRelationsDb(txn, room_id);
|
|
|
|
auto cursor = lmdb::cursor::open(txn, orderDb);
|
2018-08-25 21:08:43 +03:00
|
|
|
|
2020-07-13 01:08:58 +03:00
|
|
|
uint64_t first, last;
|
2020-07-05 06:29:07 +03:00
|
|
|
if (cursor.get(indexVal, val, MDB_LAST)) {
|
2021-03-03 01:15:12 +03:00
|
|
|
last = lmdb::from_sv<uint64_t>(indexVal);
|
2020-07-05 06:29:07 +03:00
|
|
|
} else {
|
2018-08-25 21:08:43 +03:00
|
|
|
continue;
|
2020-07-05 06:29:07 +03:00
|
|
|
}
|
|
|
|
if (cursor.get(indexVal, val, MDB_FIRST)) {
|
2021-03-03 01:15:12 +03:00
|
|
|
first = lmdb::from_sv<uint64_t>(indexVal);
|
2020-07-05 06:29:07 +03:00
|
|
|
} else {
|
|
|
|
continue;
|
|
|
|
}
|
2018-08-25 21:08:43 +03:00
|
|
|
|
2020-07-05 06:29:07 +03:00
|
|
|
size_t message_count = static_cast<size_t>(last - first);
|
|
|
|
if (message_count < MAX_RESTORED_MESSAGES)
|
|
|
|
continue;
|
2018-08-25 21:08:43 +03:00
|
|
|
|
2020-07-10 00:15:22 +03:00
|
|
|
bool start = true;
|
|
|
|
while (cursor.get(indexVal, val, start ? MDB_FIRST : MDB_NEXT) &&
|
2020-08-10 00:36:47 +03:00
|
|
|
message_count-- > MAX_RESTORED_MESSAGES) {
|
2020-07-10 00:15:22 +03:00
|
|
|
start = false;
|
2020-07-06 04:43:14 +03:00
|
|
|
auto obj = json::parse(std::string_view(val.data(), val.size()));
|
2018-08-25 21:08:43 +03:00
|
|
|
|
2020-07-08 03:02:14 +03:00
|
|
|
if (obj.count("event_id") != 0) {
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string event_id = obj["event_id"].get<std::string>();
|
|
|
|
evToOrderDb.del(txn, event_id);
|
|
|
|
eventsDb.del(txn, event_id);
|
2020-07-08 03:02:14 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
relationsDb.del(txn, event_id);
|
2020-08-10 00:36:47 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view order{};
|
|
|
|
bool exists = m2o.get(txn, event_id, order);
|
2020-07-08 03:02:14 +03:00
|
|
|
if (exists) {
|
2021-03-03 01:15:12 +03:00
|
|
|
o2m.del(txn, order);
|
|
|
|
m2o.del(txn, event_id);
|
2020-07-08 03:02:14 +03:00
|
|
|
}
|
|
|
|
}
|
2021-03-03 01:15:12 +03:00
|
|
|
cursor.del();
|
2018-08-25 21:08:43 +03:00
|
|
|
}
|
|
|
|
cursor.close();
|
|
|
|
}
|
|
|
|
txn.commit();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Cache::deleteOldData() noexcept
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
deleteOldMessages();
|
|
|
|
} catch (const lmdb::error &e) {
|
|
|
|
nhlog::db()->error("failed to delete old messages: {}", e.what());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-27 22:49:05 +03:00
|
|
|
std::optional<mtx::events::collections::RoomAccountDataEvents>
|
|
|
|
Cache::getAccountData(lmdb::txn &txn, mtx::events::EventType type, const std::string &room_id)
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
auto db = getAccountDataDb(txn, room_id);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view data;
|
|
|
|
if (db.get(txn, to_string(type), data)) {
|
2020-08-27 22:49:05 +03:00
|
|
|
mtx::responses::utils::RoomAccountDataEvents events;
|
2021-02-10 04:37:47 +03:00
|
|
|
json j = json::array({
|
2021-03-03 01:15:12 +03:00
|
|
|
json::parse(data),
|
2021-02-10 04:37:47 +03:00
|
|
|
});
|
|
|
|
mtx::responses::utils::parse_room_account_data_events(j, events);
|
|
|
|
if (events.size() == 1)
|
|
|
|
return events.front();
|
2020-08-27 22:49:05 +03:00
|
|
|
}
|
|
|
|
} catch (...) {
|
|
|
|
}
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
2018-05-16 23:30:50 +03:00
|
|
|
bool
|
|
|
|
Cache::hasEnoughPowerLevel(const std::vector<mtx::events::EventType> &eventTypes,
|
|
|
|
const std::string &room_id,
|
|
|
|
const std::string &user_id)
|
|
|
|
{
|
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
auto db = getStatesDb(txn, room_id);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
int64_t min_event_level = std::numeric_limits<int64_t>::max();
|
|
|
|
int64_t user_level = std::numeric_limits<int64_t>::min();
|
2018-05-16 23:30:50 +03:00
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view event;
|
|
|
|
bool res = db.get(txn, to_string(EventType::RoomPowerLevels), event);
|
2018-05-16 23:30:50 +03:00
|
|
|
|
|
|
|
if (res) {
|
|
|
|
try {
|
|
|
|
StateEvent<PowerLevels> msg =
|
2020-07-06 04:43:14 +03:00
|
|
|
json::parse(std::string_view(event.data(), event.size()));
|
2018-05-16 23:30:50 +03:00
|
|
|
|
|
|
|
user_level = msg.content.user_level(user_id);
|
|
|
|
|
|
|
|
for (const auto &ty : eventTypes)
|
|
|
|
min_event_level =
|
2021-03-03 01:15:12 +03:00
|
|
|
std::min(min_event_level, msg.content.state_level(to_string(ty)));
|
2018-05-16 23:30:50 +03:00
|
|
|
} catch (const json::exception &e) {
|
2018-06-14 02:28:35 +03:00
|
|
|
nhlog::db()->warn("failed to parse m.room.power_levels event: {}",
|
|
|
|
e.what());
|
2018-05-16 23:30:50 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return user_level >= min_event_level;
|
|
|
|
}
|
|
|
|
|
2018-06-13 12:28:00 +03:00
|
|
|
std::vector<std::string>
|
|
|
|
Cache::roomMembers(const std::string &room_id)
|
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
|
|
|
|
std::vector<std::string> members;
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view user_id, unused;
|
2018-06-13 12:28:00 +03:00
|
|
|
|
|
|
|
auto db = getMembersDb(txn, room_id);
|
|
|
|
|
|
|
|
auto cursor = lmdb::cursor::open(txn, db);
|
|
|
|
while (cursor.get(user_id, unused, MDB_NEXT))
|
2021-03-03 01:15:12 +03:00
|
|
|
members.emplace_back(user_id);
|
2018-06-13 12:28:00 +03:00
|
|
|
cursor.close();
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return members;
|
|
|
|
}
|
|
|
|
|
2020-11-30 02:26:27 +03:00
|
|
|
std::map<std::string, std::optional<UserKeyCache>>
|
|
|
|
Cache::getMembersWithKeys(const std::string &room_id)
|
|
|
|
{
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view keys;
|
2020-11-30 02:26:27 +03:00
|
|
|
|
|
|
|
try {
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
std::map<std::string, std::optional<UserKeyCache>> members;
|
|
|
|
|
|
|
|
auto db = getMembersDb(txn, room_id);
|
|
|
|
auto keysDb = getUserKeysDb(txn);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view user_id, unused;
|
2020-11-30 02:26:27 +03:00
|
|
|
auto cursor = lmdb::cursor::open(txn, db);
|
|
|
|
while (cursor.get(user_id, unused, MDB_NEXT)) {
|
2021-03-03 01:15:12 +03:00
|
|
|
auto res = keysDb.get(txn, user_id, keys);
|
2020-11-30 02:26:27 +03:00
|
|
|
|
|
|
|
if (res) {
|
2021-03-03 01:15:12 +03:00
|
|
|
members[std::string(user_id)] =
|
|
|
|
json::parse(keys).get<UserKeyCache>();
|
2020-11-30 02:26:27 +03:00
|
|
|
} else {
|
2021-03-03 01:15:12 +03:00
|
|
|
members[std::string(user_id)] = {};
|
2020-11-30 02:26:27 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
cursor.close();
|
|
|
|
|
|
|
|
return members;
|
|
|
|
} catch (std::exception &) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-21 16:34:50 +03:00
|
|
|
QString
|
|
|
|
Cache::displayName(const QString &room_id, const QString &user_id)
|
|
|
|
{
|
2020-10-22 21:48:28 +03:00
|
|
|
if (auto info = getMember(room_id.toStdString(), user_id.toStdString());
|
|
|
|
info && !info->name.empty())
|
|
|
|
return QString::fromStdString(info->name);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
return user_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string
|
|
|
|
Cache::displayName(const std::string &room_id, const std::string &user_id)
|
|
|
|
{
|
2020-10-22 21:48:28 +03:00
|
|
|
if (auto info = getMember(room_id, user_id); info && !info->name.empty())
|
|
|
|
return info->name;
|
2018-04-21 16:34:50 +03:00
|
|
|
|
|
|
|
return user_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString
|
|
|
|
Cache::avatarUrl(const QString &room_id, const QString &user_id)
|
|
|
|
{
|
2020-10-22 21:48:28 +03:00
|
|
|
if (auto info = getMember(room_id.toStdString(), user_id.toStdString());
|
|
|
|
info && !info->avatar_url.empty())
|
|
|
|
return QString::fromStdString(info->avatar_url);
|
2018-04-21 16:34:50 +03:00
|
|
|
|
2020-10-22 21:48:28 +03:00
|
|
|
return "";
|
2018-04-21 16:34:50 +03:00
|
|
|
}
|
2019-01-26 05:53:43 +03:00
|
|
|
|
2020-06-08 02:45:24 +03:00
|
|
|
mtx::presence::PresenceState
|
|
|
|
Cache::presenceState(const std::string &user_id)
|
|
|
|
{
|
2020-07-10 00:15:22 +03:00
|
|
|
if (user_id.empty())
|
|
|
|
return {};
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view presenceVal;
|
2020-06-08 02:45:24 +03:00
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
auto db = getPresenceDb(txn);
|
2021-03-03 01:15:12 +03:00
|
|
|
auto res = db.get(txn, user_id, presenceVal);
|
2020-06-08 02:45:24 +03:00
|
|
|
|
|
|
|
mtx::presence::PresenceState state = mtx::presence::offline;
|
|
|
|
|
|
|
|
if (res) {
|
|
|
|
mtx::events::presence::Presence presence =
|
2020-07-06 04:43:14 +03:00
|
|
|
json::parse(std::string_view(presenceVal.data(), presenceVal.size()));
|
2020-06-08 02:45:24 +03:00
|
|
|
state = presence.presence;
|
|
|
|
}
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string
|
|
|
|
Cache::statusMessage(const std::string &user_id)
|
|
|
|
{
|
2020-07-10 00:15:22 +03:00
|
|
|
if (user_id.empty())
|
|
|
|
return {};
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view presenceVal;
|
2020-06-08 02:45:24 +03:00
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
auto db = getPresenceDb(txn);
|
2021-03-03 01:15:12 +03:00
|
|
|
auto res = db.get(txn, user_id, presenceVal);
|
2020-06-08 02:45:24 +03:00
|
|
|
|
|
|
|
std::string status_msg;
|
|
|
|
|
|
|
|
if (res) {
|
2021-03-03 01:15:12 +03:00
|
|
|
mtx::events::presence::Presence presence = json::parse(presenceVal);
|
|
|
|
status_msg = presence.status_msg;
|
2020-06-08 02:45:24 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
return status_msg;
|
|
|
|
}
|
|
|
|
|
2020-06-28 18:31:34 +03:00
|
|
|
void
|
2020-10-02 02:14:42 +03:00
|
|
|
to_json(json &j, const UserKeyCache &info)
|
2020-06-28 18:31:34 +03:00
|
|
|
{
|
2021-05-07 18:01:03 +03:00
|
|
|
j["device_keys"] = info.device_keys;
|
|
|
|
j["seen_device_keys"] = info.seen_device_keys;
|
|
|
|
j["master_keys"] = info.master_keys;
|
|
|
|
j["master_key_changed"] = info.master_key_changed;
|
|
|
|
j["user_signing_keys"] = info.user_signing_keys;
|
|
|
|
j["self_signing_keys"] = info.self_signing_keys;
|
|
|
|
j["updated_at"] = info.updated_at;
|
|
|
|
j["last_changed"] = info.last_changed;
|
2020-06-28 18:31:34 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2020-10-02 02:14:42 +03:00
|
|
|
from_json(const json &j, UserKeyCache &info)
|
2020-06-28 18:31:34 +03:00
|
|
|
{
|
2020-10-02 02:14:42 +03:00
|
|
|
info.device_keys = j.value("device_keys", std::map<std::string, mtx::crypto::DeviceKeys>{});
|
2021-05-07 18:01:03 +03:00
|
|
|
info.seen_device_keys = j.value("seen_device_keys", std::set<std::string>{});
|
|
|
|
info.master_keys = j.value("master_keys", mtx::crypto::CrossSigningKeys{});
|
|
|
|
info.master_key_changed = j.value("master_key_changed", false);
|
|
|
|
info.user_signing_keys = j.value("user_signing_keys", mtx::crypto::CrossSigningKeys{});
|
|
|
|
info.self_signing_keys = j.value("self_signing_keys", mtx::crypto::CrossSigningKeys{});
|
|
|
|
info.updated_at = j.value("updated_at", "");
|
|
|
|
info.last_changed = j.value("last_changed", "");
|
2020-06-28 18:31:34 +03:00
|
|
|
}
|
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
std::optional<UserKeyCache>
|
|
|
|
Cache::userKeys(const std::string &user_id)
|
2020-06-28 18:31:34 +03:00
|
|
|
{
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view keys;
|
2020-06-28 18:31:34 +03:00
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
try {
|
|
|
|
auto txn = lmdb::txn::begin(env_, nullptr, MDB_RDONLY);
|
|
|
|
auto db = getUserKeysDb(txn);
|
2021-03-03 01:15:12 +03:00
|
|
|
auto res = db.get(txn, user_id, keys);
|
2020-07-01 15:17:10 +03:00
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
if (res) {
|
2021-03-03 01:15:12 +03:00
|
|
|
return json::parse(keys).get<UserKeyCache>();
|
2020-10-02 02:14:42 +03:00
|
|
|
} else {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
} catch (std::exception &) {
|
2020-07-01 15:17:10 +03:00
|
|
|
return {};
|
2020-06-28 18:31:34 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
void
|
|
|
|
Cache::updateUserKeys(const std::string &sync_token, const mtx::responses::QueryKeys &keyQuery)
|
2020-06-28 18:31:34 +03:00
|
|
|
{
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
2020-10-02 02:14:42 +03:00
|
|
|
auto db = getUserKeysDb(txn);
|
2020-06-28 18:31:34 +03:00
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
std::map<std::string, UserKeyCache> updates;
|
2020-06-28 18:31:34 +03:00
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
for (const auto &[user, keys] : keyQuery.device_keys)
|
|
|
|
updates[user].device_keys = keys;
|
|
|
|
for (const auto &[user, keys] : keyQuery.master_keys)
|
|
|
|
updates[user].master_keys = keys;
|
|
|
|
for (const auto &[user, keys] : keyQuery.user_signing_keys)
|
|
|
|
updates[user].user_signing_keys = keys;
|
|
|
|
for (const auto &[user, keys] : keyQuery.self_signing_keys)
|
|
|
|
updates[user].self_signing_keys = keys;
|
2020-06-28 18:31:34 +03:00
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
for (auto &[user, update] : updates) {
|
2020-11-30 02:26:27 +03:00
|
|
|
nhlog::db()->debug("Updated user keys: {}", user);
|
|
|
|
|
2021-05-07 18:01:03 +03:00
|
|
|
auto updateToWrite = update;
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view oldKeys;
|
|
|
|
auto res = db.get(txn, user, oldKeys);
|
2020-10-02 02:14:42 +03:00
|
|
|
|
|
|
|
if (res) {
|
2021-05-07 18:01:03 +03:00
|
|
|
updateToWrite = json::parse(oldKeys).get<UserKeyCache>();
|
|
|
|
auto last_changed = updateToWrite.last_changed;
|
2020-10-02 02:14:42 +03:00
|
|
|
// skip if we are tracking this and expect it to be up to date with the last
|
|
|
|
// sync token
|
|
|
|
if (!last_changed.empty() && last_changed != sync_token)
|
|
|
|
continue;
|
2021-05-07 18:01:03 +03:00
|
|
|
|
|
|
|
if (!updateToWrite.master_keys.keys.empty() &&
|
|
|
|
update.master_keys.keys != updateToWrite.master_keys.keys) {
|
|
|
|
updateToWrite.master_key_changed = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
updateToWrite.master_keys = update.master_keys;
|
|
|
|
updateToWrite.self_signing_keys = update.self_signing_keys;
|
|
|
|
updateToWrite.user_signing_keys = update.user_signing_keys;
|
|
|
|
|
|
|
|
// If we have keys for the device already, only update the signatures.
|
|
|
|
for (const auto &[device_id, device_keys] : update.device_keys) {
|
|
|
|
if (updateToWrite.device_keys.count(device_id) &&
|
|
|
|
updateToWrite.device_keys.at(device_id).keys ==
|
|
|
|
device_keys.keys) {
|
|
|
|
updateToWrite.device_keys.at(device_id).signatures =
|
|
|
|
device_keys.signatures;
|
|
|
|
} else {
|
|
|
|
bool keyReused = false;
|
|
|
|
for (const auto &[key_id, key] : device_keys.keys) {
|
|
|
|
(void)key_id;
|
|
|
|
if (updateToWrite.seen_device_keys.count(key)) {
|
|
|
|
keyReused = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!updateToWrite.device_keys.count(device_id) &&
|
|
|
|
!keyReused)
|
|
|
|
updateToWrite.device_keys[device_id] = device_keys;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const auto &[key_id, key] : device_keys.keys) {
|
|
|
|
(void)key_id;
|
|
|
|
updateToWrite.seen_device_keys.insert(key);
|
|
|
|
}
|
|
|
|
}
|
2020-10-02 02:14:42 +03:00
|
|
|
}
|
2021-05-07 18:01:03 +03:00
|
|
|
db.put(txn, user, json(updateToWrite).dump());
|
2020-10-02 02:14:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
txn.commit();
|
2020-10-08 00:03:14 +03:00
|
|
|
|
|
|
|
std::map<std::string, VerificationStatus> tmp;
|
|
|
|
const auto local_user = utils::localUser().toStdString();
|
|
|
|
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> lock(verification_storage.verification_storage_mtx);
|
|
|
|
for (auto &[user_id, update] : updates) {
|
2020-10-08 19:38:55 +03:00
|
|
|
(void)update;
|
2020-10-08 00:03:14 +03:00
|
|
|
if (user_id == local_user) {
|
|
|
|
std::swap(tmp, verification_storage.status);
|
|
|
|
} else {
|
|
|
|
verification_storage.status.erase(user_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-05-22 15:31:38 +03:00
|
|
|
|
2020-10-08 00:03:14 +03:00
|
|
|
for (auto &[user_id, update] : updates) {
|
2020-10-08 19:38:55 +03:00
|
|
|
(void)update;
|
2020-10-08 00:03:14 +03:00
|
|
|
if (user_id == local_user) {
|
2020-10-08 19:38:55 +03:00
|
|
|
for (const auto &[user, status] : tmp) {
|
|
|
|
(void)status;
|
2020-10-08 00:03:14 +03:00
|
|
|
emit verificationStatusChanged(user);
|
2020-10-08 19:38:55 +03:00
|
|
|
}
|
2020-10-08 00:03:14 +03:00
|
|
|
}
|
2021-05-22 15:31:38 +03:00
|
|
|
emit verificationStatusChanged(user_id);
|
2020-10-08 00:03:14 +03:00
|
|
|
}
|
2020-06-28 18:31:34 +03:00
|
|
|
}
|
|
|
|
|
2020-07-06 19:02:21 +03:00
|
|
|
void
|
2020-10-02 02:14:42 +03:00
|
|
|
Cache::deleteUserKeys(lmdb::txn &txn, lmdb::dbi &db, const std::vector<std::string> &user_ids)
|
2020-07-06 19:02:21 +03:00
|
|
|
{
|
2020-10-02 02:14:42 +03:00
|
|
|
for (const auto &user_id : user_ids)
|
2021-03-03 01:15:12 +03:00
|
|
|
db.del(txn, user_id);
|
2020-07-06 19:02:21 +03:00
|
|
|
}
|
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
void
|
|
|
|
Cache::markUserKeysOutOfDate(lmdb::txn &txn,
|
|
|
|
lmdb::dbi &db,
|
|
|
|
const std::vector<std::string> &user_ids,
|
|
|
|
const std::string &sync_token)
|
2020-06-28 18:31:34 +03:00
|
|
|
{
|
2020-10-02 02:14:42 +03:00
|
|
|
mtx::requests::QueryKeys query;
|
|
|
|
query.token = sync_token;
|
2020-06-28 18:31:34 +03:00
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
for (const auto &user : user_ids) {
|
2020-11-30 02:26:27 +03:00
|
|
|
nhlog::db()->debug("Marking user keys out of date: {}", user);
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view oldKeys;
|
|
|
|
auto res = db.get(txn, user, oldKeys);
|
2020-06-28 18:31:34 +03:00
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
if (!res)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
auto cacheEntry =
|
|
|
|
json::parse(std::string_view(oldKeys.data(), oldKeys.size())).get<UserKeyCache>();
|
|
|
|
cacheEntry.last_changed = sync_token;
|
2021-03-03 01:15:12 +03:00
|
|
|
db.put(txn, user, json(cacheEntry).dump());
|
2020-10-02 02:14:42 +03:00
|
|
|
|
|
|
|
query.device_keys[user] = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!query.device_keys.empty())
|
|
|
|
http::client()->query_keys(query,
|
|
|
|
[this, sync_token](const mtx::responses::QueryKeys &keys,
|
|
|
|
mtx::http::RequestErr err) {
|
|
|
|
if (err) {
|
|
|
|
nhlog::net()->warn(
|
|
|
|
"failed to query device keys: {} {}",
|
|
|
|
err->matrix_error.error,
|
|
|
|
static_cast<int>(err->status_code));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
emit userKeysUpdate(sync_token, keys);
|
|
|
|
});
|
2020-06-28 18:31:34 +03:00
|
|
|
}
|
|
|
|
|
2020-10-27 19:45:28 +03:00
|
|
|
void
|
|
|
|
Cache::query_keys(const std::string &user_id,
|
|
|
|
std::function<void(const UserKeyCache &, mtx::http::RequestErr)> cb)
|
|
|
|
{
|
|
|
|
auto cache_ = cache::userKeys(user_id);
|
|
|
|
|
|
|
|
if (cache_.has_value()) {
|
|
|
|
if (!cache_->updated_at.empty() && cache_->updated_at == cache_->last_changed) {
|
|
|
|
cb(cache_.value(), {});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
mtx::requests::QueryKeys req;
|
|
|
|
req.device_keys[user_id] = {};
|
|
|
|
|
|
|
|
std::string last_changed;
|
|
|
|
if (cache_)
|
|
|
|
last_changed = cache_->last_changed;
|
|
|
|
req.token = last_changed;
|
|
|
|
|
2021-05-22 15:31:38 +03:00
|
|
|
// use context object so that we can disconnect again
|
|
|
|
QObject *context{new QObject(this)};
|
|
|
|
QObject::connect(this,
|
|
|
|
&Cache::verificationStatusChanged,
|
|
|
|
context,
|
|
|
|
[cb, user_id, context_ = context](std::string updated_user) mutable {
|
|
|
|
if (user_id == updated_user) {
|
|
|
|
context_->deleteLater();
|
|
|
|
auto keys = cache::userKeys(user_id);
|
|
|
|
cb(keys.value_or(UserKeyCache{}), {});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2021-03-05 03:11:08 +03:00
|
|
|
http::client()->query_keys(
|
|
|
|
req,
|
2021-05-15 00:35:34 +03:00
|
|
|
[cb, user_id, last_changed, this](const mtx::responses::QueryKeys &res,
|
|
|
|
mtx::http::RequestErr err) {
|
2021-03-05 03:11:08 +03:00
|
|
|
if (err) {
|
|
|
|
nhlog::net()->warn("failed to query device keys: {},{}",
|
|
|
|
mtx::errors::to_string(err->matrix_error.errcode),
|
|
|
|
static_cast<int>(err->status_code));
|
|
|
|
cb({}, err);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-05-15 00:35:34 +03:00
|
|
|
emit userKeysUpdate(last_changed, res);
|
2021-03-05 03:11:08 +03:00
|
|
|
});
|
2020-10-27 19:45:28 +03:00
|
|
|
}
|
|
|
|
|
2020-06-28 18:31:34 +03:00
|
|
|
void
|
2020-10-02 02:14:42 +03:00
|
|
|
to_json(json &j, const VerificationCache &info)
|
2020-06-28 18:31:34 +03:00
|
|
|
{
|
2020-10-08 00:03:14 +03:00
|
|
|
j["device_verified"] = info.device_verified;
|
|
|
|
j["device_blocked"] = info.device_blocked;
|
2020-06-28 18:31:34 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2020-10-02 02:14:42 +03:00
|
|
|
from_json(const json &j, VerificationCache &info)
|
2020-06-28 18:31:34 +03:00
|
|
|
{
|
2020-10-08 00:03:14 +03:00
|
|
|
info.device_verified = j.at("device_verified").get<std::vector<std::string>>();
|
|
|
|
info.device_blocked = j.at("device_blocked").get<std::vector<std::string>>();
|
2020-06-28 18:31:34 +03:00
|
|
|
}
|
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
std::optional<VerificationCache>
|
2020-10-08 00:03:14 +03:00
|
|
|
Cache::verificationCache(const std::string &user_id)
|
2020-06-28 18:31:34 +03:00
|
|
|
{
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view verifiedVal;
|
2020-06-28 18:31:34 +03:00
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
2020-10-02 02:14:42 +03:00
|
|
|
auto db = getVerificationDb(txn);
|
2020-06-28 18:31:34 +03:00
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
try {
|
|
|
|
VerificationCache verified_state;
|
2021-03-03 01:15:12 +03:00
|
|
|
auto res = db.get(txn, user_id, verifiedVal);
|
2020-10-02 02:14:42 +03:00
|
|
|
if (res) {
|
2021-03-03 01:15:12 +03:00
|
|
|
verified_state = json::parse(verifiedVal);
|
2020-10-02 02:14:42 +03:00
|
|
|
return verified_state;
|
|
|
|
} else {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
} catch (std::exception &) {
|
2020-07-01 15:17:10 +03:00
|
|
|
return {};
|
2020-06-28 18:31:34 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
void
|
|
|
|
Cache::markDeviceVerified(const std::string &user_id, const std::string &key)
|
2020-06-28 18:31:34 +03:00
|
|
|
{
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view val;
|
2020-10-02 02:14:42 +03:00
|
|
|
|
2020-06-28 18:31:34 +03:00
|
|
|
auto txn = lmdb::txn::begin(env_);
|
2020-10-02 02:14:42 +03:00
|
|
|
auto db = getVerificationDb(txn);
|
2020-06-28 18:31:34 +03:00
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
try {
|
|
|
|
VerificationCache verified_state;
|
2021-03-03 01:15:12 +03:00
|
|
|
auto res = db.get(txn, user_id, val);
|
2020-10-02 02:14:42 +03:00
|
|
|
if (res) {
|
2021-03-03 01:15:12 +03:00
|
|
|
verified_state = json::parse(val);
|
2020-10-02 02:14:42 +03:00
|
|
|
}
|
2020-06-28 18:31:34 +03:00
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
for (const auto &device : verified_state.device_verified)
|
|
|
|
if (device == key)
|
|
|
|
return;
|
2020-06-28 18:31:34 +03:00
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
verified_state.device_verified.push_back(key);
|
2021-03-03 01:15:12 +03:00
|
|
|
db.put(txn, user_id, json(verified_state).dump());
|
2020-10-02 02:14:42 +03:00
|
|
|
txn.commit();
|
|
|
|
} catch (std::exception &) {
|
|
|
|
}
|
2020-10-08 00:03:14 +03:00
|
|
|
|
|
|
|
const auto local_user = utils::localUser().toStdString();
|
|
|
|
std::map<std::string, VerificationStatus> tmp;
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> lock(verification_storage.verification_storage_mtx);
|
|
|
|
if (user_id == local_user) {
|
|
|
|
std::swap(tmp, verification_storage.status);
|
|
|
|
} else {
|
|
|
|
verification_storage.status.erase(user_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (user_id == local_user) {
|
2020-10-08 19:38:55 +03:00
|
|
|
for (const auto &[user, status] : tmp) {
|
|
|
|
(void)status;
|
2020-10-08 00:03:14 +03:00
|
|
|
emit verificationStatusChanged(user);
|
2020-10-08 19:38:55 +03:00
|
|
|
}
|
2020-10-08 00:03:14 +03:00
|
|
|
} else {
|
|
|
|
emit verificationStatusChanged(user_id);
|
|
|
|
}
|
2020-10-02 02:14:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Cache::markDeviceUnverified(const std::string &user_id, const std::string &key)
|
|
|
|
{
|
2021-03-03 01:15:12 +03:00
|
|
|
std::string_view val;
|
2020-10-02 02:14:42 +03:00
|
|
|
|
|
|
|
auto txn = lmdb::txn::begin(env_);
|
|
|
|
auto db = getVerificationDb(txn);
|
|
|
|
|
|
|
|
try {
|
|
|
|
VerificationCache verified_state;
|
2021-03-03 01:15:12 +03:00
|
|
|
auto res = db.get(txn, user_id, val);
|
2020-10-02 02:14:42 +03:00
|
|
|
if (res) {
|
2021-03-03 01:15:12 +03:00
|
|
|
verified_state = json::parse(val);
|
2020-10-02 02:14:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
verified_state.device_verified.erase(
|
|
|
|
std::remove(verified_state.device_verified.begin(),
|
|
|
|
verified_state.device_verified.end(),
|
|
|
|
key),
|
|
|
|
verified_state.device_verified.end());
|
|
|
|
|
2021-03-03 01:15:12 +03:00
|
|
|
db.put(txn, user_id, json(verified_state).dump());
|
2020-10-02 02:14:42 +03:00
|
|
|
txn.commit();
|
|
|
|
} catch (std::exception &) {
|
|
|
|
}
|
2020-10-08 00:03:14 +03:00
|
|
|
|
|
|
|
const auto local_user = utils::localUser().toStdString();
|
|
|
|
std::map<std::string, VerificationStatus> tmp;
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> lock(verification_storage.verification_storage_mtx);
|
|
|
|
if (user_id == local_user) {
|
|
|
|
std::swap(tmp, verification_storage.status);
|
|
|
|
} else {
|
|
|
|
verification_storage.status.erase(user_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (user_id == local_user) {
|
2020-10-08 19:38:55 +03:00
|
|
|
for (const auto &[user, status] : tmp) {
|
|
|
|
(void)status;
|
2020-10-08 00:03:14 +03:00
|
|
|
emit verificationStatusChanged(user);
|
2020-10-08 19:38:55 +03:00
|
|
|
}
|
2020-10-08 00:03:14 +03:00
|
|
|
} else {
|
|
|
|
emit verificationStatusChanged(user_id);
|
|
|
|
}
|
2020-10-02 02:14:42 +03:00
|
|
|
}
|
|
|
|
|
2020-10-08 00:03:14 +03:00
|
|
|
VerificationStatus
|
|
|
|
Cache::verificationStatus(const std::string &user_id)
|
2020-10-02 02:14:42 +03:00
|
|
|
{
|
2020-10-08 00:03:14 +03:00
|
|
|
std::unique_lock<std::mutex> lock(verification_storage.verification_storage_mtx);
|
|
|
|
if (verification_storage.status.count(user_id))
|
|
|
|
return verification_storage.status.at(user_id);
|
2020-10-02 02:14:42 +03:00
|
|
|
|
2020-10-08 00:03:14 +03:00
|
|
|
VerificationStatus status;
|
|
|
|
|
|
|
|
if (auto verifCache = verificationCache(user_id)) {
|
|
|
|
status.verified_devices = verifCache->device_verified;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto local_user = utils::localUser().toStdString();
|
|
|
|
|
2021-05-07 13:19:46 +03:00
|
|
|
crypto::Trust trustlevel = crypto::Trust::Unverified;
|
|
|
|
if (user_id == local_user) {
|
2020-10-08 00:03:14 +03:00
|
|
|
status.verified_devices.push_back(http::client()->device_id());
|
2021-05-07 13:19:46 +03:00
|
|
|
trustlevel = crypto::Trust::Verified;
|
|
|
|
}
|
2020-10-08 00:03:14 +03:00
|
|
|
|
|
|
|
verification_storage.status[user_id] = status;
|
|
|
|
|
|
|
|
auto verifyAtLeastOneSig = [](const auto &toVerif,
|
|
|
|
const std::map<std::string, std::string> &keys,
|
|
|
|
const std::string &keyOwner) {
|
|
|
|
if (!toVerif.signatures.count(keyOwner))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (const auto &[key_id, signature] : toVerif.signatures.at(keyOwner)) {
|
|
|
|
if (!keys.count(key_id))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (mtx::crypto::ed25519_verify_signature(
|
|
|
|
keys.at(key_id), json(toVerif), signature))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
};
|
2020-10-02 02:14:42 +03:00
|
|
|
|
|
|
|
try {
|
2020-10-08 00:03:14 +03:00
|
|
|
// for local user verify this device_key -> our master_key -> our self_signing_key
|
|
|
|
// -> our device_keys
|
|
|
|
//
|
|
|
|
// for other user verify this device_key -> our master_key -> our user_signing_key
|
|
|
|
// -> their master_key -> their self_signing_key -> their device_keys
|
|
|
|
//
|
|
|
|
// This means verifying the other user adds 2 extra steps,verifying our user_signing
|
|
|
|
// key and their master key
|
|
|
|
auto ourKeys = userKeys(local_user);
|
|
|
|
auto theirKeys = userKeys(user_id);
|
|
|
|
if (!ourKeys || !theirKeys)
|
|
|
|
return status;
|
|
|
|
|
|
|
|
if (!mtx::crypto::ed25519_verify_signature(
|
|
|
|
olm::client()->identity_keys().ed25519,
|
|
|
|
json(ourKeys->master_keys),
|
|
|
|
ourKeys->master_keys.signatures.at(local_user)
|
|
|
|
.at("ed25519:" + http::client()->device_id())))
|
|
|
|
return status;
|
|
|
|
|
|
|
|
auto master_keys = ourKeys->master_keys.keys;
|
|
|
|
|
|
|
|
if (user_id != local_user) {
|
2021-05-07 18:01:03 +03:00
|
|
|
bool theirMasterKeyVerified =
|
|
|
|
verifyAtLeastOneSig(
|
|
|
|
ourKeys->user_signing_keys, master_keys, local_user) &&
|
|
|
|
verifyAtLeastOneSig(
|
|
|
|
theirKeys->master_keys, ourKeys->user_signing_keys.keys, local_user);
|
|
|
|
|
|
|
|
if (theirMasterKeyVerified)
|
|
|
|
trustlevel = crypto::Trust::Verified;
|
|
|
|
else if (!theirKeys->master_key_changed)
|
|
|
|
trustlevel = crypto::Trust::TOFU;
|
|
|
|
else
|
2020-10-08 00:03:14 +03:00
|
|
|
return status;
|
|
|
|
|
|
|
|
master_keys = theirKeys->master_keys.keys;
|
2020-10-02 02:14:42 +03:00
|
|
|
}
|
|
|
|
|
2021-05-07 18:01:03 +03:00
|
|
|
status.user_verified = trustlevel;
|
2020-10-08 00:03:14 +03:00
|
|
|
|
|
|
|
if (!verifyAtLeastOneSig(theirKeys->self_signing_keys, master_keys, user_id))
|
|
|
|
return status;
|
|
|
|
|
|
|
|
for (const auto &[device, device_key] : theirKeys->device_keys) {
|
2020-10-08 19:38:55 +03:00
|
|
|
(void)device;
|
2021-05-07 13:19:46 +03:00
|
|
|
try {
|
|
|
|
auto identkey =
|
|
|
|
device_key.keys.at("curve25519:" + device_key.device_id);
|
|
|
|
if (verifyAtLeastOneSig(
|
|
|
|
device_key, theirKeys->self_signing_keys.keys, user_id)) {
|
|
|
|
status.verified_devices.push_back(device_key.device_id);
|
|
|
|
status.verified_device_keys[identkey] = trustlevel;
|
|
|
|
}
|
|
|
|
} catch (...) {
|
|
|
|
}
|
2020-10-08 00:03:14 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
verification_storage.status[user_id] = status;
|
|
|
|
return status;
|
2020-10-02 02:14:42 +03:00
|
|
|
} catch (std::exception &) {
|
2020-10-08 00:03:14 +03:00
|
|
|
return status;
|
2020-10-02 02:14:42 +03:00
|
|
|
}
|
2020-06-28 18:31:34 +03:00
|
|
|
}
|
|
|
|
|
2019-06-27 21:53:44 +03:00
|
|
|
void
|
|
|
|
to_json(json &j, const RoomInfo &info)
|
|
|
|
{
|
|
|
|
j["name"] = info.name;
|
|
|
|
j["topic"] = info.topic;
|
|
|
|
j["avatar_url"] = info.avatar_url;
|
2019-07-04 20:18:32 +03:00
|
|
|
j["version"] = info.version;
|
2019-06-27 21:53:44 +03:00
|
|
|
j["is_invite"] = info.is_invite;
|
|
|
|
j["join_rule"] = info.join_rule;
|
|
|
|
j["guest_access"] = info.guest_access;
|
|
|
|
|
|
|
|
if (info.member_count != 0)
|
|
|
|
j["member_count"] = info.member_count;
|
|
|
|
|
|
|
|
if (info.tags.size() != 0)
|
|
|
|
j["tags"] = info.tags;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
from_json(const json &j, RoomInfo &info)
|
|
|
|
{
|
2019-07-04 20:18:32 +03:00
|
|
|
info.name = j.at("name");
|
|
|
|
info.topic = j.at("topic");
|
|
|
|
info.avatar_url = j.at("avatar_url");
|
|
|
|
info.version = j.value(
|
|
|
|
"version", QCoreApplication::translate("RoomInfo", "no version stored").toStdString());
|
2019-06-27 21:53:44 +03:00
|
|
|
info.is_invite = j.at("is_invite");
|
|
|
|
info.join_rule = j.at("join_rule");
|
|
|
|
info.guest_access = j.at("guest_access");
|
|
|
|
|
|
|
|
if (j.count("member_count"))
|
|
|
|
info.member_count = j.at("member_count");
|
|
|
|
|
|
|
|
if (j.count("tags"))
|
|
|
|
info.tags = j.at("tags").get<std::vector<std::string>>();
|
|
|
|
}
|
2019-12-15 01:39:02 +03:00
|
|
|
|
|
|
|
void
|
|
|
|
to_json(json &j, const ReadReceiptKey &key)
|
|
|
|
{
|
|
|
|
j = json{{"event_id", key.event_id}, {"room_id", key.room_id}};
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
from_json(const json &j, ReadReceiptKey &key)
|
|
|
|
{
|
|
|
|
key.event_id = j.at("event_id").get<std::string>();
|
|
|
|
key.room_id = j.at("room_id").get<std::string>();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
to_json(json &j, const MemberInfo &info)
|
|
|
|
{
|
|
|
|
j["name"] = info.name;
|
|
|
|
j["avatar_url"] = info.avatar_url;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
from_json(const json &j, MemberInfo &info)
|
|
|
|
{
|
|
|
|
info.name = j.at("name");
|
|
|
|
info.avatar_url = j.at("avatar_url");
|
|
|
|
}
|
|
|
|
|
2020-11-30 02:26:27 +03:00
|
|
|
void
|
|
|
|
to_json(nlohmann::json &obj, const DeviceAndMasterKeys &msg)
|
|
|
|
{
|
|
|
|
obj["devices"] = msg.devices;
|
|
|
|
obj["master_keys"] = msg.master_keys;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
from_json(const nlohmann::json &obj, DeviceAndMasterKeys &msg)
|
|
|
|
{
|
|
|
|
msg.devices = obj.at("devices").get<decltype(msg.devices)>();
|
|
|
|
msg.master_keys = obj.at("master_keys").get<decltype(msg.master_keys)>();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
to_json(nlohmann::json &obj, const SharedWithUsers &msg)
|
|
|
|
{
|
|
|
|
obj["keys"] = msg.keys;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
from_json(const nlohmann::json &obj, SharedWithUsers &msg)
|
|
|
|
{
|
|
|
|
msg.keys = obj.at("keys").get<std::map<std::string, DeviceAndMasterKeys>>();
|
|
|
|
}
|
|
|
|
|
2019-12-15 01:39:02 +03:00
|
|
|
void
|
|
|
|
to_json(nlohmann::json &obj, const OutboundGroupSessionData &msg)
|
|
|
|
{
|
|
|
|
obj["session_id"] = msg.session_id;
|
|
|
|
obj["session_key"] = msg.session_key;
|
|
|
|
obj["message_index"] = msg.message_index;
|
2021-03-15 18:24:01 +03:00
|
|
|
obj["ts"] = msg.timestamp;
|
2020-11-30 02:26:27 +03:00
|
|
|
|
|
|
|
obj["initially"] = msg.initially;
|
|
|
|
obj["currently"] = msg.currently;
|
2019-12-15 01:39:02 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
from_json(const nlohmann::json &obj, OutboundGroupSessionData &msg)
|
|
|
|
{
|
|
|
|
msg.session_id = obj.at("session_id");
|
|
|
|
msg.session_key = obj.at("session_key");
|
|
|
|
msg.message_index = obj.at("message_index");
|
2021-03-15 18:24:01 +03:00
|
|
|
msg.timestamp = obj.value("ts", 0ULL);
|
2020-11-30 02:26:27 +03:00
|
|
|
|
|
|
|
msg.initially = obj.value("initially", SharedWithUsers{});
|
|
|
|
msg.currently = obj.value("currently", SharedWithUsers{});
|
2019-12-15 01:39:02 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
to_json(nlohmann::json &obj, const DevicePublicKeys &msg)
|
|
|
|
{
|
|
|
|
obj["ed25519"] = msg.ed25519;
|
|
|
|
obj["curve25519"] = msg.curve25519;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
from_json(const nlohmann::json &obj, DevicePublicKeys &msg)
|
|
|
|
{
|
|
|
|
msg.ed25519 = obj.at("ed25519");
|
|
|
|
msg.curve25519 = obj.at("curve25519");
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
to_json(nlohmann::json &obj, const MegolmSessionIndex &msg)
|
|
|
|
{
|
|
|
|
obj["room_id"] = msg.room_id;
|
|
|
|
obj["session_id"] = msg.session_id;
|
|
|
|
obj["sender_key"] = msg.sender_key;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
from_json(const nlohmann::json &obj, MegolmSessionIndex &msg)
|
|
|
|
{
|
|
|
|
msg.room_id = obj.at("room_id");
|
|
|
|
msg.session_id = obj.at("session_id");
|
|
|
|
msg.sender_key = obj.at("sender_key");
|
|
|
|
}
|
2019-12-15 04:56:04 +03:00
|
|
|
|
2020-10-20 14:46:05 +03:00
|
|
|
void
|
|
|
|
to_json(nlohmann::json &obj, const StoredOlmSession &msg)
|
|
|
|
{
|
|
|
|
obj["ts"] = msg.last_message_ts;
|
|
|
|
obj["s"] = msg.pickled_session;
|
|
|
|
}
|
|
|
|
void
|
|
|
|
from_json(const nlohmann::json &obj, StoredOlmSession &msg)
|
|
|
|
{
|
|
|
|
msg.last_message_ts = obj.at("ts").get<uint64_t>();
|
|
|
|
msg.pickled_session = obj.at("s").get<std::string>();
|
|
|
|
}
|
|
|
|
|
2019-12-15 04:56:04 +03:00
|
|
|
namespace cache {
|
|
|
|
void
|
|
|
|
init(const QString &user_id)
|
|
|
|
{
|
|
|
|
qRegisterMetaType<RoomMember>();
|
|
|
|
qRegisterMetaType<RoomSearchResult>();
|
|
|
|
qRegisterMetaType<RoomInfo>();
|
|
|
|
qRegisterMetaType<QMap<QString, RoomInfo>>();
|
|
|
|
qRegisterMetaType<std::map<QString, RoomInfo>>();
|
|
|
|
qRegisterMetaType<std::map<QString, mtx::responses::Timeline>>();
|
2020-10-08 17:17:38 +03:00
|
|
|
qRegisterMetaType<mtx::responses::QueryKeys>();
|
2019-12-15 04:56:04 +03:00
|
|
|
|
|
|
|
instance_ = std::make_unique<Cache>(user_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
Cache *
|
|
|
|
client()
|
|
|
|
{
|
|
|
|
return instance_.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string
|
|
|
|
displayName(const std::string &room_id, const std::string &user_id)
|
|
|
|
{
|
|
|
|
return instance_->displayName(room_id, user_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
QString
|
|
|
|
displayName(const QString &room_id, const QString &user_id)
|
|
|
|
{
|
|
|
|
return instance_->displayName(room_id, user_id);
|
|
|
|
}
|
|
|
|
QString
|
|
|
|
avatarUrl(const QString &room_id, const QString &user_id)
|
|
|
|
{
|
|
|
|
return instance_->avatarUrl(room_id, user_id);
|
|
|
|
}
|
|
|
|
|
2020-06-08 02:45:24 +03:00
|
|
|
mtx::presence::PresenceState
|
|
|
|
presenceState(const std::string &user_id)
|
|
|
|
{
|
2021-05-15 00:35:34 +03:00
|
|
|
if (!instance_)
|
|
|
|
return {};
|
2020-06-08 02:45:24 +03:00
|
|
|
return instance_->presenceState(user_id);
|
|
|
|
}
|
|
|
|
std::string
|
|
|
|
statusMessage(const std::string &user_id)
|
|
|
|
{
|
|
|
|
return instance_->statusMessage(user_id);
|
|
|
|
}
|
2020-06-28 18:31:34 +03:00
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
// user cache stores user keys
|
|
|
|
std::optional<UserKeyCache>
|
|
|
|
userKeys(const std::string &user_id)
|
|
|
|
{
|
|
|
|
return instance_->userKeys(user_id);
|
|
|
|
}
|
|
|
|
void
|
|
|
|
updateUserKeys(const std::string &sync_token, const mtx::responses::QueryKeys &keyQuery)
|
2020-06-28 18:31:34 +03:00
|
|
|
{
|
2020-10-02 02:14:42 +03:00
|
|
|
instance_->updateUserKeys(sync_token, keyQuery);
|
2020-06-28 18:31:34 +03:00
|
|
|
}
|
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
// device & user verification cache
|
2020-10-08 00:03:14 +03:00
|
|
|
std::optional<VerificationStatus>
|
2020-10-02 02:14:42 +03:00
|
|
|
verificationStatus(const std::string &user_id)
|
2020-06-28 18:31:34 +03:00
|
|
|
{
|
2020-10-02 02:14:42 +03:00
|
|
|
return instance_->verificationStatus(user_id);
|
2020-06-28 18:31:34 +03:00
|
|
|
}
|
|
|
|
|
2020-10-02 02:14:42 +03:00
|
|
|
void
|
2020-10-08 00:03:14 +03:00
|
|
|
markDeviceVerified(const std::string &user_id, const std::string &device)
|
2020-06-28 18:31:34 +03:00
|
|
|
{
|
2020-10-08 00:03:14 +03:00
|
|
|
instance_->markDeviceVerified(user_id, device);
|
2020-06-28 18:31:34 +03:00
|
|
|
}
|
2020-06-08 02:45:24 +03:00
|
|
|
|
2019-12-15 04:56:04 +03:00
|
|
|
void
|
2020-10-08 00:03:14 +03:00
|
|
|
markDeviceUnverified(const std::string &user_id, const std::string &device)
|
2019-12-15 04:56:04 +03:00
|
|
|
{
|
2020-10-08 00:03:14 +03:00
|
|
|
instance_->markDeviceUnverified(user_id, device);
|
2019-12-15 04:56:04 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::string>
|
|
|
|
joinedRooms()
|
|
|
|
{
|
|
|
|
return instance_->joinedRooms();
|
|
|
|
}
|
|
|
|
|
|
|
|
QMap<QString, RoomInfo>
|
|
|
|
roomInfo(bool withInvites)
|
|
|
|
{
|
|
|
|
return instance_->roomInfo(withInvites);
|
|
|
|
}
|
2021-05-24 15:04:07 +03:00
|
|
|
QHash<QString, RoomInfo>
|
2019-12-15 04:56:04 +03:00
|
|
|
invites()
|
|
|
|
{
|
|
|
|
return instance_->invites();
|
|
|
|
}
|
|
|
|
|
|
|
|
QString
|
|
|
|
getRoomName(lmdb::txn &txn, lmdb::dbi &statesdb, lmdb::dbi &membersdb)
|
|
|
|
{
|
|
|
|
return instance_->getRoomName(txn, statesdb, membersdb);
|
|
|
|
}
|
|
|
|
mtx::events::state::JoinRule
|
|
|
|
getRoomJoinRule(lmdb::txn &txn, lmdb::dbi &statesdb)
|
|
|
|
{
|
|
|
|
return instance_->getRoomJoinRule(txn, statesdb);
|
|
|
|
}
|
|
|
|
bool
|
|
|
|
getRoomGuestAccess(lmdb::txn &txn, lmdb::dbi &statesdb)
|
|
|
|
{
|
|
|
|
return instance_->getRoomGuestAccess(txn, statesdb);
|
|
|
|
}
|
|
|
|
QString
|
|
|
|
getRoomTopic(lmdb::txn &txn, lmdb::dbi &statesdb)
|
|
|
|
{
|
|
|
|
return instance_->getRoomTopic(txn, statesdb);
|
|
|
|
}
|
|
|
|
QString
|
2020-11-26 00:45:33 +03:00
|
|
|
getRoomAvatarUrl(lmdb::txn &txn, lmdb::dbi &statesdb, lmdb::dbi &membersdb)
|
2019-12-15 04:56:04 +03:00
|
|
|
{
|
2020-11-26 00:45:33 +03:00
|
|
|
return instance_->getRoomAvatarUrl(txn, statesdb, membersdb);
|
2019-12-15 04:56:04 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
QString
|
|
|
|
getRoomVersion(lmdb::txn &txn, lmdb::dbi &statesdb)
|
|
|
|
{
|
|
|
|
return instance_->getRoomVersion(txn, statesdb);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<RoomMember>
|
|
|
|
getMembers(const std::string &room_id, std::size_t startIndex, std::size_t len)
|
|
|
|
{
|
|
|
|
return instance_->getMembers(room_id, startIndex, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
saveState(const mtx::responses::Sync &res)
|
|
|
|
{
|
|
|
|
instance_->saveState(res);
|
|
|
|
}
|
|
|
|
bool
|
|
|
|
isInitialized()
|
|
|
|
{
|
|
|
|
return instance_->isInitialized();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string
|
|
|
|
nextBatchToken()
|
|
|
|
{
|
|
|
|
return instance_->nextBatchToken();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
deleteData()
|
|
|
|
{
|
|
|
|
instance_->deleteData();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
removeInvite(lmdb::txn &txn, const std::string &room_id)
|
|
|
|
{
|
|
|
|
instance_->removeInvite(txn, room_id);
|
|
|
|
}
|
|
|
|
void
|
|
|
|
removeInvite(const std::string &room_id)
|
|
|
|
{
|
|
|
|
instance_->removeInvite(room_id);
|
|
|
|
}
|
|
|
|
void
|
|
|
|
removeRoom(lmdb::txn &txn, const std::string &roomid)
|
|
|
|
{
|
|
|
|
instance_->removeRoom(txn, roomid);
|
|
|
|
}
|
|
|
|
void
|
|
|
|
removeRoom(const std::string &roomid)
|
|
|
|
{
|
|
|
|
instance_->removeRoom(roomid);
|
|
|
|
}
|
|
|
|
void
|
|
|
|
removeRoom(const QString &roomid)
|
|
|
|
{
|
|
|
|
instance_->removeRoom(roomid.toStdString());
|
|
|
|
}
|
|
|
|
void
|
|
|
|
setup()
|
|
|
|
{
|
|
|
|
instance_->setup();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2020-05-02 17:44:50 +03:00
|
|
|
runMigrations()
|
|
|
|
{
|
|
|
|
return instance_->runMigrations();
|
|
|
|
}
|
|
|
|
|
|
|
|
cache::CacheVersion
|
|
|
|
formatVersion()
|
2019-12-15 04:56:04 +03:00
|
|
|
{
|
2020-05-02 17:44:50 +03:00
|
|
|
return instance_->formatVersion();
|
2019-12-15 04:56:04 +03:00
|
|
|
}
|
2020-05-02 17:44:50 +03:00
|
|
|
|
2019-12-15 04:56:04 +03:00
|
|
|
void
|
|
|
|
setCurrentFormat()
|
|
|
|
{
|
|
|
|
instance_->setCurrentFormat();
|
|
|
|
}
|
|
|
|
|
2020-10-28 15:06:28 +03:00
|
|
|
std::vector<QString>
|
|
|
|
roomIds()
|
2019-12-15 04:56:04 +03:00
|
|
|
{
|
2020-10-28 15:06:28 +03:00
|
|
|
return instance_->roomIds();
|
2019-12-15 04:56:04 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
QMap<QString, mtx::responses::Notifications>
|
|
|
|
getTimelineMentions()
|
|
|
|
{
|
|
|
|
return instance_->getTimelineMentions();
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Retrieve all the user ids from a room.
|
|
|
|
std::vector<std::string>
|
|
|
|
roomMembers(const std::string &room_id)
|
|
|
|
{
|
|
|
|
return instance_->roomMembers(room_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Check if the given user has power leve greater than than
|
|
|
|
//! lowest power level of the given events.
|
|
|
|
bool
|
|
|
|
hasEnoughPowerLevel(const std::vector<mtx::events::EventType> &eventTypes,
|
|
|
|
const std::string &room_id,
|
|
|
|
const std::string &user_id)
|
|
|
|
{
|
|
|
|
return instance_->hasEnoughPowerLevel(eventTypes, room_id, user_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
updateReadReceipt(lmdb::txn &txn, const std::string &room_id, const Receipts &receipts)
|
|
|
|
{
|
|
|
|
instance_->updateReadReceipt(txn, room_id, receipts);
|
|
|
|
}
|
|
|
|
|
|
|
|
UserReceipts
|
|
|
|
readReceipts(const QString &event_id, const QString &room_id)
|
|
|
|
{
|
|
|
|
return instance_->readReceipts(event_id, room_id);
|
|
|
|
}
|
|
|
|
|
2021-02-10 03:03:20 +03:00
|
|
|
std::optional<uint64_t>
|
|
|
|
getEventIndex(const std::string &room_id, std::string_view event_id)
|
|
|
|
{
|
|
|
|
return instance_->getEventIndex(room_id, event_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<std::pair<uint64_t, std::string>>
|
|
|
|
lastInvisibleEventAfter(const std::string &room_id, std::string_view event_id)
|
|
|
|
{
|
|
|
|
return instance_->lastInvisibleEventAfter(room_id, event_id);
|
|
|
|
}
|
|
|
|
|
2019-12-15 04:56:04 +03:00
|
|
|
RoomInfo
|
|
|
|
singleRoomInfo(const std::string &room_id)
|
|
|
|
{
|
|
|
|
return instance_->singleRoomInfo(room_id);
|
|
|
|
}
|
|
|
|
std::vector<std::string>
|
|
|
|
roomsWithStateUpdates(const mtx::responses::Sync &res)
|
|
|
|
{
|
|
|
|
return instance_->roomsWithStateUpdates(res);
|
|
|
|
}
|
|
|
|
std::vector<std::string>
|
|
|
|
roomsWithTagUpdates(const mtx::responses::Sync &res)
|
|
|
|
{
|
|
|
|
return instance_->roomsWithTagUpdates(res);
|
|
|
|
}
|
|
|
|
std::map<QString, RoomInfo>
|
|
|
|
getRoomInfo(const std::vector<std::string> &rooms)
|
|
|
|
{
|
|
|
|
return instance_->getRoomInfo(rooms);
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Calculates which the read status of a room.
|
|
|
|
//! Whether all the events in the timeline have been read.
|
|
|
|
bool
|
|
|
|
calculateRoomReadStatus(const std::string &room_id)
|
|
|
|
{
|
|
|
|
return instance_->calculateRoomReadStatus(room_id);
|
|
|
|
}
|
|
|
|
void
|
|
|
|
calculateRoomReadStatus()
|
|
|
|
{
|
|
|
|
instance_->calculateRoomReadStatus();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<RoomSearchResult>
|
|
|
|
searchRooms(const std::string &query, std::uint8_t max_items)
|
|
|
|
{
|
|
|
|
return instance_->searchRooms(query, max_items);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
markSentNotification(const std::string &event_id)
|
|
|
|
{
|
|
|
|
instance_->markSentNotification(event_id);
|
|
|
|
}
|
|
|
|
//! Removes an event from the sent notifications.
|
|
|
|
void
|
|
|
|
removeReadNotification(const std::string &event_id)
|
|
|
|
{
|
|
|
|
instance_->removeReadNotification(event_id);
|
|
|
|
}
|
|
|
|
//! Check if we have sent a desktop notification for the given event id.
|
|
|
|
bool
|
|
|
|
isNotificationSent(const std::string &event_id)
|
|
|
|
{
|
|
|
|
return instance_->isNotificationSent(event_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Add all notifications containing a user mention to the db.
|
|
|
|
void
|
|
|
|
saveTimelineMentions(const mtx::responses::Notifications &res)
|
|
|
|
{
|
|
|
|
instance_->saveTimelineMentions(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Remove old unused data.
|
|
|
|
void
|
|
|
|
deleteOldMessages()
|
|
|
|
{
|
|
|
|
instance_->deleteOldMessages();
|
|
|
|
}
|
|
|
|
void
|
|
|
|
deleteOldData() noexcept
|
|
|
|
{
|
|
|
|
instance_->deleteOldData();
|
|
|
|
}
|
|
|
|
//! Retrieve all saved room ids.
|
|
|
|
std::vector<std::string>
|
|
|
|
getRoomIds(lmdb::txn &txn)
|
|
|
|
{
|
|
|
|
return instance_->getRoomIds(txn);
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Mark a room that uses e2e encryption.
|
|
|
|
void
|
|
|
|
setEncryptedRoom(lmdb::txn &txn, const std::string &room_id)
|
|
|
|
{
|
|
|
|
instance_->setEncryptedRoom(txn, room_id);
|
|
|
|
}
|
|
|
|
bool
|
|
|
|
isRoomEncrypted(const std::string &room_id)
|
|
|
|
{
|
|
|
|
return instance_->isRoomEncrypted(room_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Check if a user is a member of the room.
|
|
|
|
bool
|
|
|
|
isRoomMember(const std::string &user_id, const std::string &room_id)
|
|
|
|
{
|
|
|
|
return instance_->isRoomMember(user_id, room_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Outbound Megolm Sessions
|
|
|
|
//
|
|
|
|
void
|
|
|
|
saveOutboundMegolmSession(const std::string &room_id,
|
|
|
|
const OutboundGroupSessionData &data,
|
2020-11-30 02:26:27 +03:00
|
|
|
mtx::crypto::OutboundGroupSessionPtr &session)
|
2019-12-15 04:56:04 +03:00
|
|
|
{
|
2020-11-30 02:26:27 +03:00
|
|
|
instance_->saveOutboundMegolmSession(room_id, data, session);
|
2019-12-15 04:56:04 +03:00
|
|
|
}
|
|
|
|
OutboundGroupSessionDataRef
|
|
|
|
getOutboundMegolmSession(const std::string &room_id)
|
|
|
|
{
|
|
|
|
return instance_->getOutboundMegolmSession(room_id);
|
|
|
|
}
|
|
|
|
bool
|
|
|
|
outboundMegolmSessionExists(const std::string &room_id) noexcept
|
|
|
|
{
|
|
|
|
return instance_->outboundMegolmSessionExists(room_id);
|
|
|
|
}
|
|
|
|
void
|
2020-11-27 06:56:44 +03:00
|
|
|
updateOutboundMegolmSession(const std::string &room_id,
|
2020-11-30 02:26:27 +03:00
|
|
|
const OutboundGroupSessionData &data,
|
2020-11-27 06:56:44 +03:00
|
|
|
mtx::crypto::OutboundGroupSessionPtr &session)
|
2019-12-15 04:56:04 +03:00
|
|
|
{
|
2020-11-30 02:26:27 +03:00
|
|
|
instance_->updateOutboundMegolmSession(room_id, data, session);
|
2019-12-15 04:56:04 +03:00
|
|
|
}
|
2020-10-03 19:38:28 +03:00
|
|
|
void
|
|
|
|
dropOutboundMegolmSession(const std::string &room_id)
|
|
|
|
{
|
|
|
|
instance_->dropOutboundMegolmSession(room_id);
|
|
|
|
}
|
2019-12-15 04:56:04 +03:00
|
|
|
|
|
|
|
void
|
|
|
|
importSessionKeys(const mtx::crypto::ExportedSessionKeys &keys)
|
|
|
|
{
|
|
|
|
instance_->importSessionKeys(keys);
|
|
|
|
}
|
|
|
|
mtx::crypto::ExportedSessionKeys
|
|
|
|
exportSessionKeys()
|
|
|
|
{
|
|
|
|
return instance_->exportSessionKeys();
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Inbound Megolm Sessions
|
|
|
|
//
|
|
|
|
void
|
|
|
|
saveInboundMegolmSession(const MegolmSessionIndex &index,
|
|
|
|
mtx::crypto::InboundGroupSessionPtr session)
|
|
|
|
{
|
|
|
|
instance_->saveInboundMegolmSession(index, std::move(session));
|
|
|
|
}
|
2020-11-27 06:19:03 +03:00
|
|
|
mtx::crypto::InboundGroupSessionPtr
|
2019-12-15 04:56:04 +03:00
|
|
|
getInboundMegolmSession(const MegolmSessionIndex &index)
|
|
|
|
{
|
|
|
|
return instance_->getInboundMegolmSession(index);
|
|
|
|
}
|
|
|
|
bool
|
|
|
|
inboundMegolmSessionExists(const MegolmSessionIndex &index)
|
|
|
|
{
|
|
|
|
return instance_->inboundMegolmSessionExists(index);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Olm Sessions
|
|
|
|
//
|
|
|
|
void
|
2020-10-20 14:46:05 +03:00
|
|
|
saveOlmSession(const std::string &curve25519,
|
|
|
|
mtx::crypto::OlmSessionPtr session,
|
|
|
|
uint64_t timestamp)
|
2019-12-15 04:56:04 +03:00
|
|
|
{
|
2020-10-20 14:46:05 +03:00
|
|
|
instance_->saveOlmSession(curve25519, std::move(session), timestamp);
|
2019-12-15 04:56:04 +03:00
|
|
|
}
|
|
|
|
std::vector<std::string>
|
|
|
|
getOlmSessions(const std::string &curve25519)
|
|
|
|
{
|
|
|
|
return instance_->getOlmSessions(curve25519);
|
|
|
|
}
|
|
|
|
std::optional<mtx::crypto::OlmSessionPtr>
|
|
|
|
getOlmSession(const std::string &curve25519, const std::string &session_id)
|
|
|
|
{
|
|
|
|
return instance_->getOlmSession(curve25519, session_id);
|
|
|
|
}
|
2020-10-20 14:46:05 +03:00
|
|
|
std::optional<mtx::crypto::OlmSessionPtr>
|
|
|
|
getLatestOlmSession(const std::string &curve25519)
|
|
|
|
{
|
|
|
|
return instance_->getLatestOlmSession(curve25519);
|
|
|
|
}
|
2019-12-15 04:56:04 +03:00
|
|
|
|
|
|
|
void
|
|
|
|
saveOlmAccount(const std::string &pickled)
|
|
|
|
{
|
|
|
|
instance_->saveOlmAccount(pickled);
|
|
|
|
}
|
|
|
|
std::string
|
|
|
|
restoreOlmAccount()
|
|
|
|
{
|
|
|
|
return instance_->restoreOlmAccount();
|
|
|
|
}
|
2020-12-17 00:10:09 +03:00
|
|
|
|
|
|
|
void
|
|
|
|
storeSecret(const std::string &name, const std::string &secret)
|
|
|
|
{
|
|
|
|
instance_->storeSecret(name, secret);
|
|
|
|
}
|
|
|
|
std::optional<std::string>
|
|
|
|
secret(const std::string &name)
|
|
|
|
{
|
|
|
|
return instance_->secret(name);
|
|
|
|
}
|
2019-12-15 04:56:04 +03:00
|
|
|
} // namespace cache
|