improvement: more efficient /sync with gaps
parent
506c2a3146
commit
ea3aaa6b5c
|
@ -134,9 +134,9 @@ checksum = "0e4cec68f03f32e44924783795810fa50a7035d8c8ebe78580ad7e6c703fba38"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "cc"
|
name = "cc"
|
||||||
version = "1.0.59"
|
version = "1.0.60"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "66120af515773fb005778dc07c261bd201ec8ce50bd6e7144c927753fe013381"
|
checksum = "ef611cc68ff783f18535d77ddd080185275713d852c4f5cbb6122c462a7a825c"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "cfg-if"
|
name = "cfg-if"
|
||||||
|
@ -213,7 +213,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "1373a16a4937bc34efec7b391f9c1500c30b8478a701a4f44c9165cc0475a6e0"
|
checksum = "1373a16a4937bc34efec7b391f9c1500c30b8478a701a4f44c9165cc0475a6e0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"percent-encoding",
|
"percent-encoding",
|
||||||
"time 0.2.19",
|
"time 0.2.20",
|
||||||
"version_check",
|
"version_check",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -342,9 +342,9 @@ checksum = "134951f4028bdadb9b84baf4232681efbf277da25144b9b0ad65df75946c422b"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "either"
|
name = "either"
|
||||||
version = "1.6.0"
|
version = "1.6.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "cd56b59865bce947ac5958779cfa508f6c3b9497cc762b7e24a12d11ccde2c4f"
|
checksum = "e78d4f1cc4ae33bbfc157ed5d5a5ef3bc29227303d595861deb238fcec4e9457"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "encoding_rs"
|
name = "encoding_rs"
|
||||||
|
@ -1370,7 +1370,7 @@ dependencies = [
|
||||||
"rocket_codegen",
|
"rocket_codegen",
|
||||||
"rocket_http",
|
"rocket_http",
|
||||||
"state",
|
"state",
|
||||||
"time 0.2.19",
|
"time 0.2.20",
|
||||||
"tokio",
|
"tokio",
|
||||||
"toml",
|
"toml",
|
||||||
"version_check",
|
"version_check",
|
||||||
|
@ -1405,7 +1405,7 @@ dependencies = [
|
||||||
"ref-cast",
|
"ref-cast",
|
||||||
"smallvec",
|
"smallvec",
|
||||||
"state",
|
"state",
|
||||||
"time 0.2.19",
|
"time 0.2.20",
|
||||||
"tokio",
|
"tokio",
|
||||||
"tokio-rustls",
|
"tokio-rustls",
|
||||||
"unicode-xid",
|
"unicode-xid",
|
||||||
|
@ -1414,7 +1414,7 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ruma"
|
name = "ruma"
|
||||||
version = "0.0.1"
|
version = "0.0.1"
|
||||||
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#6ccb3ecaf69167ba405379826a9d87a98f168df8"
|
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#425d34d4cfb5aefe5bab6957d71bc9389384c1e5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"ruma-api",
|
"ruma-api",
|
||||||
"ruma-appservice-api",
|
"ruma-appservice-api",
|
||||||
|
@ -1430,7 +1430,7 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ruma-api"
|
name = "ruma-api"
|
||||||
version = "0.17.0-alpha.1"
|
version = "0.17.0-alpha.1"
|
||||||
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#6ccb3ecaf69167ba405379826a9d87a98f168df8"
|
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#425d34d4cfb5aefe5bab6957d71bc9389384c1e5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"http",
|
"http",
|
||||||
"percent-encoding",
|
"percent-encoding",
|
||||||
|
@ -1445,7 +1445,7 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ruma-api-macros"
|
name = "ruma-api-macros"
|
||||||
version = "0.17.0-alpha.1"
|
version = "0.17.0-alpha.1"
|
||||||
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#6ccb3ecaf69167ba405379826a9d87a98f168df8"
|
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#425d34d4cfb5aefe5bab6957d71bc9389384c1e5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro-crate",
|
"proc-macro-crate",
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
|
@ -1456,7 +1456,7 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ruma-appservice-api"
|
name = "ruma-appservice-api"
|
||||||
version = "0.2.0-alpha.1"
|
version = "0.2.0-alpha.1"
|
||||||
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#6ccb3ecaf69167ba405379826a9d87a98f168df8"
|
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#425d34d4cfb5aefe5bab6957d71bc9389384c1e5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"ruma-api",
|
"ruma-api",
|
||||||
"ruma-common",
|
"ruma-common",
|
||||||
|
@ -1469,7 +1469,7 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ruma-client-api"
|
name = "ruma-client-api"
|
||||||
version = "0.10.0-alpha.1"
|
version = "0.10.0-alpha.1"
|
||||||
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#6ccb3ecaf69167ba405379826a9d87a98f168df8"
|
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#425d34d4cfb5aefe5bab6957d71bc9389384c1e5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"assign",
|
"assign",
|
||||||
"http",
|
"http",
|
||||||
|
@ -1488,7 +1488,7 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ruma-common"
|
name = "ruma-common"
|
||||||
version = "0.2.0"
|
version = "0.2.0"
|
||||||
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#6ccb3ecaf69167ba405379826a9d87a98f168df8"
|
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#425d34d4cfb5aefe5bab6957d71bc9389384c1e5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"js_int",
|
"js_int",
|
||||||
"ruma-api",
|
"ruma-api",
|
||||||
|
@ -1502,7 +1502,7 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ruma-events"
|
name = "ruma-events"
|
||||||
version = "0.22.0-alpha.1"
|
version = "0.22.0-alpha.1"
|
||||||
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#6ccb3ecaf69167ba405379826a9d87a98f168df8"
|
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#425d34d4cfb5aefe5bab6957d71bc9389384c1e5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"js_int",
|
"js_int",
|
||||||
"ruma-common",
|
"ruma-common",
|
||||||
|
@ -1517,7 +1517,7 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ruma-events-macros"
|
name = "ruma-events-macros"
|
||||||
version = "0.22.0-alpha.1"
|
version = "0.22.0-alpha.1"
|
||||||
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#6ccb3ecaf69167ba405379826a9d87a98f168df8"
|
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#425d34d4cfb5aefe5bab6957d71bc9389384c1e5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro-crate",
|
"proc-macro-crate",
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
|
@ -1528,7 +1528,7 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ruma-federation-api"
|
name = "ruma-federation-api"
|
||||||
version = "0.0.3"
|
version = "0.0.3"
|
||||||
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#6ccb3ecaf69167ba405379826a9d87a98f168df8"
|
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#425d34d4cfb5aefe5bab6957d71bc9389384c1e5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"js_int",
|
"js_int",
|
||||||
"ruma-api",
|
"ruma-api",
|
||||||
|
@ -1543,7 +1543,7 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ruma-identifiers"
|
name = "ruma-identifiers"
|
||||||
version = "0.17.4"
|
version = "0.17.4"
|
||||||
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#6ccb3ecaf69167ba405379826a9d87a98f168df8"
|
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#425d34d4cfb5aefe5bab6957d71bc9389384c1e5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"rand",
|
"rand",
|
||||||
"ruma-identifiers-macros",
|
"ruma-identifiers-macros",
|
||||||
|
@ -1555,7 +1555,7 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ruma-identifiers-macros"
|
name = "ruma-identifiers-macros"
|
||||||
version = "0.17.4"
|
version = "0.17.4"
|
||||||
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#6ccb3ecaf69167ba405379826a9d87a98f168df8"
|
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#425d34d4cfb5aefe5bab6957d71bc9389384c1e5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
|
@ -1566,7 +1566,7 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ruma-identifiers-validation"
|
name = "ruma-identifiers-validation"
|
||||||
version = "0.1.1"
|
version = "0.1.1"
|
||||||
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#6ccb3ecaf69167ba405379826a9d87a98f168df8"
|
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#425d34d4cfb5aefe5bab6957d71bc9389384c1e5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"serde",
|
"serde",
|
||||||
"strum",
|
"strum",
|
||||||
|
@ -1575,7 +1575,7 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ruma-serde"
|
name = "ruma-serde"
|
||||||
version = "0.2.3"
|
version = "0.2.3"
|
||||||
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#6ccb3ecaf69167ba405379826a9d87a98f168df8"
|
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#425d34d4cfb5aefe5bab6957d71bc9389384c1e5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"form_urlencoded",
|
"form_urlencoded",
|
||||||
"itoa",
|
"itoa",
|
||||||
|
@ -1587,7 +1587,7 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ruma-signatures"
|
name = "ruma-signatures"
|
||||||
version = "0.6.0-dev.1"
|
version = "0.6.0-dev.1"
|
||||||
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#6ccb3ecaf69167ba405379826a9d87a98f168df8"
|
source = "git+https://github.com/timokoesters/ruma?branch=timo-fed-fixes#425d34d4cfb5aefe5bab6957d71bc9389384c1e5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"base64",
|
"base64",
|
||||||
"ring",
|
"ring",
|
||||||
|
@ -1831,7 +1831,7 @@ checksum = "7345c971d1ef21ffdbd103a75990a15eb03604fc8b8852ca8cb418ee1a099028"
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "state-res"
|
name = "state-res"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
source = "git+https://github.com/timokoesters/state-res?branch=spec-comp#1d01b6e65b6afd50e65085fb40f1e7d2782f519e"
|
source = "git+https://github.com/timokoesters/state-res?branch=spec-comp#d11a3feb5307715ab5d86af8f25d4bccfee6264b"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"itertools",
|
"itertools",
|
||||||
"js_int",
|
"js_int",
|
||||||
|
@ -1981,9 +1981,9 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "time"
|
name = "time"
|
||||||
version = "0.2.19"
|
version = "0.2.20"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "80c1a1fd93112fc50b11c43a1def21f926be3c18884fad676ea879572da070a1"
|
checksum = "0d4953c513c9bf1b97e9cdd83f11d60c4b0a83462880a360d80d96953a953fee"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"const_fn",
|
"const_fn",
|
||||||
"libc",
|
"libc",
|
||||||
|
|
|
@ -49,7 +49,10 @@ pub fn get_context_route(
|
||||||
.filter_map(|r| r.ok()) // Remove buggy events
|
.filter_map(|r| r.ok()) // Remove buggy events
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
let start_token = events_before.last().map(|(count, _)| count.to_string());
|
let start_token = events_before
|
||||||
|
.last()
|
||||||
|
.and_then(|(pdu_id, _)| db.rooms.pdu_count(pdu_id).ok())
|
||||||
|
.map(|count| count.to_string());
|
||||||
|
|
||||||
let events_before = events_before
|
let events_before = events_before
|
||||||
.into_iter()
|
.into_iter()
|
||||||
|
@ -68,7 +71,10 @@ pub fn get_context_route(
|
||||||
.filter_map(|r| r.ok()) // Remove buggy events
|
.filter_map(|r| r.ok()) // Remove buggy events
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
let end_token = events_after.last().map(|(count, _)| count.to_string());
|
let end_token = events_after
|
||||||
|
.last()
|
||||||
|
.and_then(|(pdu_id, _)| db.rooms.pdu_count(pdu_id).ok())
|
||||||
|
.map(|count| count.to_string());
|
||||||
|
|
||||||
let events_after = events_after
|
let events_after = events_after
|
||||||
.into_iter()
|
.into_iter()
|
||||||
|
|
|
@ -601,8 +601,7 @@ async fn join_room_by_id_helper(
|
||||||
.cloned()
|
.cloned()
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
let power_level =
|
let power_level = resolved_control_events.get(&(EventType::RoomPowerLevels, "".into()));
|
||||||
resolved_control_events.get(&(EventType::RoomPowerLevels, Some("".into())));
|
|
||||||
// Sort the remaining non control events
|
// Sort the remaining non control events
|
||||||
let sorted_event_ids = state_res::StateResolution::mainline_sort(
|
let sorted_event_ids = state_res::StateResolution::mainline_sort(
|
||||||
room_id,
|
room_id,
|
||||||
|
@ -644,13 +643,7 @@ async fn join_room_by_id_helper(
|
||||||
)?;
|
)?;
|
||||||
|
|
||||||
if state_events.contains(ev_id) {
|
if state_events.contains(ev_id) {
|
||||||
state.insert(
|
state.insert((pdu.kind(), pdu.state_key()), pdu_id);
|
||||||
(
|
|
||||||
pdu.kind(),
|
|
||||||
pdu.state_key().expect("State events have a state key"),
|
|
||||||
),
|
|
||||||
pdu_id,
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -117,6 +117,12 @@ pub fn get_message_events_route(
|
||||||
.pdus_after(&sender_id, &body.room_id, from)
|
.pdus_after(&sender_id, &body.room_id, from)
|
||||||
.take(limit)
|
.take(limit)
|
||||||
.filter_map(|r| r.ok()) // Filter out buggy events
|
.filter_map(|r| r.ok()) // Filter out buggy events
|
||||||
|
.filter_map(|(pdu_id, pdu)| {
|
||||||
|
db.rooms
|
||||||
|
.pdu_count(&pdu_id)
|
||||||
|
.map(|pdu_count| (pdu_count, pdu))
|
||||||
|
.ok()
|
||||||
|
})
|
||||||
.take_while(|&(k, _)| Some(Ok(k)) != to) // Stop at `to`
|
.take_while(|&(k, _)| Some(Ok(k)) != to) // Stop at `to`
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
|
@ -141,6 +147,12 @@ pub fn get_message_events_route(
|
||||||
.pdus_until(&sender_id, &body.room_id, from)
|
.pdus_until(&sender_id, &body.room_id, from)
|
||||||
.take(limit)
|
.take(limit)
|
||||||
.filter_map(|r| r.ok()) // Filter out buggy events
|
.filter_map(|r| r.ok()) // Filter out buggy events
|
||||||
|
.filter_map(|(pdu_id, pdu)| {
|
||||||
|
db.rooms
|
||||||
|
.pdu_count(&pdu_id)
|
||||||
|
.map(|pdu_count| (pdu_count, pdu))
|
||||||
|
.ok()
|
||||||
|
})
|
||||||
.take_while(|&(k, _)| Some(Ok(k)) != to) // Stop at `to`
|
.take_while(|&(k, _)| Some(Ok(k)) != to) // Stop at `to`
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
|
|
|
@ -105,50 +105,92 @@ pub async fn sync_events_route(
|
||||||
.room_state_get(&room_id, &EventType::RoomEncryption, "")?
|
.room_state_get(&room_id, &EventType::RoomEncryption, "")?
|
||||||
.is_some();
|
.is_some();
|
||||||
|
|
||||||
// TODO: optimize this?
|
// Database queries:
|
||||||
let mut send_member_count = false;
|
let since_state_hash = db
|
||||||
let mut joined_since_last_sync = false;
|
|
||||||
let mut new_encrypted_room = false;
|
|
||||||
for (state_key, pdu) in db
|
|
||||||
.rooms
|
.rooms
|
||||||
.pdus_since(&sender_id, &room_id, since)?
|
.pdus_until(sender_id, &room_id, since)
|
||||||
.filter_map(|r| r.ok())
|
.next()
|
||||||
.filter_map(|(_, pdu)| Some((pdu.state_key.clone()?, pdu)))
|
.and_then(|pdu| pdu.ok())
|
||||||
{
|
.and_then(|pdu| db.rooms.pdu_state_hash(&pdu.0).ok()?);
|
||||||
if pdu.kind == EventType::RoomMember {
|
|
||||||
send_member_count = true;
|
|
||||||
|
|
||||||
let content = serde_json::from_value::<
|
let since_members = since_state_hash
|
||||||
Raw<ruma::events::room::member::MemberEventContent>,
|
.as_ref()
|
||||||
>(pdu.content.clone())
|
.and_then(|state_hash| db.rooms.state_type(state_hash, &EventType::RoomMember).ok());
|
||||||
|
|
||||||
|
let since_encryption = since_state_hash.as_ref().and_then(|state_hash| {
|
||||||
|
db.rooms
|
||||||
|
.state_get(&state_hash, &EventType::RoomEncryption, "")
|
||||||
|
.ok()
|
||||||
|
});
|
||||||
|
|
||||||
|
let current_members = db.rooms.room_state_type(&room_id, &EventType::RoomMember)?;
|
||||||
|
|
||||||
|
// Calculations:
|
||||||
|
let new_encrypted_room = encrypted_room && since_encryption.is_none();
|
||||||
|
|
||||||
|
let send_member_count = since_members.as_ref().map_or(true, |since_members| {
|
||||||
|
current_members.len() != since_members.len()
|
||||||
|
});
|
||||||
|
|
||||||
|
let since_sender_member = since_members.as_ref().and_then(|members| {
|
||||||
|
members.get(sender_id.as_str()).and_then(|pdu| {
|
||||||
|
serde_json::from_value::<Raw<ruma::events::room::member::MemberEventContent>>(
|
||||||
|
pdu.content.clone(),
|
||||||
|
)
|
||||||
.expect("Raw::from_value always works")
|
.expect("Raw::from_value always works")
|
||||||
.deserialize()
|
.deserialize()
|
||||||
.map_err(|_| Error::bad_database("Invalid PDU in database."))?;
|
.map_err(|_| Error::bad_database("Invalid PDU in database."))
|
||||||
|
.ok()
|
||||||
|
})
|
||||||
|
});
|
||||||
|
|
||||||
if pdu.state_key == Some(sender_id.to_string())
|
if encrypted_room {
|
||||||
&& content.membership == MembershipState::Join
|
for (user_id, current_member) in current_members {
|
||||||
{
|
let current_membership = serde_json::from_value::<
|
||||||
joined_since_last_sync = true;
|
Raw<ruma::events::room::member::MemberEventContent>,
|
||||||
} else if encrypted_room && content.membership == MembershipState::Join {
|
>(current_member.content.clone())
|
||||||
// A new user joined an encrypted room
|
.expect("Raw::from_value always works")
|
||||||
let user_id = UserId::try_from(state_key)
|
.deserialize()
|
||||||
.map_err(|_| Error::bad_database("Invalid UserId in member PDU."))?;
|
.map_err(|_| Error::bad_database("Invalid PDU in database."))?
|
||||||
// Add encryption update if we didn't share an encrypted room already
|
.membership;
|
||||||
if !share_encrypted_room(&db, &sender_id, &user_id, &room_id) {
|
|
||||||
device_list_updates.insert(user_id);
|
let since_membership = since_members
|
||||||
|
.as_ref()
|
||||||
|
.and_then(|members| {
|
||||||
|
members.get(&user_id).and_then(|since_member| {
|
||||||
|
serde_json::from_value::<
|
||||||
|
Raw<ruma::events::room::member::MemberEventContent>,
|
||||||
|
>(since_member.content.clone())
|
||||||
|
.expect("Raw::from_value always works")
|
||||||
|
.deserialize()
|
||||||
|
.map_err(|_| Error::bad_database("Invalid PDU in database."))
|
||||||
|
.ok()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.map_or(MembershipState::Leave, |member| member.membership);
|
||||||
|
|
||||||
|
let user_id = UserId::try_from(user_id)
|
||||||
|
.map_err(|_| Error::bad_database("Invalid UserId in member PDU."))?;
|
||||||
|
|
||||||
|
match (since_membership, current_membership) {
|
||||||
|
(MembershipState::Leave, MembershipState::Join) => {
|
||||||
|
// A new user joined an encrypted room
|
||||||
|
if !share_encrypted_room(&db, &sender_id, &user_id, &room_id) {
|
||||||
|
device_list_updates.insert(user_id);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else if encrypted_room && content.membership == MembershipState::Leave {
|
(MembershipState::Join, MembershipState::Leave) => {
|
||||||
// Write down users that have left encrypted rooms we are in
|
// Write down users that have left encrypted rooms we are in
|
||||||
left_encrypted_users.insert(
|
left_encrypted_users.insert(user_id);
|
||||||
UserId::try_from(state_key)
|
}
|
||||||
.map_err(|_| Error::bad_database("Invalid UserId in member PDU."))?,
|
_ => {}
|
||||||
);
|
|
||||||
}
|
}
|
||||||
} else if pdu.kind == EventType::RoomEncryption {
|
|
||||||
new_encrypted_room = true;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let joined_since_last_sync =
|
||||||
|
since_sender_member.map_or(true, |member| member.membership != MembershipState::Join);
|
||||||
|
|
||||||
if joined_since_last_sync && encrypted_room || new_encrypted_room {
|
if joined_since_last_sync && encrypted_room || new_encrypted_room {
|
||||||
// If the user is in a new encrypted room, give them all joined users
|
// If the user is in a new encrypted room, give them all joined users
|
||||||
device_list_updates.extend(
|
device_list_updates.extend(
|
||||||
|
@ -390,23 +432,37 @@ pub async fn sync_events_route(
|
||||||
state: sync_events::State { events: Vec::new() },
|
state: sync_events::State { events: Vec::new() },
|
||||||
};
|
};
|
||||||
|
|
||||||
let mut left_since_last_sync = false;
|
let since_member = db
|
||||||
for pdu in db.rooms.pdus_since(&sender_id, &room_id, since)? {
|
.rooms
|
||||||
let (_, pdu) = pdu?;
|
.pdus_until(sender_id, &room_id, since)
|
||||||
if pdu.kind == EventType::RoomMember && pdu.state_key == Some(sender_id.to_string()) {
|
.next()
|
||||||
let content = serde_json::from_value::<
|
.and_then(|pdu| pdu.ok())
|
||||||
Raw<ruma::events::room::member::MemberEventContent>,
|
.and_then(|pdu| {
|
||||||
>(pdu.content.clone())
|
db.rooms
|
||||||
|
.pdu_state_hash(&pdu.0)
|
||||||
|
.ok()?
|
||||||
|
.ok_or_else(|| Error::bad_database("Pdu in db doesn't have a state hash."))
|
||||||
|
.ok()
|
||||||
|
})
|
||||||
|
.and_then(|state_hash| {
|
||||||
|
db.rooms
|
||||||
|
.state_get(&state_hash, &EventType::RoomMember, sender_id.as_str())
|
||||||
|
.ok()?
|
||||||
|
.ok_or_else(|| Error::bad_database("State hash in db doesn't have a state."))
|
||||||
|
.ok()
|
||||||
|
})
|
||||||
|
.and_then(|pdu| {
|
||||||
|
serde_json::from_value::<Raw<ruma::events::room::member::MemberEventContent>>(
|
||||||
|
pdu.content.clone(),
|
||||||
|
)
|
||||||
.expect("Raw::from_value always works")
|
.expect("Raw::from_value always works")
|
||||||
.deserialize()
|
.deserialize()
|
||||||
.map_err(|_| Error::bad_database("Invalid PDU in database."))?;
|
.map_err(|_| Error::bad_database("Invalid PDU in database."))
|
||||||
|
.ok()
|
||||||
|
});
|
||||||
|
|
||||||
if content.membership == MembershipState::Leave {
|
let left_since_last_sync =
|
||||||
left_since_last_sync = true;
|
since_member.map_or(false, |member| member.membership == MembershipState::Join);
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if left_since_last_sync {
|
if left_since_last_sync {
|
||||||
device_list_left.extend(
|
device_list_left.extend(
|
||||||
|
|
|
@ -31,7 +31,7 @@ use std::{
|
||||||
///
|
///
|
||||||
/// This is created when a state group is added to the database by
|
/// This is created when a state group is added to the database by
|
||||||
/// hashing the entire state.
|
/// hashing the entire state.
|
||||||
pub type StateHashId = Vec<u8>;
|
pub type StateHashId = IVec;
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct Rooms {
|
pub struct Rooms {
|
||||||
|
@ -100,7 +100,7 @@ impl StateStore for Rooms {
|
||||||
impl Rooms {
|
impl Rooms {
|
||||||
/// Builds a StateMap by iterating over all keys that start
|
/// Builds a StateMap by iterating over all keys that start
|
||||||
/// with state_hash, this gives the full state for the given state_hash.
|
/// with state_hash, this gives the full state for the given state_hash.
|
||||||
pub fn state_full(&self, state_hash: StateHashId) -> Result<StateMap<EventId>> {
|
pub fn state_full(&self, state_hash: &StateHashId) -> Result<StateMap<PduEvent>> {
|
||||||
self.stateid_pduid
|
self.stateid_pduid
|
||||||
.scan_prefix(&state_hash)
|
.scan_prefix(&state_hash)
|
||||||
.values()
|
.values()
|
||||||
|
@ -115,61 +115,87 @@ impl Rooms {
|
||||||
})
|
})
|
||||||
.map(|pdu| {
|
.map(|pdu| {
|
||||||
let pdu = pdu?;
|
let pdu = pdu?;
|
||||||
Ok(((pdu.kind, pdu.state_key), pdu.event_id))
|
Ok((
|
||||||
|
(
|
||||||
|
pdu.kind.clone(),
|
||||||
|
pdu.state_key
|
||||||
|
.as_ref()
|
||||||
|
.ok_or_else(|| Error::bad_database("State event has no state key."))?
|
||||||
|
.clone(),
|
||||||
|
),
|
||||||
|
pdu,
|
||||||
|
))
|
||||||
})
|
})
|
||||||
.collect::<Result<StateMap<_>>>()
|
.collect::<Result<StateMap<_>>>()
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO make this return Result
|
/// Returns all state entries for this type.
|
||||||
/// Fetches the previous StateHash ID to `current`.
|
pub fn state_type(
|
||||||
pub fn prev_state_hash(&self, current: StateHashId) -> Option<StateHashId> {
|
&self,
|
||||||
let mut found = false;
|
state_hash: &StateHashId,
|
||||||
for pair in self.pduid_statehash.iter().rev() {
|
event_type: &EventType,
|
||||||
let prev = pair.ok()?.1;
|
) -> Result<HashMap<String, PduEvent>> {
|
||||||
if current == prev.as_ref() {
|
let mut prefix = state_hash.to_vec();
|
||||||
found = true;
|
prefix.push(0xff);
|
||||||
}
|
prefix.extend_from_slice(&event_type.to_string().as_bytes());
|
||||||
if current != prev.as_ref() && found {
|
prefix.push(0xff);
|
||||||
return Some(prev.to_vec());
|
|
||||||
}
|
let mut hashmap = HashMap::new();
|
||||||
|
for pdu in self
|
||||||
|
.stateid_pduid
|
||||||
|
.scan_prefix(&prefix)
|
||||||
|
.values()
|
||||||
|
.map(|pdu_id| {
|
||||||
|
Ok::<_, Error>(
|
||||||
|
serde_json::from_slice::<PduEvent>(&self.pduid_pdu.get(pdu_id?)?.ok_or_else(
|
||||||
|
|| Error::bad_database("PDU in state not found in database."),
|
||||||
|
)?)
|
||||||
|
.map_err(|_| Error::bad_database("Invalid PDU bytes in room state."))?,
|
||||||
|
)
|
||||||
|
})
|
||||||
|
{
|
||||||
|
let pdu = pdu?;
|
||||||
|
let state_key = pdu.state_key.clone().ok_or_else(|| {
|
||||||
|
Error::bad_database("Room state contains event without state_key.")
|
||||||
|
})?;
|
||||||
|
hashmap.insert(state_key, pdu);
|
||||||
}
|
}
|
||||||
None
|
Ok(hashmap)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns a single PDU from `room_id` with key (`event_type`, `state_key`).
|
||||||
|
pub fn state_get(
|
||||||
|
&self,
|
||||||
|
state_hash: &StateHashId,
|
||||||
|
event_type: &EventType,
|
||||||
|
state_key: &str,
|
||||||
|
) -> Result<Option<PduEvent>> {
|
||||||
|
let mut key = state_hash.to_vec();
|
||||||
|
key.push(0xff);
|
||||||
|
key.extend_from_slice(&event_type.to_string().as_bytes());
|
||||||
|
key.push(0xff);
|
||||||
|
key.extend_from_slice(&state_key.as_bytes());
|
||||||
|
|
||||||
|
self.stateid_pduid.get(&key)?.map_or(Ok(None), |pdu_id| {
|
||||||
|
Ok::<_, Error>(Some(
|
||||||
|
serde_json::from_slice::<PduEvent>(
|
||||||
|
&self.pduid_pdu.get(pdu_id)?.ok_or_else(|| {
|
||||||
|
Error::bad_database("PDU in state not found in database.")
|
||||||
|
})?,
|
||||||
|
)
|
||||||
|
.map_err(|_| Error::bad_database("Invalid PDU bytes in room state."))?,
|
||||||
|
))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the last state hash key added to the db.
|
||||||
|
pub fn pdu_state_hash(&self, pdu_id: &[u8]) -> Result<Option<StateHashId>> {
|
||||||
|
Ok(self.pduid_statehash.get(pdu_id)?)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns the last state hash key added to the db.
|
/// Returns the last state hash key added to the db.
|
||||||
pub fn current_state_hash(&self, room_id: &RoomId) -> Result<Option<StateHashId>> {
|
pub fn current_state_hash(&self, room_id: &RoomId) -> Result<Option<StateHashId>> {
|
||||||
Ok(self
|
Ok(self.roomid_statehash.get(room_id.as_bytes())?)
|
||||||
.roomid_statehash
|
|
||||||
.get(room_id.as_bytes())?
|
|
||||||
.map(|bytes| bytes.to_vec()))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// This fetches auth event_ids from the current state using the
|
|
||||||
/// full `roomstateid_pdu` tree.
|
|
||||||
pub fn get_auth_event_ids(
|
|
||||||
&self,
|
|
||||||
room_id: &RoomId,
|
|
||||||
kind: &EventType,
|
|
||||||
sender: &UserId,
|
|
||||||
state_key: Option<&str>,
|
|
||||||
content: serde_json::Value,
|
|
||||||
) -> Result<Vec<EventId>> {
|
|
||||||
let auth_events = state_res::auth_types_for_event(
|
|
||||||
kind.clone(),
|
|
||||||
sender,
|
|
||||||
state_key.map(|s| s.to_string()),
|
|
||||||
content,
|
|
||||||
);
|
|
||||||
|
|
||||||
let mut events = vec![];
|
|
||||||
for (event_type, state_key) in auth_events {
|
|
||||||
if let Some(state_key) = state_key.as_ref() {
|
|
||||||
if let Some(id) = self.room_state_get(room_id, &event_type, state_key)? {
|
|
||||||
events.push(id.event_id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(events)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// This fetches auth events from the current state.
|
/// This fetches auth events from the current state.
|
||||||
|
@ -190,10 +216,8 @@ impl Rooms {
|
||||||
|
|
||||||
let mut events = StateMap::new();
|
let mut events = StateMap::new();
|
||||||
for (event_type, state_key) in auth_events {
|
for (event_type, state_key) in auth_events {
|
||||||
if let Some(s_key) = state_key.as_ref() {
|
if let Some(pdu) = self.room_state_get(room_id, &event_type, &state_key)? {
|
||||||
if let Some(pdu) = self.room_state_get(room_id, &event_type, s_key)? {
|
events.insert((event_type, state_key), pdu);
|
||||||
events.insert((event_type, state_key), pdu);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(events)
|
Ok(events)
|
||||||
|
@ -206,7 +230,7 @@ impl Rooms {
|
||||||
// We only hash the pdu's event ids, not the whole pdu
|
// We only hash the pdu's event ids, not the whole pdu
|
||||||
let bytes = pdu_id_bytes.join(&0xff);
|
let bytes = pdu_id_bytes.join(&0xff);
|
||||||
let hash = digest::digest(&digest::SHA256, &bytes);
|
let hash = digest::digest(&digest::SHA256, &bytes);
|
||||||
Ok(hash.as_ref().to_vec())
|
Ok(hash.as_ref().into())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Checks if a room exists.
|
/// Checks if a room exists.
|
||||||
|
@ -230,7 +254,7 @@ impl Rooms {
|
||||||
) -> Result<()> {
|
) -> Result<()> {
|
||||||
let state_hash =
|
let state_hash =
|
||||||
self.calculate_hash(&state.values().map(|pdu_id| &**pdu_id).collect::<Vec<_>>())?;
|
self.calculate_hash(&state.values().map(|pdu_id| &**pdu_id).collect::<Vec<_>>())?;
|
||||||
let mut prefix = state_hash.clone();
|
let mut prefix = state_hash.to_vec();
|
||||||
prefix.push(0xff);
|
prefix.push(0xff);
|
||||||
|
|
||||||
for ((event_type, state_key), pdu_id) in state {
|
for ((event_type, state_key), pdu_id) in state {
|
||||||
|
@ -248,41 +272,11 @@ impl Rooms {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns the full room state.
|
/// Returns the full room state.
|
||||||
pub fn room_state_full(
|
pub fn room_state_full(&self, room_id: &RoomId) -> Result<StateMap<PduEvent>> {
|
||||||
&self,
|
|
||||||
room_id: &RoomId,
|
|
||||||
) -> Result<HashMap<(EventType, String), PduEvent>> {
|
|
||||||
if let Some(current_state_hash) = self.current_state_hash(room_id)? {
|
if let Some(current_state_hash) = self.current_state_hash(room_id)? {
|
||||||
let mut prefix = current_state_hash;
|
self.state_full(¤t_state_hash)
|
||||||
prefix.push(0xff);
|
|
||||||
|
|
||||||
let mut hashmap = HashMap::new();
|
|
||||||
for pdu in self
|
|
||||||
.stateid_pduid
|
|
||||||
.scan_prefix(prefix)
|
|
||||||
.values()
|
|
||||||
.map(|pdu_id| {
|
|
||||||
Ok::<_, Error>(
|
|
||||||
serde_json::from_slice::<PduEvent>(
|
|
||||||
&self.pduid_pdu.get(pdu_id?)?.ok_or_else(|| {
|
|
||||||
Error::bad_database("PDU in state not found in database.")
|
|
||||||
})?,
|
|
||||||
)
|
|
||||||
.map_err(|_| {
|
|
||||||
Error::bad_database("Invalid PDU bytes in current room state.")
|
|
||||||
})?,
|
|
||||||
)
|
|
||||||
})
|
|
||||||
{
|
|
||||||
let pdu = pdu?;
|
|
||||||
let state_key = pdu.state_key.clone().ok_or_else(|| {
|
|
||||||
Error::bad_database("Room state contains event without state_key.")
|
|
||||||
})?;
|
|
||||||
hashmap.insert((pdu.kind.clone(), state_key), pdu);
|
|
||||||
}
|
|
||||||
Ok(hashmap)
|
|
||||||
} else {
|
} else {
|
||||||
Ok(HashMap::new())
|
Ok(BTreeMap::new())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -293,36 +287,7 @@ impl Rooms {
|
||||||
event_type: &EventType,
|
event_type: &EventType,
|
||||||
) -> Result<HashMap<String, PduEvent>> {
|
) -> Result<HashMap<String, PduEvent>> {
|
||||||
if let Some(current_state_hash) = self.current_state_hash(room_id)? {
|
if let Some(current_state_hash) = self.current_state_hash(room_id)? {
|
||||||
let mut prefix = current_state_hash;
|
self.state_type(¤t_state_hash, event_type)
|
||||||
prefix.push(0xff);
|
|
||||||
prefix.extend_from_slice(&event_type.to_string().as_bytes());
|
|
||||||
prefix.push(0xff);
|
|
||||||
|
|
||||||
let mut hashmap = HashMap::new();
|
|
||||||
for pdu in self
|
|
||||||
.stateid_pduid
|
|
||||||
.scan_prefix(&prefix)
|
|
||||||
.values()
|
|
||||||
.map(|pdu_id| {
|
|
||||||
Ok::<_, Error>(
|
|
||||||
serde_json::from_slice::<PduEvent>(
|
|
||||||
&self.pduid_pdu.get(pdu_id?)?.ok_or_else(|| {
|
|
||||||
Error::bad_database("PDU in state not found in database.")
|
|
||||||
})?,
|
|
||||||
)
|
|
||||||
.map_err(|_| {
|
|
||||||
Error::bad_database("Invalid PDU bytes in current room state.")
|
|
||||||
})?,
|
|
||||||
)
|
|
||||||
})
|
|
||||||
{
|
|
||||||
let pdu = pdu?;
|
|
||||||
let state_key = pdu.state_key.clone().ok_or_else(|| {
|
|
||||||
Error::bad_database("Room state contains event without state_key.")
|
|
||||||
})?;
|
|
||||||
hashmap.insert(state_key, pdu);
|
|
||||||
}
|
|
||||||
Ok(hashmap)
|
|
||||||
} else {
|
} else {
|
||||||
Ok(HashMap::new())
|
Ok(HashMap::new())
|
||||||
}
|
}
|
||||||
|
@ -336,20 +301,7 @@ impl Rooms {
|
||||||
state_key: &str,
|
state_key: &str,
|
||||||
) -> Result<Option<PduEvent>> {
|
) -> Result<Option<PduEvent>> {
|
||||||
if let Some(current_state_hash) = self.current_state_hash(room_id)? {
|
if let Some(current_state_hash) = self.current_state_hash(room_id)? {
|
||||||
let mut key = current_state_hash;
|
self.state_get(¤t_state_hash, event_type, state_key)
|
||||||
key.push(0xff);
|
|
||||||
key.extend_from_slice(&event_type.to_string().as_bytes());
|
|
||||||
key.push(0xff);
|
|
||||||
key.extend_from_slice(&state_key.as_bytes());
|
|
||||||
|
|
||||||
self.stateid_pduid.get(&key)?.map_or(Ok(None), |pdu_id| {
|
|
||||||
Ok::<_, Error>(Some(
|
|
||||||
serde_json::from_slice::<PduEvent>(&self.pduid_pdu.get(pdu_id)?.ok_or_else(
|
|
||||||
|| Error::bad_database("PDU in state not found in database."),
|
|
||||||
)?)
|
|
||||||
.map_err(|_| Error::bad_database("Invalid PDU bytes in current room state."))?,
|
|
||||||
))
|
|
||||||
})
|
|
||||||
} else {
|
} else {
|
||||||
Ok(None)
|
Ok(None)
|
||||||
}
|
}
|
||||||
|
@ -562,14 +514,15 @@ impl Rooms {
|
||||||
/// This adds all current state events (not including the incoming event)
|
/// This adds all current state events (not including the incoming event)
|
||||||
/// to `stateid_pduid` and adds the incoming event to `pduid_statehash`.
|
/// to `stateid_pduid` and adds the incoming event to `pduid_statehash`.
|
||||||
/// The incoming event is the `pdu_id` passed to this method.
|
/// The incoming event is the `pdu_id` passed to this method.
|
||||||
fn append_to_state(&self, new_pdu_id: &[u8], new_pdu: &PduEvent) -> Result<StateHashId> {
|
pub fn append_to_state(&self, new_pdu_id: &[u8], new_pdu: &PduEvent) -> Result<StateHashId> {
|
||||||
let old_state =
|
let old_state =
|
||||||
if let Some(old_state_hash) = self.roomid_statehash.get(new_pdu.room_id.as_bytes())? {
|
if let Some(old_state_hash) = self.roomid_statehash.get(new_pdu.room_id.as_bytes())? {
|
||||||
// Store state for event. The state does not include the event itself.
|
// Store state for event. The state does not include the event itself.
|
||||||
// Instead it's the state before the pdu, so the room's old state.
|
// Instead it's the state before the pdu, so the room's old state.
|
||||||
self.pduid_statehash.insert(new_pdu_id, &old_state_hash)?;
|
self.pduid_statehash
|
||||||
|
.insert(dbg!(new_pdu_id), &old_state_hash)?;
|
||||||
if new_pdu.state_key.is_none() {
|
if new_pdu.state_key.is_none() {
|
||||||
return Ok(old_state_hash.to_vec());
|
return Ok(old_state_hash);
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut prefix = old_state_hash.to_vec();
|
let mut prefix = old_state_hash.to_vec();
|
||||||
|
@ -841,9 +794,7 @@ impl Rooms {
|
||||||
|
|
||||||
let pdu_id = self.append_pdu(&pdu, &pdu_json, globals, account_data)?;
|
let pdu_id = self.append_pdu(&pdu, &pdu_json, globals, account_data)?;
|
||||||
|
|
||||||
if pdu.state_key.is_some() {
|
self.append_to_state(&pdu_id, &pdu)?;
|
||||||
self.append_to_state(&pdu_id, &pdu)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
for server in self
|
for server in self
|
||||||
.room_servers(room_id)
|
.room_servers(room_id)
|
||||||
|
@ -905,7 +856,7 @@ impl Rooms {
|
||||||
user_id: &UserId,
|
user_id: &UserId,
|
||||||
room_id: &RoomId,
|
room_id: &RoomId,
|
||||||
until: u64,
|
until: u64,
|
||||||
) -> impl Iterator<Item = Result<(u64, PduEvent)>> {
|
) -> impl Iterator<Item = Result<(IVec, PduEvent)>> {
|
||||||
// Create the first part of the full pdu id
|
// Create the first part of the full pdu id
|
||||||
let mut prefix = room_id.to_string().as_bytes().to_vec();
|
let mut prefix = room_id.to_string().as_bytes().to_vec();
|
||||||
prefix.push(0xff);
|
prefix.push(0xff);
|
||||||
|
@ -916,23 +867,18 @@ impl Rooms {
|
||||||
let current: &[u8] = ¤t;
|
let current: &[u8] = ¤t;
|
||||||
|
|
||||||
let user_id = user_id.clone();
|
let user_id = user_id.clone();
|
||||||
let prefixlen = prefix.len();
|
|
||||||
self.pduid_pdu
|
self.pduid_pdu
|
||||||
.range(..current)
|
.range(..current)
|
||||||
.rev()
|
.rev()
|
||||||
.filter_map(|r| r.ok())
|
.filter_map(|r| r.ok())
|
||||||
.take_while(move |(k, _)| k.starts_with(&prefix))
|
.take_while(move |(k, _)| k.starts_with(&prefix))
|
||||||
.map(move |(k, v)| {
|
.map(move |(pdu_id, v)| {
|
||||||
let mut pdu = serde_json::from_slice::<PduEvent>(&v)
|
let mut pdu = serde_json::from_slice::<PduEvent>(&v)
|
||||||
.map_err(|_| Error::bad_database("PDU in db is invalid."))?;
|
.map_err(|_| Error::bad_database("PDU in db is invalid."))?;
|
||||||
if pdu.sender != user_id {
|
if pdu.sender != user_id {
|
||||||
pdu.unsigned.remove("transaction_id");
|
pdu.unsigned.remove("transaction_id");
|
||||||
}
|
}
|
||||||
Ok((
|
Ok((pdu_id, pdu))
|
||||||
utils::u64_from_bytes(&k[prefixlen..])
|
|
||||||
.map_err(|_| Error::bad_database("Invalid pdu id in db."))?,
|
|
||||||
pdu,
|
|
||||||
))
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -943,7 +889,7 @@ impl Rooms {
|
||||||
user_id: &UserId,
|
user_id: &UserId,
|
||||||
room_id: &RoomId,
|
room_id: &RoomId,
|
||||||
from: u64,
|
from: u64,
|
||||||
) -> impl Iterator<Item = Result<(u64, PduEvent)>> {
|
) -> impl Iterator<Item = Result<(IVec, PduEvent)>> {
|
||||||
// Create the first part of the full pdu id
|
// Create the first part of the full pdu id
|
||||||
let mut prefix = room_id.to_string().as_bytes().to_vec();
|
let mut prefix = room_id.to_string().as_bytes().to_vec();
|
||||||
prefix.push(0xff);
|
prefix.push(0xff);
|
||||||
|
@ -954,22 +900,17 @@ impl Rooms {
|
||||||
let current: &[u8] = ¤t;
|
let current: &[u8] = ¤t;
|
||||||
|
|
||||||
let user_id = user_id.clone();
|
let user_id = user_id.clone();
|
||||||
let prefixlen = prefix.len();
|
|
||||||
self.pduid_pdu
|
self.pduid_pdu
|
||||||
.range(current..)
|
.range(current..)
|
||||||
.filter_map(|r| r.ok())
|
.filter_map(|r| r.ok())
|
||||||
.take_while(move |(k, _)| k.starts_with(&prefix))
|
.take_while(move |(k, _)| k.starts_with(&prefix))
|
||||||
.map(move |(k, v)| {
|
.map(move |(pdu_id, v)| {
|
||||||
let mut pdu = serde_json::from_slice::<PduEvent>(&v)
|
let mut pdu = serde_json::from_slice::<PduEvent>(&v)
|
||||||
.map_err(|_| Error::bad_database("PDU in db is invalid."))?;
|
.map_err(|_| Error::bad_database("PDU in db is invalid."))?;
|
||||||
if pdu.sender != user_id {
|
if pdu.sender != user_id {
|
||||||
pdu.unsigned.remove("transaction_id");
|
pdu.unsigned.remove("transaction_id");
|
||||||
}
|
}
|
||||||
Ok((
|
Ok((pdu_id, pdu))
|
||||||
utils::u64_from_bytes(&k[prefixlen..])
|
|
||||||
.map_err(|_| Error::bad_database("Invalid pdu id in db."))?,
|
|
||||||
pdu,
|
|
||||||
))
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -123,6 +123,7 @@ fn setup_rocket() -> rocket::Rocket {
|
||||||
server_server::get_server_keys,
|
server_server::get_server_keys,
|
||||||
server_server::get_server_keys_deprecated,
|
server_server::get_server_keys_deprecated,
|
||||||
server_server::get_public_rooms_route,
|
server_server::get_public_rooms_route,
|
||||||
|
server_server::get_public_rooms_filtered_route,
|
||||||
server_server::send_transaction_message_route,
|
server_server::send_transaction_message_route,
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
use crate::{Error, Result};
|
use crate::Error;
|
||||||
use js_int::UInt;
|
use js_int::UInt;
|
||||||
use ruma::{
|
use ruma::{
|
||||||
events::pdu::PduStub,
|
events::pdu::PduStub,
|
||||||
|
@ -35,7 +35,7 @@ pub struct PduEvent {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl PduEvent {
|
impl PduEvent {
|
||||||
pub fn redact(&mut self, reason: &PduEvent) -> Result<()> {
|
pub fn redact(&mut self, reason: &PduEvent) -> crate::Result<()> {
|
||||||
self.unsigned.clear();
|
self.unsigned.clear();
|
||||||
|
|
||||||
let allowed: &[&str] = match self.kind {
|
let allowed: &[&str] = match self.kind {
|
||||||
|
@ -244,7 +244,7 @@ impl From<&state_res::StateEvent> for PduEvent {
|
||||||
.expect("time is valid"),
|
.expect("time is valid"),
|
||||||
kind: pdu.kind(),
|
kind: pdu.kind(),
|
||||||
content: pdu.content().clone(),
|
content: pdu.content().clone(),
|
||||||
state_key: pdu.state_key(),
|
state_key: Some(pdu.state_key()),
|
||||||
prev_events: pdu.prev_event_ids(),
|
prev_events: pdu.prev_event_ids(),
|
||||||
depth: pdu.depth().clone(),
|
depth: pdu.depth().clone(),
|
||||||
auth_events: pdu.auth_events(),
|
auth_events: pdu.auth_events(),
|
||||||
|
|
|
@ -329,8 +329,10 @@ pub fn send_transaction_message_route<'a>(
|
||||||
let pdu = serde_json::from_value::<PduEvent>(value.clone())
|
let pdu = serde_json::from_value::<PduEvent>(value.clone())
|
||||||
.expect("all ruma pdus are conduit pdus");
|
.expect("all ruma pdus are conduit pdus");
|
||||||
if db.rooms.exists(&pdu.room_id)? {
|
if db.rooms.exists(&pdu.room_id)? {
|
||||||
db.rooms
|
let pdu_id = db
|
||||||
|
.rooms
|
||||||
.append_pdu(&pdu, &value, &db.globals, &db.account_data)?;
|
.append_pdu(&pdu, &value, &db.globals, &db.account_data)?;
|
||||||
|
db.rooms.append_to_state(&pdu_id, &pdu)?;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(send_transaction_message::v1::Response {
|
Ok(send_transaction_message::v1::Response {
|
||||||
|
|
|
@ -1,59 +0,0 @@
|
||||||
use std::collections::HashMap;
|
|
||||||
|
|
||||||
fn stateres(state_a: HashMap<StateTuple, PduEvent>, state_b: HashMap<StateTuple, PduEvent>) {
|
|
||||||
let mut unconflicted = todo!("state at fork event");
|
|
||||||
|
|
||||||
let mut conflicted: HashMap<StateTuple, PduEvent> = state_a
|
|
||||||
.iter()
|
|
||||||
.filter(|(key_a, value_a)| match state_b.remove(key_a) {
|
|
||||||
Some(value_b) if value_a == value_b => unconflicted.insert(key_a, value_a),
|
|
||||||
_ => false,
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
// We removed unconflicted from state_b, now we can easily insert all events that are only in fork b
|
|
||||||
conflicted.extend(state_b);
|
|
||||||
|
|
||||||
let partial_state = unconflicted.clone();
|
|
||||||
|
|
||||||
let full_conflicted = conflicted.clone(); // TODO: auth events
|
|
||||||
|
|
||||||
let output_rev = Vec::new();
|
|
||||||
let event_map = HashMap::new();
|
|
||||||
let incoming_edges = HashMap::new();
|
|
||||||
|
|
||||||
for event in full_conflicted {
|
|
||||||
event_map.insert(event.event_id, event);
|
|
||||||
incoming_edges.insert(event.event_id, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
for e in conflicted_control_events {
|
|
||||||
for a in e.auth_events {
|
|
||||||
incoming_edges[a.event_id] += 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
while incoming_edges.len() > 0 {
|
|
||||||
let mut count_0 = incoming_edges
|
|
||||||
.iter()
|
|
||||||
.filter(|(_, c)| c == 0)
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
|
|
||||||
count_0.sort_by(|(x, _), (y, _)| {
|
|
||||||
x.power_level
|
|
||||||
.cmp(&a.power_level)
|
|
||||||
.then_with(|| x.origin_server.ts.cmp(&y.origin_server_ts))
|
|
||||||
.then_with(|| x.event_id.cmp(&y.event_id))
|
|
||||||
});
|
|
||||||
|
|
||||||
for (id, count) in count_0 {
|
|
||||||
output_rev.push(event_map[id]);
|
|
||||||
|
|
||||||
for auth_event in event_map[id].auth_events {
|
|
||||||
incoming_edges[auth_event.event_id] -= 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
incoming_edges.remove(id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Reference in New Issue