mirror of
https://github.com/element-hq/synapse.git
synced 2024-12-14 11:57:44 +00:00
Add rooms.bump_stamp
to Sliding Sync /sync
for easier client-side sorting (#17395)
`bump_stamp` corresponds to the `stream_ordering` of the latest `DEFAULT_BUMP_EVENT_TYPES` in the room. This helps clients sort more readily without them needing to pull in a bunch of the timeline to determine the last activity. `bump_event_types` is a thing because for example, we don't want display name changes to mark the room as unread and bump it to the top. For encrypted rooms, we just have to consider any activity as a bump because we can't see the content and the client has to figure it out for themselves. Outside of Synapse, `bump_stamp` is just a free-form counter so other implementations could use `received_ts`or `origin_server_ts` (see the [*Security considerations* section in MSC3575 about the potential pitfalls of using `origin_server_ts`](https://github.com/matrix-org/matrix-spec-proposals/blob/kegan/sync-v3/proposals/3575-sync.md#security-considerations)). It doesn't have any guarantee about always going up. In the Synapse case, it could go down if an event was redacted/removed (or purged in cases of retention policies). In the future, we could add `bump_event_types` as [MSC3575](https://github.com/matrix-org/matrix-spec-proposals/pull/3575) mentions if people need to customize the event types. --- In the Sliding Sync proxy, a similar [`timestamp` field was added](https://github.com/matrix-org/sliding-sync/pull/247) for the same purpose but the name is not obvious what it pertains to or what it's for. The `timestamp` field was also added to Ruma in https://github.com/ruma/ruma/pull/1622
This commit is contained in:
parent
62134dcc77
commit
3fef535ff2
9 changed files with 295 additions and 34 deletions
1
changelog.d/17395.feature
Normal file
1
changelog.d/17395.feature
Normal file
|
@ -0,0 +1 @@
|
||||||
|
Add `rooms.bump_stamp` for easier client-side sorting in experimental [MSC3575](https://github.com/matrix-org/matrix-spec-proposals/pull/3575) Sliding Sync `/sync` endpoint.
|
|
@ -128,9 +128,13 @@ class EventTypes:
|
||||||
SpaceParent: Final = "m.space.parent"
|
SpaceParent: Final = "m.space.parent"
|
||||||
|
|
||||||
Reaction: Final = "m.reaction"
|
Reaction: Final = "m.reaction"
|
||||||
|
Sticker: Final = "m.sticker"
|
||||||
|
LiveLocationShareStart: Final = "m.beacon_info"
|
||||||
|
|
||||||
CallInvite: Final = "m.call.invite"
|
CallInvite: Final = "m.call.invite"
|
||||||
|
|
||||||
|
PollStart: Final = "m.poll.start"
|
||||||
|
|
||||||
|
|
||||||
class ToDeviceEventTypes:
|
class ToDeviceEventTypes:
|
||||||
RoomKeyRequest: Final = "m.room_key_request"
|
RoomKeyRequest: Final = "m.room_key_request"
|
||||||
|
|
|
@ -54,6 +54,17 @@ if TYPE_CHECKING:
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
# The event types that clients should consider as new activity.
|
||||||
|
DEFAULT_BUMP_EVENT_TYPES = {
|
||||||
|
EventTypes.Message,
|
||||||
|
EventTypes.Encrypted,
|
||||||
|
EventTypes.Sticker,
|
||||||
|
EventTypes.CallInvite,
|
||||||
|
EventTypes.PollStart,
|
||||||
|
EventTypes.LiveLocationShareStart,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
def filter_membership_for_sync(
|
def filter_membership_for_sync(
|
||||||
*, membership: str, user_id: str, sender: Optional[str]
|
*, membership: str, user_id: str, sender: Optional[str]
|
||||||
) -> bool:
|
) -> bool:
|
||||||
|
@ -285,6 +296,7 @@ class _RoomMembershipForUser:
|
||||||
range
|
range
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
room_id: str
|
||||||
event_id: Optional[str]
|
event_id: Optional[str]
|
||||||
event_pos: PersistedEventPosition
|
event_pos: PersistedEventPosition
|
||||||
membership: str
|
membership: str
|
||||||
|
@ -469,7 +481,9 @@ class SlidingSyncHandler:
|
||||||
#
|
#
|
||||||
# Both sides of range are inclusive so we `+ 1`
|
# Both sides of range are inclusive so we `+ 1`
|
||||||
max_num_rooms = range[1] - range[0] + 1
|
max_num_rooms = range[1] - range[0] + 1
|
||||||
for room_id, _ in sorted_room_info[range[0] :]:
|
for room_membership in sorted_room_info[range[0] :]:
|
||||||
|
room_id = room_membership.room_id
|
||||||
|
|
||||||
if len(room_ids_in_list) >= max_num_rooms:
|
if len(room_ids_in_list) >= max_num_rooms:
|
||||||
break
|
break
|
||||||
|
|
||||||
|
@ -519,7 +533,7 @@ class SlidingSyncHandler:
|
||||||
user=sync_config.user,
|
user=sync_config.user,
|
||||||
room_id=room_id,
|
room_id=room_id,
|
||||||
room_sync_config=room_sync_config,
|
room_sync_config=room_sync_config,
|
||||||
rooms_membership_for_user_at_to_token=sync_room_map[room_id],
|
room_membership_for_user_at_to_token=sync_room_map[room_id],
|
||||||
from_token=from_token,
|
from_token=from_token,
|
||||||
to_token=to_token,
|
to_token=to_token,
|
||||||
)
|
)
|
||||||
|
@ -591,6 +605,7 @@ class SlidingSyncHandler:
|
||||||
# (below) because they are potentially from the current snapshot time
|
# (below) because they are potentially from the current snapshot time
|
||||||
# instead from the time of the `to_token`.
|
# instead from the time of the `to_token`.
|
||||||
room_for_user.room_id: _RoomMembershipForUser(
|
room_for_user.room_id: _RoomMembershipForUser(
|
||||||
|
room_id=room_for_user.room_id,
|
||||||
event_id=room_for_user.event_id,
|
event_id=room_for_user.event_id,
|
||||||
event_pos=room_for_user.event_pos,
|
event_pos=room_for_user.event_pos,
|
||||||
membership=room_for_user.membership,
|
membership=room_for_user.membership,
|
||||||
|
@ -691,6 +706,7 @@ class SlidingSyncHandler:
|
||||||
is not None
|
is not None
|
||||||
):
|
):
|
||||||
sync_room_id_set[room_id] = _RoomMembershipForUser(
|
sync_room_id_set[room_id] = _RoomMembershipForUser(
|
||||||
|
room_id=room_id,
|
||||||
event_id=first_membership_change_after_to_token.prev_event_id,
|
event_id=first_membership_change_after_to_token.prev_event_id,
|
||||||
event_pos=first_membership_change_after_to_token.prev_event_pos,
|
event_pos=first_membership_change_after_to_token.prev_event_pos,
|
||||||
membership=first_membership_change_after_to_token.prev_membership,
|
membership=first_membership_change_after_to_token.prev_membership,
|
||||||
|
@ -785,6 +801,7 @@ class SlidingSyncHandler:
|
||||||
# is their own leave event
|
# is their own leave event
|
||||||
if last_membership_change_in_from_to_range.membership == Membership.LEAVE:
|
if last_membership_change_in_from_to_range.membership == Membership.LEAVE:
|
||||||
filtered_sync_room_id_set[room_id] = _RoomMembershipForUser(
|
filtered_sync_room_id_set[room_id] = _RoomMembershipForUser(
|
||||||
|
room_id=room_id,
|
||||||
event_id=last_membership_change_in_from_to_range.event_id,
|
event_id=last_membership_change_in_from_to_range.event_id,
|
||||||
event_pos=last_membership_change_in_from_to_range.event_pos,
|
event_pos=last_membership_change_in_from_to_range.event_pos,
|
||||||
membership=last_membership_change_in_from_to_range.membership,
|
membership=last_membership_change_in_from_to_range.membership,
|
||||||
|
@ -969,7 +986,7 @@ class SlidingSyncHandler:
|
||||||
self,
|
self,
|
||||||
sync_room_map: Dict[str, _RoomMembershipForUser],
|
sync_room_map: Dict[str, _RoomMembershipForUser],
|
||||||
to_token: StreamToken,
|
to_token: StreamToken,
|
||||||
) -> List[Tuple[str, _RoomMembershipForUser]]:
|
) -> List[_RoomMembershipForUser]:
|
||||||
"""
|
"""
|
||||||
Sort by `stream_ordering` of the last event that the user should see in the
|
Sort by `stream_ordering` of the last event that the user should see in the
|
||||||
room. `stream_ordering` is unique so we get a stable sort.
|
room. `stream_ordering` is unique so we get a stable sort.
|
||||||
|
@ -1007,12 +1024,17 @@ class SlidingSyncHandler:
|
||||||
else:
|
else:
|
||||||
# Otherwise, if the user has left/been invited/knocked/been banned from
|
# Otherwise, if the user has left/been invited/knocked/been banned from
|
||||||
# a room, they shouldn't see anything past that point.
|
# a room, they shouldn't see anything past that point.
|
||||||
|
#
|
||||||
|
# FIXME: It's possible that people should see beyond this point in
|
||||||
|
# invited/knocked cases if for example the room has
|
||||||
|
# `invite`/`world_readable` history visibility, see
|
||||||
|
# https://github.com/matrix-org/matrix-spec-proposals/pull/3575#discussion_r1653045932
|
||||||
last_activity_in_room_map[room_id] = room_for_user.event_pos.stream
|
last_activity_in_room_map[room_id] = room_for_user.event_pos.stream
|
||||||
|
|
||||||
return sorted(
|
return sorted(
|
||||||
sync_room_map.items(),
|
sync_room_map.values(),
|
||||||
# Sort by the last activity (stream_ordering) in the room
|
# Sort by the last activity (stream_ordering) in the room
|
||||||
key=lambda room_info: last_activity_in_room_map[room_info[0]],
|
key=lambda room_info: last_activity_in_room_map[room_info.room_id],
|
||||||
# We want descending order
|
# We want descending order
|
||||||
reverse=True,
|
reverse=True,
|
||||||
)
|
)
|
||||||
|
@ -1022,7 +1044,7 @@ class SlidingSyncHandler:
|
||||||
user: UserID,
|
user: UserID,
|
||||||
room_id: str,
|
room_id: str,
|
||||||
room_sync_config: RoomSyncConfig,
|
room_sync_config: RoomSyncConfig,
|
||||||
rooms_membership_for_user_at_to_token: _RoomMembershipForUser,
|
room_membership_for_user_at_to_token: _RoomMembershipForUser,
|
||||||
from_token: Optional[StreamToken],
|
from_token: Optional[StreamToken],
|
||||||
to_token: StreamToken,
|
to_token: StreamToken,
|
||||||
) -> SlidingSyncResult.RoomResult:
|
) -> SlidingSyncResult.RoomResult:
|
||||||
|
@ -1036,7 +1058,7 @@ class SlidingSyncHandler:
|
||||||
room_id: The room ID to fetch data for
|
room_id: The room ID to fetch data for
|
||||||
room_sync_config: Config for what data we should fetch for a room in the
|
room_sync_config: Config for what data we should fetch for a room in the
|
||||||
sync response.
|
sync response.
|
||||||
rooms_membership_for_user_at_to_token: Membership information for the user
|
room_membership_for_user_at_to_token: Membership information for the user
|
||||||
in the room at the time of `to_token`.
|
in the room at the time of `to_token`.
|
||||||
from_token: The point in the stream to sync from.
|
from_token: The point in the stream to sync from.
|
||||||
to_token: The point in the stream to sync up to.
|
to_token: The point in the stream to sync up to.
|
||||||
|
@ -1056,7 +1078,7 @@ class SlidingSyncHandler:
|
||||||
if (
|
if (
|
||||||
room_sync_config.timeline_limit > 0
|
room_sync_config.timeline_limit > 0
|
||||||
# No timeline for invite/knock rooms (just `stripped_state`)
|
# No timeline for invite/knock rooms (just `stripped_state`)
|
||||||
and rooms_membership_for_user_at_to_token.membership
|
and room_membership_for_user_at_to_token.membership
|
||||||
not in (Membership.INVITE, Membership.KNOCK)
|
not in (Membership.INVITE, Membership.KNOCK)
|
||||||
):
|
):
|
||||||
limited = False
|
limited = False
|
||||||
|
@ -1069,12 +1091,12 @@ class SlidingSyncHandler:
|
||||||
# We're going to paginate backwards from the `to_token`
|
# We're going to paginate backwards from the `to_token`
|
||||||
from_bound = to_token.room_key
|
from_bound = to_token.room_key
|
||||||
# People shouldn't see past their leave/ban event
|
# People shouldn't see past their leave/ban event
|
||||||
if rooms_membership_for_user_at_to_token.membership in (
|
if room_membership_for_user_at_to_token.membership in (
|
||||||
Membership.LEAVE,
|
Membership.LEAVE,
|
||||||
Membership.BAN,
|
Membership.BAN,
|
||||||
):
|
):
|
||||||
from_bound = (
|
from_bound = (
|
||||||
rooms_membership_for_user_at_to_token.event_pos.to_room_stream_token()
|
room_membership_for_user_at_to_token.event_pos.to_room_stream_token()
|
||||||
)
|
)
|
||||||
|
|
||||||
# Determine whether we should limit the timeline to the token range.
|
# Determine whether we should limit the timeline to the token range.
|
||||||
|
@ -1089,7 +1111,7 @@ class SlidingSyncHandler:
|
||||||
to_bound = (
|
to_bound = (
|
||||||
from_token.room_key
|
from_token.room_key
|
||||||
if from_token is not None
|
if from_token is not None
|
||||||
and not rooms_membership_for_user_at_to_token.newly_joined
|
and not room_membership_for_user_at_to_token.newly_joined
|
||||||
else None
|
else None
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -1126,7 +1148,7 @@ class SlidingSyncHandler:
|
||||||
self.storage_controllers,
|
self.storage_controllers,
|
||||||
user.to_string(),
|
user.to_string(),
|
||||||
timeline_events,
|
timeline_events,
|
||||||
is_peeking=rooms_membership_for_user_at_to_token.membership
|
is_peeking=room_membership_for_user_at_to_token.membership
|
||||||
!= Membership.JOIN,
|
!= Membership.JOIN,
|
||||||
filter_send_to_client=True,
|
filter_send_to_client=True,
|
||||||
)
|
)
|
||||||
|
@ -1181,16 +1203,16 @@ class SlidingSyncHandler:
|
||||||
# Figure out any stripped state events for invite/knocks. This allows the
|
# Figure out any stripped state events for invite/knocks. This allows the
|
||||||
# potential joiner to identify the room.
|
# potential joiner to identify the room.
|
||||||
stripped_state: List[JsonDict] = []
|
stripped_state: List[JsonDict] = []
|
||||||
if rooms_membership_for_user_at_to_token.membership in (
|
if room_membership_for_user_at_to_token.membership in (
|
||||||
Membership.INVITE,
|
Membership.INVITE,
|
||||||
Membership.KNOCK,
|
Membership.KNOCK,
|
||||||
):
|
):
|
||||||
# This should never happen. If someone is invited/knocked on room, then
|
# This should never happen. If someone is invited/knocked on room, then
|
||||||
# there should be an event for it.
|
# there should be an event for it.
|
||||||
assert rooms_membership_for_user_at_to_token.event_id is not None
|
assert room_membership_for_user_at_to_token.event_id is not None
|
||||||
|
|
||||||
invite_or_knock_event = await self.store.get_event(
|
invite_or_knock_event = await self.store.get_event(
|
||||||
rooms_membership_for_user_at_to_token.event_id
|
room_membership_for_user_at_to_token.event_id
|
||||||
)
|
)
|
||||||
|
|
||||||
stripped_state = []
|
stripped_state = []
|
||||||
|
@ -1206,7 +1228,7 @@ class SlidingSyncHandler:
|
||||||
stripped_state.append(strip_event(invite_or_knock_event))
|
stripped_state.append(strip_event(invite_or_knock_event))
|
||||||
|
|
||||||
# TODO: Handle state resets. For example, if we see
|
# TODO: Handle state resets. For example, if we see
|
||||||
# `rooms_membership_for_user_at_to_token.membership = Membership.LEAVE` but
|
# `room_membership_for_user_at_to_token.membership = Membership.LEAVE` but
|
||||||
# `required_state` doesn't include it, we should indicate to the client that a
|
# `required_state` doesn't include it, we should indicate to the client that a
|
||||||
# state reset happened. Perhaps we should indicate this by setting `initial:
|
# state reset happened. Perhaps we should indicate this by setting `initial:
|
||||||
# True` and empty `required_state`.
|
# True` and empty `required_state`.
|
||||||
|
@ -1226,7 +1248,7 @@ class SlidingSyncHandler:
|
||||||
# `invite`/`knock` rooms only have `stripped_state`. See
|
# `invite`/`knock` rooms only have `stripped_state`. See
|
||||||
# https://github.com/matrix-org/matrix-spec-proposals/pull/3575#discussion_r1653045932
|
# https://github.com/matrix-org/matrix-spec-proposals/pull/3575#discussion_r1653045932
|
||||||
room_state: Optional[StateMap[EventBase]] = None
|
room_state: Optional[StateMap[EventBase]] = None
|
||||||
if rooms_membership_for_user_at_to_token.membership not in (
|
if room_membership_for_user_at_to_token.membership not in (
|
||||||
Membership.INVITE,
|
Membership.INVITE,
|
||||||
Membership.KNOCK,
|
Membership.KNOCK,
|
||||||
):
|
):
|
||||||
|
@ -1303,7 +1325,7 @@ class SlidingSyncHandler:
|
||||||
# initial sync
|
# initial sync
|
||||||
if initial:
|
if initial:
|
||||||
# People shouldn't see past their leave/ban event
|
# People shouldn't see past their leave/ban event
|
||||||
if rooms_membership_for_user_at_to_token.membership in (
|
if room_membership_for_user_at_to_token.membership in (
|
||||||
Membership.LEAVE,
|
Membership.LEAVE,
|
||||||
Membership.BAN,
|
Membership.BAN,
|
||||||
):
|
):
|
||||||
|
@ -1311,7 +1333,7 @@ class SlidingSyncHandler:
|
||||||
room_id,
|
room_id,
|
||||||
stream_position=to_token.copy_and_replace(
|
stream_position=to_token.copy_and_replace(
|
||||||
StreamKeyType.ROOM,
|
StreamKeyType.ROOM,
|
||||||
rooms_membership_for_user_at_to_token.event_pos.to_room_stream_token(),
|
room_membership_for_user_at_to_token.event_pos.to_room_stream_token(),
|
||||||
),
|
),
|
||||||
state_filter=state_filter,
|
state_filter=state_filter,
|
||||||
# Partially-stated rooms should have all state events except for
|
# Partially-stated rooms should have all state events except for
|
||||||
|
@ -1341,6 +1363,20 @@ class SlidingSyncHandler:
|
||||||
# we can return updates instead of the full required state.
|
# we can return updates instead of the full required state.
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
# Figure out the last bump event in the room
|
||||||
|
last_bump_event_result = (
|
||||||
|
await self.store.get_last_event_pos_in_room_before_stream_ordering(
|
||||||
|
room_id, to_token.room_key, event_types=DEFAULT_BUMP_EVENT_TYPES
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
# By default, just choose the membership event position
|
||||||
|
bump_stamp = room_membership_for_user_at_to_token.event_pos.stream
|
||||||
|
# But if we found a bump event, use that instead
|
||||||
|
if last_bump_event_result is not None:
|
||||||
|
_, bump_event_pos = last_bump_event_result
|
||||||
|
bump_stamp = bump_event_pos.stream
|
||||||
|
|
||||||
return SlidingSyncResult.RoomResult(
|
return SlidingSyncResult.RoomResult(
|
||||||
# TODO: Dummy value
|
# TODO: Dummy value
|
||||||
name=None,
|
name=None,
|
||||||
|
@ -1358,6 +1394,7 @@ class SlidingSyncHandler:
|
||||||
prev_batch=prev_batch_token,
|
prev_batch=prev_batch_token,
|
||||||
limited=limited,
|
limited=limited,
|
||||||
num_live=num_live,
|
num_live=num_live,
|
||||||
|
bump_stamp=bump_stamp,
|
||||||
# TODO: Dummy values
|
# TODO: Dummy values
|
||||||
joined_count=0,
|
joined_count=0,
|
||||||
invited_count=0,
|
invited_count=0,
|
||||||
|
|
|
@ -982,6 +982,7 @@ class SlidingSyncRestServlet(RestServlet):
|
||||||
serialized_rooms: Dict[str, JsonDict] = {}
|
serialized_rooms: Dict[str, JsonDict] = {}
|
||||||
for room_id, room_result in rooms.items():
|
for room_id, room_result in rooms.items():
|
||||||
serialized_rooms[room_id] = {
|
serialized_rooms[room_id] = {
|
||||||
|
"bump_stamp": room_result.bump_stamp,
|
||||||
"joined_count": room_result.joined_count,
|
"joined_count": room_result.joined_count,
|
||||||
"invited_count": room_result.invited_count,
|
"invited_count": room_result.invited_count,
|
||||||
"notification_count": room_result.notification_count,
|
"notification_count": room_result.notification_count,
|
||||||
|
|
|
@ -1178,6 +1178,7 @@ class StreamWorkerStore(EventsWorkerStore, SQLBaseStore):
|
||||||
self,
|
self,
|
||||||
room_id: str,
|
room_id: str,
|
||||||
end_token: RoomStreamToken,
|
end_token: RoomStreamToken,
|
||||||
|
event_types: Optional[Collection[str]] = None,
|
||||||
) -> Optional[Tuple[str, PersistedEventPosition]]:
|
) -> Optional[Tuple[str, PersistedEventPosition]]:
|
||||||
"""
|
"""
|
||||||
Returns the ID and event position of the last event in a room at or before a
|
Returns the ID and event position of the last event in a room at or before a
|
||||||
|
@ -1186,6 +1187,7 @@ class StreamWorkerStore(EventsWorkerStore, SQLBaseStore):
|
||||||
Args:
|
Args:
|
||||||
room_id
|
room_id
|
||||||
end_token: The token used to stream from
|
end_token: The token used to stream from
|
||||||
|
event_types: Optional allowlist of event types to filter by
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
The ID of the most recent event and it's position, or None if there are no
|
The ID of the most recent event and it's position, or None if there are no
|
||||||
|
@ -1207,9 +1209,17 @@ class StreamWorkerStore(EventsWorkerStore, SQLBaseStore):
|
||||||
min_stream = end_token.stream
|
min_stream = end_token.stream
|
||||||
max_stream = end_token.get_max_stream_pos()
|
max_stream = end_token.get_max_stream_pos()
|
||||||
|
|
||||||
# We use `union all` because we don't need any of the deduplication logic
|
event_type_clause = ""
|
||||||
# (`union` is really a union + distinct). `UNION ALL` does preserve the
|
event_type_args: List[str] = []
|
||||||
# ordering of the operand queries but there is no actual gurantee that it
|
if event_types is not None and len(event_types) > 0:
|
||||||
|
event_type_clause, event_type_args = make_in_list_sql_clause(
|
||||||
|
txn.database_engine, "type", event_types
|
||||||
|
)
|
||||||
|
event_type_clause = f"AND {event_type_clause}"
|
||||||
|
|
||||||
|
# We use `UNION ALL` because we don't need any of the deduplication logic
|
||||||
|
# (`UNION` is really a `UNION` + `DISTINCT`). `UNION ALL` does preserve the
|
||||||
|
# ordering of the operand queries but there is no actual guarantee that it
|
||||||
# has this behavior in all scenarios so we need the extra `ORDER BY` at the
|
# has this behavior in all scenarios so we need the extra `ORDER BY` at the
|
||||||
# bottom.
|
# bottom.
|
||||||
sql = """
|
sql = """
|
||||||
|
@ -1218,6 +1228,7 @@ class StreamWorkerStore(EventsWorkerStore, SQLBaseStore):
|
||||||
FROM events
|
FROM events
|
||||||
LEFT JOIN rejections USING (event_id)
|
LEFT JOIN rejections USING (event_id)
|
||||||
WHERE room_id = ?
|
WHERE room_id = ?
|
||||||
|
%s
|
||||||
AND ? < stream_ordering AND stream_ordering <= ?
|
AND ? < stream_ordering AND stream_ordering <= ?
|
||||||
AND NOT outlier
|
AND NOT outlier
|
||||||
AND rejections.event_id IS NULL
|
AND rejections.event_id IS NULL
|
||||||
|
@ -1229,6 +1240,7 @@ class StreamWorkerStore(EventsWorkerStore, SQLBaseStore):
|
||||||
FROM events
|
FROM events
|
||||||
LEFT JOIN rejections USING (event_id)
|
LEFT JOIN rejections USING (event_id)
|
||||||
WHERE room_id = ?
|
WHERE room_id = ?
|
||||||
|
%s
|
||||||
AND stream_ordering <= ?
|
AND stream_ordering <= ?
|
||||||
AND NOT outlier
|
AND NOT outlier
|
||||||
AND rejections.event_id IS NULL
|
AND rejections.event_id IS NULL
|
||||||
|
@ -1236,16 +1248,17 @@ class StreamWorkerStore(EventsWorkerStore, SQLBaseStore):
|
||||||
LIMIT 1
|
LIMIT 1
|
||||||
) AS b
|
) AS b
|
||||||
ORDER BY stream_ordering DESC
|
ORDER BY stream_ordering DESC
|
||||||
"""
|
""" % (
|
||||||
|
event_type_clause,
|
||||||
|
event_type_clause,
|
||||||
|
)
|
||||||
txn.execute(
|
txn.execute(
|
||||||
sql,
|
sql,
|
||||||
(
|
[room_id]
|
||||||
room_id,
|
+ event_type_args
|
||||||
min_stream,
|
+ [min_stream, max_stream, room_id]
|
||||||
max_stream,
|
+ event_type_args
|
||||||
room_id,
|
+ [min_stream],
|
||||||
min_stream,
|
|
||||||
),
|
|
||||||
)
|
)
|
||||||
|
|
||||||
for instance_name, stream_ordering, topological_ordering, event_id in txn:
|
for instance_name, stream_ordering, topological_ordering, event_id in txn:
|
||||||
|
|
|
@ -183,6 +183,13 @@ class SlidingSyncResult:
|
||||||
events because if a room not in the sliding window bumps into the window because
|
events because if a room not in the sliding window bumps into the window because
|
||||||
of an @mention it will have `initial: true` yet contain a single live event
|
of an @mention it will have `initial: true` yet contain a single live event
|
||||||
(with potentially other old events in the timeline).
|
(with potentially other old events in the timeline).
|
||||||
|
bump_stamp: The `stream_ordering` of the last event according to the
|
||||||
|
`bump_event_types`. This helps clients sort more readily without them
|
||||||
|
needing to pull in a bunch of the timeline to determine the last activity.
|
||||||
|
`bump_event_types` is a thing because for example, we don't want display
|
||||||
|
name changes to mark the room as unread and bump it to the top. For
|
||||||
|
encrypted rooms, we just have to consider any activity as a bump because we
|
||||||
|
can't see the content and the client has to figure it out for themselves.
|
||||||
joined_count: The number of users with membership of join, including the client's
|
joined_count: The number of users with membership of join, including the client's
|
||||||
own user ID. (same as sync `v2 m.joined_member_count`)
|
own user ID. (same as sync `v2 m.joined_member_count`)
|
||||||
invited_count: The number of users with membership of invite. (same as sync v2
|
invited_count: The number of users with membership of invite. (same as sync v2
|
||||||
|
@ -211,6 +218,7 @@ class SlidingSyncResult:
|
||||||
limited: Optional[bool]
|
limited: Optional[bool]
|
||||||
# Only optional because it won't be included for invite/knock rooms with `stripped_state`
|
# Only optional because it won't be included for invite/knock rooms with `stripped_state`
|
||||||
num_live: Optional[int]
|
num_live: Optional[int]
|
||||||
|
bump_stamp: int
|
||||||
joined_count: int
|
joined_count: int
|
||||||
invited_count: int
|
invited_count: int
|
||||||
notification_count: int
|
notification_count: int
|
||||||
|
|
|
@ -2844,7 +2844,7 @@ class SortRoomsTestCase(HomeserverTestCase):
|
||||||
)
|
)
|
||||||
|
|
||||||
# Sort the rooms (what we're testing)
|
# Sort the rooms (what we're testing)
|
||||||
sorted_room_info = self.get_success(
|
sorted_sync_rooms = self.get_success(
|
||||||
self.sliding_sync_handler.sort_rooms(
|
self.sliding_sync_handler.sort_rooms(
|
||||||
sync_room_map=sync_room_map,
|
sync_room_map=sync_room_map,
|
||||||
to_token=after_rooms_token,
|
to_token=after_rooms_token,
|
||||||
|
@ -2852,7 +2852,7 @@ class SortRoomsTestCase(HomeserverTestCase):
|
||||||
)
|
)
|
||||||
|
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
[room_id for room_id, _ in sorted_room_info],
|
[room_membership.room_id for room_membership in sorted_sync_rooms],
|
||||||
[room_id2, room_id1],
|
[room_id2, room_id1],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -2927,7 +2927,7 @@ class SortRoomsTestCase(HomeserverTestCase):
|
||||||
)
|
)
|
||||||
|
|
||||||
# Sort the rooms (what we're testing)
|
# Sort the rooms (what we're testing)
|
||||||
sorted_room_info = self.get_success(
|
sorted_sync_rooms = self.get_success(
|
||||||
self.sliding_sync_handler.sort_rooms(
|
self.sliding_sync_handler.sort_rooms(
|
||||||
sync_room_map=sync_room_map,
|
sync_room_map=sync_room_map,
|
||||||
to_token=after_rooms_token,
|
to_token=after_rooms_token,
|
||||||
|
@ -2935,7 +2935,7 @@ class SortRoomsTestCase(HomeserverTestCase):
|
||||||
)
|
)
|
||||||
|
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
[room_id for room_id, _ in sorted_room_info],
|
[room_membership.room_id for room_membership in sorted_sync_rooms],
|
||||||
[room_id2, room_id1, room_id3],
|
[room_id2, room_id1, room_id3],
|
||||||
"Corresponding map to disambiguate the opaque room IDs: "
|
"Corresponding map to disambiguate the opaque room IDs: "
|
||||||
+ str(
|
+ str(
|
||||||
|
@ -2946,3 +2946,63 @@ class SortRoomsTestCase(HomeserverTestCase):
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def test_default_bump_event_types(self) -> None:
|
||||||
|
"""
|
||||||
|
Test that we only consider the *latest* event in the room when sorting (not
|
||||||
|
`bump_event_types`).
|
||||||
|
"""
|
||||||
|
user1_id = self.register_user("user1", "pass")
|
||||||
|
user1_tok = self.login(user1_id, "pass")
|
||||||
|
|
||||||
|
room_id1 = self.helper.create_room_as(
|
||||||
|
user1_id,
|
||||||
|
tok=user1_tok,
|
||||||
|
)
|
||||||
|
message_response = self.helper.send(room_id1, "message in room1", tok=user1_tok)
|
||||||
|
room_id2 = self.helper.create_room_as(
|
||||||
|
user1_id,
|
||||||
|
tok=user1_tok,
|
||||||
|
)
|
||||||
|
self.helper.send(room_id2, "message in room2", tok=user1_tok)
|
||||||
|
|
||||||
|
# Send a reaction in room1 which isn't in `DEFAULT_BUMP_EVENT_TYPES` but we only
|
||||||
|
# care about sorting by the *latest* event in the room.
|
||||||
|
self.helper.send_event(
|
||||||
|
room_id1,
|
||||||
|
type=EventTypes.Reaction,
|
||||||
|
content={
|
||||||
|
"m.relates_to": {
|
||||||
|
"event_id": message_response["event_id"],
|
||||||
|
"key": "👍",
|
||||||
|
"rel_type": "m.annotation",
|
||||||
|
}
|
||||||
|
},
|
||||||
|
tok=user1_tok,
|
||||||
|
)
|
||||||
|
|
||||||
|
after_rooms_token = self.event_sources.get_current_token()
|
||||||
|
|
||||||
|
# Get the rooms the user should be syncing with
|
||||||
|
sync_room_map = self.get_success(
|
||||||
|
self.sliding_sync_handler.get_sync_room_ids_for_user(
|
||||||
|
UserID.from_string(user1_id),
|
||||||
|
from_token=None,
|
||||||
|
to_token=after_rooms_token,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
# Sort the rooms (what we're testing)
|
||||||
|
sorted_sync_rooms = self.get_success(
|
||||||
|
self.sliding_sync_handler.sort_rooms(
|
||||||
|
sync_room_map=sync_room_map,
|
||||||
|
to_token=after_rooms_token,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
[room_membership.room_id for room_membership in sorted_sync_rooms],
|
||||||
|
# room1 sorts before room2 because it has the latest event (the reaction).
|
||||||
|
# We only care about the *latest* event in the room.
|
||||||
|
[room_id1, room_id2],
|
||||||
|
)
|
||||||
|
|
|
@ -2029,6 +2029,102 @@ class SlidingSyncTestCase(unittest.HomeserverTestCase):
|
||||||
channel.json_body["rooms"][room_id1],
|
channel.json_body["rooms"][room_id1],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def test_rooms_bump_stamp(self) -> None:
|
||||||
|
"""
|
||||||
|
Test that `bump_stamp` is present and pointing to relevant events.
|
||||||
|
"""
|
||||||
|
user1_id = self.register_user("user1", "pass")
|
||||||
|
user1_tok = self.login(user1_id, "pass")
|
||||||
|
|
||||||
|
room_id1 = self.helper.create_room_as(
|
||||||
|
user1_id,
|
||||||
|
tok=user1_tok,
|
||||||
|
)
|
||||||
|
event_response1 = message_response = self.helper.send(
|
||||||
|
room_id1, "message in room1", tok=user1_tok
|
||||||
|
)
|
||||||
|
event_pos1 = self.get_success(
|
||||||
|
self.store.get_position_for_event(event_response1["event_id"])
|
||||||
|
)
|
||||||
|
room_id2 = self.helper.create_room_as(
|
||||||
|
user1_id,
|
||||||
|
tok=user1_tok,
|
||||||
|
)
|
||||||
|
send_response2 = self.helper.send(room_id2, "message in room2", tok=user1_tok)
|
||||||
|
event_pos2 = self.get_success(
|
||||||
|
self.store.get_position_for_event(send_response2["event_id"])
|
||||||
|
)
|
||||||
|
|
||||||
|
# Send a reaction in room1 but it shouldn't affect the `bump_stamp`
|
||||||
|
# because reactions are not part of the `DEFAULT_BUMP_EVENT_TYPES`
|
||||||
|
self.helper.send_event(
|
||||||
|
room_id1,
|
||||||
|
type=EventTypes.Reaction,
|
||||||
|
content={
|
||||||
|
"m.relates_to": {
|
||||||
|
"event_id": message_response["event_id"],
|
||||||
|
"key": "👍",
|
||||||
|
"rel_type": "m.annotation",
|
||||||
|
}
|
||||||
|
},
|
||||||
|
tok=user1_tok,
|
||||||
|
)
|
||||||
|
|
||||||
|
# Make the Sliding Sync request
|
||||||
|
timeline_limit = 100
|
||||||
|
channel = self.make_request(
|
||||||
|
"POST",
|
||||||
|
self.sync_endpoint,
|
||||||
|
{
|
||||||
|
"lists": {
|
||||||
|
"foo-list": {
|
||||||
|
"ranges": [[0, 1]],
|
||||||
|
"required_state": [],
|
||||||
|
"timeline_limit": timeline_limit,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
access_token=user1_tok,
|
||||||
|
)
|
||||||
|
self.assertEqual(channel.code, 200, channel.json_body)
|
||||||
|
|
||||||
|
# Make sure it has the foo-list we requested
|
||||||
|
self.assertListEqual(
|
||||||
|
list(channel.json_body["lists"].keys()),
|
||||||
|
["foo-list"],
|
||||||
|
channel.json_body["lists"].keys(),
|
||||||
|
)
|
||||||
|
|
||||||
|
# Make sure the list includes the rooms in the right order
|
||||||
|
self.assertListEqual(
|
||||||
|
list(channel.json_body["lists"]["foo-list"]["ops"]),
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"op": "SYNC",
|
||||||
|
"range": [0, 1],
|
||||||
|
# room1 sorts before room2 because it has the latest event (the
|
||||||
|
# reaction)
|
||||||
|
"room_ids": [room_id1, room_id2],
|
||||||
|
}
|
||||||
|
],
|
||||||
|
channel.json_body["lists"]["foo-list"],
|
||||||
|
)
|
||||||
|
|
||||||
|
# The `bump_stamp` for room1 should point at the latest message (not the
|
||||||
|
# reaction since it's not one of the `DEFAULT_BUMP_EVENT_TYPES`)
|
||||||
|
self.assertEqual(
|
||||||
|
channel.json_body["rooms"][room_id1]["bump_stamp"],
|
||||||
|
event_pos1.stream,
|
||||||
|
channel.json_body["rooms"][room_id1],
|
||||||
|
)
|
||||||
|
|
||||||
|
# The `bump_stamp` for room2 should point at the latest message
|
||||||
|
self.assertEqual(
|
||||||
|
channel.json_body["rooms"][room_id2]["bump_stamp"],
|
||||||
|
event_pos2.stream,
|
||||||
|
channel.json_body["rooms"][room_id2],
|
||||||
|
)
|
||||||
|
|
||||||
def test_rooms_newly_joined_incremental_sync(self) -> None:
|
def test_rooms_newly_joined_incremental_sync(self) -> None:
|
||||||
"""
|
"""
|
||||||
Test that when we make an incremental sync with a `newly_joined` `rooms`, we are
|
Test that when we make an incremental sync with a `newly_joined` `rooms`, we are
|
||||||
|
|
|
@ -556,6 +556,47 @@ class GetLastEventInRoomBeforeStreamOrderingTestCase(HomeserverTestCase):
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def test_restrict_event_types(self) -> None:
|
||||||
|
"""
|
||||||
|
Test that we only consider given `event_types` when finding the last event
|
||||||
|
before a token.
|
||||||
|
"""
|
||||||
|
user1_id = self.register_user("user1", "pass")
|
||||||
|
user1_tok = self.login(user1_id, "pass")
|
||||||
|
|
||||||
|
room_id1 = self.helper.create_room_as(user1_id, tok=user1_tok, is_public=True)
|
||||||
|
event_response = self.helper.send_event(
|
||||||
|
room_id1,
|
||||||
|
type="org.matrix.special_message",
|
||||||
|
content={"body": "before1, target!"},
|
||||||
|
tok=user1_tok,
|
||||||
|
)
|
||||||
|
self.helper.send(room_id1, "before2", tok=user1_tok)
|
||||||
|
|
||||||
|
after_room_token = self.event_sources.get_current_token()
|
||||||
|
|
||||||
|
# Send some events after the token
|
||||||
|
self.helper.send_event(
|
||||||
|
room_id1,
|
||||||
|
type="org.matrix.special_message",
|
||||||
|
content={"body": "after1"},
|
||||||
|
tok=user1_tok,
|
||||||
|
)
|
||||||
|
self.helper.send(room_id1, "after2", tok=user1_tok)
|
||||||
|
|
||||||
|
last_event_result = self.get_success(
|
||||||
|
self.store.get_last_event_pos_in_room_before_stream_ordering(
|
||||||
|
room_id=room_id1,
|
||||||
|
end_token=after_room_token.room_key,
|
||||||
|
event_types=["org.matrix.special_message"],
|
||||||
|
)
|
||||||
|
)
|
||||||
|
assert last_event_result is not None
|
||||||
|
last_event_id, _ = last_event_result
|
||||||
|
|
||||||
|
# Make sure it's the last event before the token
|
||||||
|
self.assertEqual(last_event_id, event_response["event_id"])
|
||||||
|
|
||||||
|
|
||||||
class GetCurrentStateDeltaMembershipChangesForUserTestCase(HomeserverTestCase):
|
class GetCurrentStateDeltaMembershipChangesForUserTestCase(HomeserverTestCase):
|
||||||
"""
|
"""
|
||||||
|
|
Loading…
Reference in a new issue