mirror of
https://github.com/element-hq/synapse.git
synced 2026-05-27 03:04:22 +00:00
ff55aff5b2
When we port the `Event` class to Rust, the constructor will check for the existence of required fields. To support that, we tidy up the test code where we construct fake events to add all the required fields. There should be no behavioural changes. Review commit-by-commit.
373 lines
14 KiB
Python
373 lines
14 KiB
Python
#
|
|
# This file is licensed under the Affero General Public License (AGPL) version 3.
|
|
#
|
|
# Copyright (C) 2026 Element Creations, Ltd
|
|
#
|
|
# This program is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU Affero General Public License as
|
|
# published by the Free Software Foundation, either version 3 of the
|
|
# License, or (at your option) any later version.
|
|
#
|
|
# See the GNU Affero General Public License for more details:
|
|
# <https://www.gnu.org/licenses/agpl-3.0.html>.
|
|
|
|
from typing import Iterable
|
|
from unittest.mock import Mock
|
|
|
|
from twisted.test.proto_helpers import MemoryReactor
|
|
|
|
from synapse.api.constants import EventTypes
|
|
from synapse.api.errors import SynapseError
|
|
from synapse.api.room_versions import RoomVersions
|
|
from synapse.events.py_protocol import MSC4242Event, supports_msc4242_state_dag
|
|
from synapse.events.snapshot import EventContext
|
|
from synapse.rest.client import room
|
|
from synapse.server import HomeServer
|
|
from synapse.util.clock import Clock
|
|
|
|
from tests.test_utils.event_builders import make_test_event
|
|
from tests.unittest import HomeserverTestCase, override_config
|
|
|
|
|
|
class MSC4242StateDagsTests(HomeserverTestCase):
|
|
user_id = "@user1:server"
|
|
servlets = [room.register_servlets]
|
|
|
|
def make_homeserver(self, reactor: MemoryReactor, clock: Clock) -> HomeServer:
|
|
hs = self.setup_test_homeserver("server")
|
|
return hs
|
|
|
|
def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None:
|
|
self.room_id = self.helper.create_room_as(
|
|
self.user_id,
|
|
room_version=RoomVersions.MSC4242v12.identifier,
|
|
)
|
|
|
|
self.store = hs.get_datastores().main
|
|
self._storage_controllers = self.hs.get_storage_controllers()
|
|
|
|
def _get_prev_state_events(self, event_id: str) -> list[str]:
|
|
ev = self.helper.get_event(self.room_id, event_id)
|
|
prev_state_events: list[str] | None = ev.get("prev_state_events", None)
|
|
assert prev_state_events is not None
|
|
return prev_state_events
|
|
|
|
@override_config({"experimental_features": {"msc4242_enabled": True}})
|
|
def test_forward_extremities_are_calculated(self) -> None:
|
|
"""
|
|
Check that forward extremities are set as prev_state_events and that they don't change
|
|
for non-state events.
|
|
"""
|
|
# they don't change for messages
|
|
first_event_id = self.helper.send(self.room_id, body="test1")["event_id"]
|
|
first_prev_state_events = self._get_prev_state_events(first_event_id)
|
|
assert len(first_prev_state_events) == 1
|
|
second_id = self.helper.send(self.room_id, body="test2")["event_id"]
|
|
second_prev_state_events = self._get_prev_state_events(second_id)
|
|
assert len(second_prev_state_events) == 1
|
|
self.assertIncludes(
|
|
set(first_prev_state_events), set(second_prev_state_events), exact=True
|
|
)
|
|
|
|
# send an auth event, which should change the prev_state_events on *subsequent* events
|
|
join_rule_state_event_id = self.helper.send_state(
|
|
self.room_id,
|
|
EventTypes.JoinRules,
|
|
{
|
|
"join_rule": "knock",
|
|
},
|
|
tok="nope",
|
|
)["event_id"]
|
|
join_rule_prev_state_event_ids = self._get_prev_state_events(
|
|
join_rule_state_event_id
|
|
)
|
|
self.assertIncludes(
|
|
set(second_prev_state_events),
|
|
set(join_rule_prev_state_event_ids),
|
|
exact=True,
|
|
)
|
|
|
|
# prev_state_events should always point to the join rule now
|
|
third_event_id = self.helper.send(self.room_id, body="test3")["event_id"]
|
|
third_prev_state_events = self._get_prev_state_events(third_event_id)
|
|
self.assertIncludes(
|
|
set(third_prev_state_events), {join_rule_state_event_id}, exact=True
|
|
)
|
|
# and non-auth state should also update prev_state_events
|
|
name_state_event_id = self.helper.send_state(
|
|
self.room_id,
|
|
EventTypes.Name,
|
|
{
|
|
"name": "State DAGs!",
|
|
},
|
|
tok="nope",
|
|
)["event_id"]
|
|
name_prev_state_event_ids = self._get_prev_state_events(name_state_event_id)
|
|
self.assertIncludes(
|
|
set(name_prev_state_event_ids), {join_rule_state_event_id}, exact=True
|
|
)
|
|
fourth_event_id = self.helper.send(self.room_id, body="test4")["event_id"]
|
|
fourth_prev_state_events = self._get_prev_state_events(fourth_event_id)
|
|
self.assertIncludes(
|
|
set(fourth_prev_state_events), {name_state_event_id}, exact=True
|
|
)
|
|
|
|
|
|
class MSC4242EventPersistenceStateDagsStoreTestCase(HomeserverTestCase):
|
|
servlets = [
|
|
room.register_servlets,
|
|
]
|
|
|
|
def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None:
|
|
self.store = hs.get_datastores().main
|
|
persistence = hs.get_storage_controllers().persistence
|
|
assert persistence is not None
|
|
self.persistence = persistence
|
|
self.room_id = "!foo:bar"
|
|
self.seen_event_ids: set[str] = set()
|
|
self.persistence.main_store = Mock(spec=["have_seen_events"])
|
|
self.persistence.main_store.have_seen_events.side_effect = (
|
|
self._have_seen_events
|
|
)
|
|
self.rejected_event_ids_and_their_prevs: set[str] = set()
|
|
self.persistence.persist_events_store = Mock(
|
|
spec=["_get_prevs_before_rejected"]
|
|
)
|
|
self.persistence.persist_events_store._get_prevs_before_rejected.side_effect = (
|
|
self._get_prevs_before_rejected
|
|
)
|
|
|
|
async def _have_seen_events(
|
|
self, room_id: str, event_ids: Iterable[str]
|
|
) -> set[str]:
|
|
unknown_events = set(event_ids)
|
|
return self.seen_event_ids.intersection(unknown_events)
|
|
|
|
async def _get_prevs_before_rejected(
|
|
self, event_ids: Iterable[str], include_soft_failed: bool = True
|
|
) -> set[str]:
|
|
return self.rejected_event_ids_and_their_prevs
|
|
|
|
def _make_event(
|
|
self,
|
|
id: str,
|
|
prev_state_events: list[str],
|
|
rejected: bool = False,
|
|
) -> tuple[MSC4242Event, EventContext]:
|
|
ev = make_test_event(
|
|
{
|
|
"prev_state_events": prev_state_events,
|
|
"content": {
|
|
"membership": "join",
|
|
},
|
|
"sender": "@unimportant:info",
|
|
"state_key": "@unimportant:info",
|
|
"type": "m.room.member",
|
|
"room_id": self.room_id,
|
|
},
|
|
room_version=RoomVersions.MSC4242v12,
|
|
)
|
|
ev._event_id = id # type: ignore[attr-defined]
|
|
assert supports_msc4242_state_dag(ev)
|
|
ctx = Mock()
|
|
ctx.rejected = rejected
|
|
return ev, ctx
|
|
|
|
def _test(
|
|
self,
|
|
current_fwds: list[str],
|
|
new_events: list[tuple[MSC4242Event, EventContext]],
|
|
want_new_extrems: set[str],
|
|
want_raises: bool = False,
|
|
) -> None:
|
|
"""
|
|
Tests the logic of _calculate_new_state_dag_extremities.
|
|
|
|
Tests that the new extremities calculated as a result of processing current_fwds and new_events
|
|
matches want_new_extrems or raises if want_raises is True.
|
|
"""
|
|
coroutine = self.persistence._calculate_new_state_dag_extremities(
|
|
self.room_id,
|
|
frozenset(current_fwds),
|
|
new_events,
|
|
)
|
|
if want_raises:
|
|
f = self.get_failure(coroutine, SynapseError)
|
|
assert f is not None
|
|
return
|
|
|
|
new_extrems = set(self.get_success(coroutine))
|
|
self.assertIncludes(
|
|
new_extrems,
|
|
set(want_new_extrems),
|
|
exact=True,
|
|
message=f"want_new_extrems={want_new_extrems} got={new_extrems}",
|
|
)
|
|
|
|
@override_config({"experimental_features": {"msc4242_enabled": True}})
|
|
def test_calculate_new_state_dag_extremities_simple(self) -> None:
|
|
# Simple linear chain
|
|
self._test(
|
|
current_fwds=[],
|
|
new_events=[
|
|
self._make_event("$1", []),
|
|
self._make_event("$2", ["$1"]),
|
|
self._make_event("$3", ["$2"]),
|
|
self._make_event("$4", ["$3"]),
|
|
],
|
|
want_new_extrems={"$4"},
|
|
)
|
|
|
|
@override_config({"experimental_features": {"msc4242_enabled": True}})
|
|
def test_calculate_new_state_dag_extremities_fork(self) -> None:
|
|
# Simple fork so we end up with two forward extrems
|
|
self._test(
|
|
current_fwds=[],
|
|
new_events=[
|
|
self._make_event("$1", []),
|
|
self._make_event("$2", ["$1"]),
|
|
self._make_event("$3", ["$2"]),
|
|
self._make_event("$4", ["$2"]),
|
|
],
|
|
want_new_extrems={"$3", "$4"},
|
|
)
|
|
|
|
@override_config({"experimental_features": {"msc4242_enabled": True}})
|
|
def test_calculate_new_state_dag_extremities_merge(self) -> None:
|
|
# Simple fork so we end up with two forward extrems
|
|
self._test(
|
|
current_fwds=[],
|
|
new_events=[
|
|
self._make_event("$1", []),
|
|
self._make_event("$2", ["$1"]),
|
|
self._make_event("$3", ["$1"]),
|
|
self._make_event("$4", ["$2", "$3"]),
|
|
],
|
|
want_new_extrems={"$4"},
|
|
)
|
|
|
|
@override_config({"experimental_features": {"msc4242_enabled": True}})
|
|
def test_calculate_new_state_dag_extremities_fork_on_existing(self) -> None:
|
|
# Fork where we are adding to older events
|
|
self.seen_event_ids = {"$1", "$2", "$3"}
|
|
self._test(
|
|
current_fwds=["$3"],
|
|
new_events=[
|
|
self._make_event("$4", ["$3"]), # append to the forward extrem
|
|
self._make_event("$5", ["$1"]), # append to the root
|
|
],
|
|
want_new_extrems={"$4", "$5"},
|
|
)
|
|
|
|
@override_config({"experimental_features": {"msc4242_enabled": True}})
|
|
def test_calculate_new_state_dag_extremities_merge_on_existing(self) -> None:
|
|
# Merge where we are merging to older events
|
|
self.seen_event_ids = {"$1", "$2", "$3"}
|
|
self._test(
|
|
current_fwds=["$3"],
|
|
new_events=[
|
|
self._make_event("$4", ["$3", "$2"]),
|
|
],
|
|
want_new_extrems={"$4"},
|
|
)
|
|
|
|
@override_config({"experimental_features": {"msc4242_enabled": True}})
|
|
def test_calculate_new_state_dag_extremities_merge_on_not_current(self) -> None:
|
|
# Merge where we are merging to older events
|
|
self.seen_event_ids = {"$1", "$2", "$3"}
|
|
self._test(
|
|
current_fwds=["$3"],
|
|
new_events=[
|
|
self._make_event("$4", ["$1", "$2"]),
|
|
],
|
|
want_new_extrems={"$3", "$4"},
|
|
)
|
|
|
|
@override_config({"experimental_features": {"msc4242_enabled": True}})
|
|
def test_calculate_new_state_dag_extremities_append_with_rejected(self) -> None:
|
|
# rejected events cannot be forward extremities
|
|
self.seen_event_ids = {"$1", "$2", "$3"}
|
|
self._test(
|
|
current_fwds=["$3"],
|
|
new_events=[
|
|
self._make_event("$4", ["$3"], rejected=True),
|
|
],
|
|
want_new_extrems={"$3"},
|
|
)
|
|
|
|
self._test(
|
|
current_fwds=["$3"],
|
|
new_events=[
|
|
self._make_event("$4", ["$3"], rejected=True),
|
|
self._make_event("$5", ["$4"], rejected=True),
|
|
],
|
|
want_new_extrems={"$3"},
|
|
)
|
|
|
|
@override_config({"experimental_features": {"msc4242_enabled": True}})
|
|
def test_calculate_new_state_dag_extremities_append_with_rejected_in_chain(
|
|
self,
|
|
) -> None:
|
|
# rejected events cannot be forward extremities, but events that come after them can.
|
|
# this shouldn't cause multiple forward extremities.
|
|
self.seen_event_ids = {"$1", "$2", "$3"}
|
|
self.rejected_event_ids_and_their_prevs = {"$4", "$3"}
|
|
self._test(
|
|
current_fwds=["$3"],
|
|
new_events=[
|
|
self._make_event("$4", ["$3"], rejected=True),
|
|
self._make_event("$5", ["$4"]),
|
|
],
|
|
want_new_extrems={"$5"},
|
|
)
|
|
|
|
@override_config({"experimental_features": {"msc4242_enabled": True}})
|
|
def test_calculate_new_state_dag_extremities_missing_prevs_raises(self) -> None:
|
|
self._test(
|
|
current_fwds=[],
|
|
new_events=[
|
|
self._make_event("$1", []),
|
|
self._make_event("$2", ["$1"]),
|
|
self._make_event("$3", ["$unknown"]),
|
|
self._make_event("$4", ["$3"]),
|
|
],
|
|
want_new_extrems={"$4"},
|
|
want_raises=True,
|
|
)
|
|
|
|
@override_config({"experimental_features": {"msc4242_enabled": True}})
|
|
def test_calculate_new_state_dag_extremities_complex(self) -> None:
|
|
"""
|
|
1
|
|
| \
|
|
2 4
|
|
|
|
|
3
|
|
|
|
Exists already, then becomes...
|
|
|
|
1______
|
|
| \\ |
|
|
2 4 5R
|
|
| | |
|
|
3--7 6R
|
|
| \\ / \
|
|
10R 8 9
|
|
|
|
"""
|
|
# Merge where we are merging to older events
|
|
self.seen_event_ids = {"$1", "$2", "$3", "$4"}
|
|
self.rejected_event_ids_and_their_prevs = {"$1", "$5", "$6", "$3", "$10"}
|
|
self._test(
|
|
current_fwds=["$3", "$4"],
|
|
new_events=[
|
|
self._make_event("$5", ["$1"], rejected=True),
|
|
self._make_event("$6", ["$5"], rejected=True),
|
|
self._make_event("$7", ["$4", "$3"]),
|
|
self._make_event("$8", ["$6", "$7"]),
|
|
self._make_event("$9", ["$6"]),
|
|
self._make_event("$10", ["$3"], rejected=True),
|
|
],
|
|
want_new_extrems={"$8", "$9"},
|
|
)
|