Add relations and order without hidden events to db
This commit is contained in:
parent
82eff09062
commit
0da1a6d5fc
119
src/Cache.cpp
119
src/Cache.cpp
@ -1245,23 +1245,23 @@ Cache::Messages
|
|||||||
Cache::getTimelineMessages(lmdb::txn &txn, const std::string &room_id, int64_t index, bool forward)
|
Cache::getTimelineMessages(lmdb::txn &txn, const std::string &room_id, int64_t index, bool forward)
|
||||||
{
|
{
|
||||||
// TODO(nico): Limit the messages returned by this maybe?
|
// TODO(nico): Limit the messages returned by this maybe?
|
||||||
auto orderDb = getEventOrderDb(txn, room_id);
|
auto orderDb = getOrderToMessageDb(txn, room_id);
|
||||||
auto eventsDb = getEventsDb(txn, room_id);
|
auto eventsDb = getEventsDb(txn, room_id);
|
||||||
|
|
||||||
Messages messages{};
|
Messages messages{};
|
||||||
|
|
||||||
lmdb::val indexVal, val;
|
lmdb::val indexVal, event_id;
|
||||||
|
|
||||||
auto cursor = lmdb::cursor::open(txn, orderDb);
|
auto cursor = lmdb::cursor::open(txn, orderDb);
|
||||||
if (index == std::numeric_limits<int64_t>::max()) {
|
if (index == std::numeric_limits<int64_t>::max()) {
|
||||||
if (cursor.get(indexVal, val, forward ? MDB_FIRST : MDB_LAST)) {
|
if (cursor.get(indexVal, event_id, forward ? MDB_FIRST : MDB_LAST)) {
|
||||||
index = *indexVal.data<int64_t>();
|
index = *indexVal.data<int64_t>();
|
||||||
} else {
|
} else {
|
||||||
messages.end_of_cache = true;
|
messages.end_of_cache = true;
|
||||||
return messages;
|
return messages;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (cursor.get(indexVal, val, MDB_SET)) {
|
if (cursor.get(indexVal, event_id, MDB_SET)) {
|
||||||
index = *indexVal.data<int64_t>();
|
index = *indexVal.data<int64_t>();
|
||||||
} else {
|
} else {
|
||||||
messages.end_of_cache = true;
|
messages.end_of_cache = true;
|
||||||
@ -1272,16 +1272,10 @@ Cache::getTimelineMessages(lmdb::txn &txn, const std::string &room_id, int64_t i
|
|||||||
int counter = 0;
|
int counter = 0;
|
||||||
|
|
||||||
bool ret;
|
bool ret;
|
||||||
while ((ret = cursor.get(indexVal, val, forward ? MDB_NEXT : MDB_LAST)) &&
|
while ((ret = cursor.get(indexVal, event_id, forward ? MDB_NEXT : MDB_LAST)) &&
|
||||||
counter++ < BATCH_SIZE) {
|
counter++ < BATCH_SIZE) {
|
||||||
auto obj = json::parse(std::string_view(val.data(), val.size()));
|
|
||||||
|
|
||||||
if (obj.count("event_id") == 0)
|
|
||||||
break;
|
|
||||||
|
|
||||||
lmdb::val event;
|
lmdb::val event;
|
||||||
bool success = lmdb::dbi_get(
|
bool success = lmdb::dbi_get(txn, eventsDb, event_id, event);
|
||||||
txn, eventsDb, lmdb::val(obj["event_id"].get<std::string>()), event);
|
|
||||||
if (!success)
|
if (!success)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
@ -1290,11 +1284,6 @@ Cache::getTimelineMessages(lmdb::txn &txn, const std::string &room_id, int64_t i
|
|||||||
json::parse(std::string_view(event.data(), event.size())), te);
|
json::parse(std::string_view(event.data(), event.size())), te);
|
||||||
|
|
||||||
messages.timeline.events.push_back(std::move(te.data));
|
messages.timeline.events.push_back(std::move(te.data));
|
||||||
|
|
||||||
if (forward && messages.timeline.prev_batch.empty() && obj.contains("prev_batch"))
|
|
||||||
messages.timeline.prev_batch = obj["prev_batch"];
|
|
||||||
else if (!forward && obj.contains("prev_batch"))
|
|
||||||
messages.timeline.prev_batch = obj["prev_batch"];
|
|
||||||
}
|
}
|
||||||
cursor.close();
|
cursor.close();
|
||||||
|
|
||||||
@ -1363,7 +1352,7 @@ Cache::roomInfo(bool withInvites)
|
|||||||
std::string
|
std::string
|
||||||
Cache::getLastEventId(lmdb::txn &txn, const std::string &room_id)
|
Cache::getLastEventId(lmdb::txn &txn, const std::string &room_id)
|
||||||
{
|
{
|
||||||
auto orderDb = getEventOrderDb(txn, room_id);
|
auto orderDb = getOrderToMessageDb(txn, room_id);
|
||||||
|
|
||||||
lmdb::val indexVal, val;
|
lmdb::val indexVal, val;
|
||||||
|
|
||||||
@ -1372,18 +1361,13 @@ Cache::getLastEventId(lmdb::txn &txn, const std::string &room_id)
|
|||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
auto obj = json::parse(std::string_view(val.data(), val.size()));
|
return std::string(val.data(), val.size());
|
||||||
|
|
||||||
if (obj.count("event_id") == 0)
|
|
||||||
return {};
|
|
||||||
else
|
|
||||||
return obj["event_id"];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
DescInfo
|
DescInfo
|
||||||
Cache::getLastMessageInfo(lmdb::txn &txn, const std::string &room_id)
|
Cache::getLastMessageInfo(lmdb::txn &txn, const std::string &room_id)
|
||||||
{
|
{
|
||||||
auto orderDb = getEventOrderDb(txn, room_id);
|
auto orderDb = getOrderToMessageDb(txn, room_id);
|
||||||
auto eventsDb = getEventsDb(txn, room_id);
|
auto eventsDb = getEventsDb(txn, room_id);
|
||||||
if (orderDb.size(txn) == 0)
|
if (orderDb.size(txn) == 0)
|
||||||
return DescInfo{};
|
return DescInfo{};
|
||||||
@ -1392,19 +1376,13 @@ Cache::getLastMessageInfo(lmdb::txn &txn, const std::string &room_id)
|
|||||||
|
|
||||||
DescInfo fallbackDesc{};
|
DescInfo fallbackDesc{};
|
||||||
|
|
||||||
lmdb::val indexVal, val;
|
lmdb::val indexVal, event_id;
|
||||||
|
|
||||||
auto cursor = lmdb::cursor::open(txn, orderDb);
|
auto cursor = lmdb::cursor::open(txn, orderDb);
|
||||||
cursor.get(indexVal, val, MDB_LAST);
|
cursor.get(indexVal, event_id, MDB_LAST);
|
||||||
while (cursor.get(indexVal, val, MDB_PREV)) {
|
while (cursor.get(indexVal, event_id, MDB_PREV)) {
|
||||||
auto temp = json::parse(std::string_view(val.data(), val.size()));
|
|
||||||
|
|
||||||
if (temp.count("event_id") == 0)
|
|
||||||
break;
|
|
||||||
|
|
||||||
lmdb::val event;
|
lmdb::val event;
|
||||||
bool success = lmdb::dbi_get(
|
bool success = lmdb::dbi_get(txn, eventsDb, event_id, event);
|
||||||
txn, eventsDb, lmdb::val(temp["event_id"].get<std::string>()), event);
|
|
||||||
if (!success)
|
if (!success)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
@ -2015,11 +1993,17 @@ Cache::saveTimelineMessages(lmdb::txn &txn,
|
|||||||
const std::string &room_id,
|
const std::string &room_id,
|
||||||
const mtx::responses::Timeline &res)
|
const mtx::responses::Timeline &res)
|
||||||
{
|
{
|
||||||
auto eventsDb = getEventsDb(txn, room_id);
|
auto eventsDb = getEventsDb(txn, room_id);
|
||||||
|
auto relationsDb = getRelationsDb(txn, room_id);
|
||||||
|
|
||||||
auto orderDb = getEventOrderDb(txn, room_id);
|
auto orderDb = getEventOrderDb(txn, room_id);
|
||||||
if (res.limited)
|
auto msg2orderDb = getMessageToOrderDb(txn, room_id);
|
||||||
|
auto order2msgDb = getOrderToMessageDb(txn, room_id);
|
||||||
|
if (res.limited) {
|
||||||
lmdb::dbi_drop(txn, orderDb, false);
|
lmdb::dbi_drop(txn, orderDb, false);
|
||||||
|
lmdb::dbi_drop(txn, msg2orderDb, false);
|
||||||
|
lmdb::dbi_drop(txn, order2msgDb, false);
|
||||||
|
}
|
||||||
|
|
||||||
using namespace mtx::events;
|
using namespace mtx::events;
|
||||||
using namespace mtx::events::state;
|
using namespace mtx::events::state;
|
||||||
@ -2031,6 +2015,12 @@ Cache::saveTimelineMessages(lmdb::txn &txn,
|
|||||||
index = *indexVal.data<int64_t>();
|
index = *indexVal.data<int64_t>();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int64_t msgIndex = 0;
|
||||||
|
auto msgCursor = lmdb::cursor::open(txn, order2msgDb);
|
||||||
|
if (msgCursor.get(indexVal, val, MDB_LAST)) {
|
||||||
|
msgIndex = *indexVal.data<int64_t>();
|
||||||
|
}
|
||||||
|
|
||||||
bool first = true;
|
bool first = true;
|
||||||
for (const auto &e : res.events) {
|
for (const auto &e : res.events) {
|
||||||
auto event = mtx::accessors::serialize_event(e);
|
auto event = mtx::accessors::serialize_event(e);
|
||||||
@ -2039,17 +2029,17 @@ Cache::saveTimelineMessages(lmdb::txn &txn,
|
|||||||
lmdb::dbi_put(
|
lmdb::dbi_put(
|
||||||
txn, eventsDb, lmdb::val(redaction->redacts), lmdb::val(event.dump()));
|
txn, eventsDb, lmdb::val(redaction->redacts), lmdb::val(event.dump()));
|
||||||
} else {
|
} else {
|
||||||
lmdb::dbi_put(txn,
|
std::string event_id_val = event["event_id"].get<std::string>();
|
||||||
eventsDb,
|
lmdb::val event_id = event_id_val;
|
||||||
lmdb::val(event["event_id"].get<std::string>()),
|
lmdb::dbi_put(txn, eventsDb, event_id, lmdb::val(event.dump()));
|
||||||
lmdb::val(event.dump()));
|
|
||||||
|
|
||||||
++index;
|
++index;
|
||||||
|
|
||||||
json orderEntry = json::object();
|
json orderEntry = json::object();
|
||||||
orderEntry["event_id"] = event["event_id"];
|
orderEntry["event_id"] = event_id_val;
|
||||||
if (first)
|
if (first)
|
||||||
orderEntry["prev_batch"] = res.prev_batch;
|
orderEntry["prev_batch"] = res.prev_batch;
|
||||||
|
first = false;
|
||||||
|
|
||||||
nhlog::db()->debug("saving '{}'", orderEntry.dump());
|
nhlog::db()->debug("saving '{}'", orderEntry.dump());
|
||||||
|
|
||||||
@ -2057,7 +2047,32 @@ Cache::saveTimelineMessages(lmdb::txn &txn,
|
|||||||
lmdb::val(&index, sizeof(index)),
|
lmdb::val(&index, sizeof(index)),
|
||||||
lmdb::val(orderEntry.dump()),
|
lmdb::val(orderEntry.dump()),
|
||||||
MDB_APPEND);
|
MDB_APPEND);
|
||||||
first = false;
|
|
||||||
|
// TODO(Nico): Allow blacklisting more event types in UI
|
||||||
|
if (event["type"] != "m.reaction" && event["type"] != "m.dummy") {
|
||||||
|
++msgIndex;
|
||||||
|
lmdb::cursor_put(msgCursor.handle(),
|
||||||
|
lmdb::val(&msgIndex, sizeof(msgIndex)),
|
||||||
|
event_id,
|
||||||
|
MDB_APPEND);
|
||||||
|
|
||||||
|
lmdb::dbi_put(txn,
|
||||||
|
msg2orderDb,
|
||||||
|
event_id,
|
||||||
|
lmdb::val(&msgIndex, sizeof(msgIndex)));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event.contains("content") &&
|
||||||
|
event["content"].contains("m.relates_to")) {
|
||||||
|
auto temp = event["content"]["m.relates_to"];
|
||||||
|
std::string relates_to = temp.contains("m.in_reply_to")
|
||||||
|
? temp["m.in_reply_to"]["event_id"]
|
||||||
|
: temp["event_id"];
|
||||||
|
|
||||||
|
if (!relates_to.empty())
|
||||||
|
lmdb::dbi_put(
|
||||||
|
txn, relationsDb, lmdb::val(relates_to), event_id);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -2201,6 +2216,8 @@ Cache::deleteOldMessages()
|
|||||||
|
|
||||||
for (const auto &room_id : room_ids) {
|
for (const auto &room_id : room_ids) {
|
||||||
auto orderDb = getEventOrderDb(txn, room_id);
|
auto orderDb = getEventOrderDb(txn, room_id);
|
||||||
|
auto o2m = getOrderToMessageDb(txn, room_id);
|
||||||
|
auto m2o = getMessageToOrderDb(txn, room_id);
|
||||||
auto eventsDb = getEventsDb(txn, room_id);
|
auto eventsDb = getEventsDb(txn, room_id);
|
||||||
auto cursor = lmdb::cursor::open(txn, orderDb);
|
auto cursor = lmdb::cursor::open(txn, orderDb);
|
||||||
|
|
||||||
@ -2224,9 +2241,17 @@ Cache::deleteOldMessages()
|
|||||||
message_count-- < MAX_RESTORED_MESSAGES) {
|
message_count-- < MAX_RESTORED_MESSAGES) {
|
||||||
auto obj = json::parse(std::string_view(val.data(), val.size()));
|
auto obj = json::parse(std::string_view(val.data(), val.size()));
|
||||||
|
|
||||||
if (obj.count("event_id") != 0)
|
if (obj.count("event_id") != 0) {
|
||||||
lmdb::dbi_del(
|
lmdb::val event_id = obj["event_id"].get<std::string>();
|
||||||
txn, eventsDb, lmdb::val(obj["event_id"].get<std::string>()));
|
lmdb::dbi_del(txn, eventsDb, event_id);
|
||||||
|
|
||||||
|
lmdb::val order{};
|
||||||
|
bool exists = lmdb::dbi_get(txn, m2o, event_id, order);
|
||||||
|
if (exists) {
|
||||||
|
lmdb::dbi_del(txn, o2m, order);
|
||||||
|
lmdb::dbi_del(txn, m2o, event_id);
|
||||||
|
}
|
||||||
|
}
|
||||||
lmdb::cursor_del(cursor);
|
lmdb::cursor_del(cursor);
|
||||||
}
|
}
|
||||||
cursor.close();
|
cursor.close();
|
||||||
|
@ -424,6 +424,24 @@ private:
|
|||||||
txn, std::string(room_id + "/event_order").c_str(), MDB_CREATE | MDB_INTEGERKEY);
|
txn, std::string(room_id + "/event_order").c_str(), MDB_CREATE | MDB_INTEGERKEY);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
lmdb::dbi getMessageToOrderDb(lmdb::txn &txn, const std::string &room_id)
|
||||||
|
{
|
||||||
|
return lmdb::dbi::open(
|
||||||
|
txn, std::string(room_id + "/msg2order").c_str(), MDB_CREATE);
|
||||||
|
}
|
||||||
|
|
||||||
|
lmdb::dbi getOrderToMessageDb(lmdb::txn &txn, const std::string &room_id)
|
||||||
|
{
|
||||||
|
return lmdb::dbi::open(
|
||||||
|
txn, std::string(room_id + "/order2msg").c_str(), MDB_CREATE | MDB_INTEGERKEY);
|
||||||
|
}
|
||||||
|
|
||||||
|
lmdb::dbi getRelationsDb(lmdb::txn &txn, const std::string &room_id)
|
||||||
|
{
|
||||||
|
return lmdb::dbi::open(
|
||||||
|
txn, std::string(room_id + "/related").c_str(), MDB_CREATE | MDB_DUPSORT);
|
||||||
|
}
|
||||||
|
|
||||||
lmdb::dbi getInviteStatesDb(lmdb::txn &txn, const std::string &room_id)
|
lmdb::dbi getInviteStatesDb(lmdb::txn &txn, const std::string &room_id)
|
||||||
{
|
{
|
||||||
return lmdb::dbi::open(
|
return lmdb::dbi::open(
|
||||||
|
Loading…
Reference in New Issue
Block a user