This repository has been archived by the owner on Apr 26, 2024. It is now read-only.
-
-
Notifications
You must be signed in to change notification settings - Fork 2.1k
initial cut at a room summary API #3574
Merged
Merged
Changes from 41 commits
Commits
Show all changes
52 commits
Select commit
Hold shift + click to select a range
7b7fd27
untested attempt at deduplicating lazy-loaded members
ara4n c341d81
Merge branch 'develop' into matthew/remove_redundant_lazy_members
ara4n f7bd5da
add include_redundant_members filter option & make it work
ara4n 589e5aa
merge and apply isort
ara4n 8e66dd1
merge in #2970
ara4n a08b37b
fix bad merge
ara4n 7362e6c
make /context lazyload & filter aware
ara4n cd28d2f
speed up /members and add at= and membership params
ara4n a17f0b6
make it work
ara4n 9ba6ef2
Merge branch 'matthew/lazy_load_apis' into matthew/members_at
ara4n c6117fa
make it work
ara4n 8f1585d
make filtering work
ara4n 42308c0
initial cut at a room summary API
ara4n 0beeecf
remove debug log
ara4n 63ce31b
namespace the summary fields correctly
ara4n c8cbede
fix key typo
bwindels f9c3c26
Merge branch 'matthew/filter_members' into matthew/remove_redundant_l…
ara4n c2870ab
Merge branch 'matthew/remove_redundant_lazy_members' into matthew/laz…
ara4n ffb7a4c
Merge branch 'matthew/lazy_load_apis' into matthew/members_at
ara4n c1c6ff1
Merge branch 'matthew/members_at' into matthew/room_summary
ara4n 7d99b0e
changelog
ara4n cd27a77
changelog
ara4n 4018a6d
changelog
ara4n c238a88
changelog
ara4n 1ba3683
return the correct counts & self-exclude from heros
ara4n f129390
spell heroes correctly
ara4n e61071a
fix heroes definition to match MSC
ara4n d32e5f8
Merge branch 'matthew/filter_members' into matthew/remove_redundant_l…
ara4n 238f750
deduplicating redundant members via event_id rather than mxid
ara4n 2a79e1a
Merge branch 'matthew/remove_redundant_lazy_members' into matthew/laz…
ara4n 48f3e43
Merge branch 'matthew/lazy_load_apis' into matthew/members_at
ara4n 7cc81ed
Merge branch 'matthew/members_at' into matthew/room_summary
ara4n 08af91d
fix merge fail
ara4n e76254b
Merge branch 'matthew/members_at' into matthew/room_summary
ara4n e952368
incorporate review
ara4n 2d9c062
Merge branch 'develop' into matthew/members_at
ara4n eebee08
convert /members?at= to take a stream token
ara4n 5df2c36
Merge branch 'develop' into matthew/room_summary
ara4n d0c0d72
incorporate all the review feedback
ara4n b327e07
fix use of get_recent_event_ids_for_room
ara4n 9e4acb7
Merge branch 'matthew/members_at' into matthew/room_summary
ara4n 32bf4fa
return early rather than big if blocks
ara4n 859ad35
incorporate PR review
ara4n dd4498f
Merge branch 'develop' into matthew/members_at
ara4n 217d5dd
Merge branch 'develop' into matthew/members_at
ara4n 8442478
Merge branch 'matthew/members_at' into matthew/room_summary
ara4n f5189b9
remove incorrectly reintroduced method
richvdh 0d5770d
cleanups
richvdh c2d9c96
Merge branch 'matthew/members_at' into matthew/room_summary
richvdh dc085c0
Merge remote-tracking branch 'origin/develop' into matthew/room_summary
richvdh 7306074
remove spurious changelogs
richvdh c3cdc21
factor out get_lazy_loaded_members_cache
ara4n File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
add support for the include_redundant_members filter param as per MSC1227 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
make the /context API filter & lazy-load aware as per MSC1227 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
speed up /members API and add `at` and `membership` params as per MSC1227 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
implement `summary` block in /sync response as per MSC688 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -75,6 +75,7 @@ class JoinedSyncResult(collections.namedtuple("JoinedSyncResult", [ | |
"ephemeral", | ||
"account_data", | ||
"unread_notifications", | ||
"summary", | ||
])): | ||
__slots__ = [] | ||
|
||
|
@@ -494,10 +495,128 @@ def get_state_at(self, room_id, stream_position, types=None, filtered_types=None | |
state = {} | ||
defer.returnValue(state) | ||
|
||
@defer.inlineCallbacks | ||
def compute_summary(self, room_id, sync_config, batch, state, now_token): | ||
""" Works out a room summary block for this room, summarising the number | ||
of joined members in the room, and providing the 'hero' members if the | ||
room has no name so clients can consistently name rooms. Also adds | ||
state events to 'state' if needed to describe the heroes. | ||
|
||
Args: | ||
room_id(str): | ||
sync_config(synapse.handlers.sync.SyncConfig): | ||
batch(synapse.handlers.sync.TimelineBatch): The timeline batch for | ||
the room that will be sent to the user. | ||
state(dict): dict of (type, state_key) -> Event as returned by | ||
compute_state_delta | ||
now_token(str): Token of the end of the current batch. | ||
|
||
Returns: | ||
A deferred dict describing the room summary | ||
""" | ||
|
||
# FIXME: this promulgates https://github.com/matrix-org/synapse/issues/3305 | ||
last_events, _ = yield self.store.get_recent_event_ids_for_room( | ||
room_id, end_token=now_token.room_key, limit=1, | ||
) | ||
|
||
if not last_events: | ||
defer.returnValue(None) | ||
return | ||
|
||
last_event = last_events[-1] | ||
state_ids = yield self.store.get_state_ids_for_event( | ||
last_event.event_id, [ | ||
(EventTypes.Member, None), | ||
(EventTypes.Name, ''), | ||
(EventTypes.CanonicalAlias, ''), | ||
] | ||
) | ||
|
||
member_ids = { | ||
state_key: event_id | ||
for (t, state_key), event_id in state_ids.iteritems() | ||
if t == EventTypes.Member | ||
} | ||
name_id = state_ids.get((EventTypes.Name, '')) | ||
canonical_alias_id = state_ids.get((EventTypes.CanonicalAlias, '')) | ||
|
||
summary = {} | ||
|
||
# FIXME: it feels very heavy to load up every single membership event | ||
# just to calculate the counts. | ||
member_events = yield self.store.get_events(member_ids.values()) | ||
|
||
joined_user_ids = [] | ||
invited_user_ids = [] | ||
|
||
for ev in member_events.values(): | ||
if ev.content.get("membership") == Membership.JOIN: | ||
joined_user_ids.append(ev.state_key) | ||
elif ev.content.get("membership") == Membership.INVITE: | ||
invited_user_ids.append(ev.state_key) | ||
|
||
# TODO: only send these when they change. | ||
summary["m.joined_member_count"] = len(joined_user_ids) | ||
summary["m.invited_member_count"] = len(invited_user_ids) | ||
|
||
if not name_id and not canonical_alias_id: | ||
# FIXME: order by stream ordering, not alphabetic | ||
|
||
me = sync_config.user.to_string() | ||
if (joined_user_ids or invited_user_ids): | ||
summary['m.heroes'] = sorted( | ||
[ | ||
user_id | ||
for user_id in (joined_user_ids + invited_user_ids) | ||
if user_id != me | ||
] | ||
)[0:5] | ||
else: | ||
summary['m.heroes'] = sorted( | ||
[user_id for user_id in member_ids.keys() if user_id != me] | ||
)[0:5] | ||
|
||
if sync_config.filter_collection.lazy_load_members(): | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Again, make this into an early return? |
||
# ensure we send membership events for heroes if needed | ||
cache_key = (sync_config.user.to_string(), sync_config.device_id) | ||
cache = self.lazy_loaded_members_cache.get(cache_key) | ||
|
||
# track which members the client should already know about via LL: | ||
# Ones which are already in state... | ||
existing_members = set( | ||
user_id for (typ, user_id) in state.keys() | ||
if typ == EventTypes.Member | ||
) | ||
|
||
# ...or ones which are in the timeline... | ||
for ev in batch.events: | ||
if ev.type == EventTypes.Member: | ||
existing_members.add(ev.state_key) | ||
|
||
# ...and then ensure any missing ones get included in state. | ||
missing_hero_event_ids = [ | ||
member_ids[hero_id] | ||
for hero_id in summary['m.heroes'] | ||
if ( | ||
cache.get(hero_id) != member_ids[hero_id] and | ||
hero_id not in existing_members | ||
) | ||
] | ||
|
||
missing_hero_state = yield self.store.get_events(missing_hero_event_ids) | ||
missing_hero_state = missing_hero_state.values() | ||
|
||
for s in missing_hero_state: | ||
cache.set(s.state_key, s.event_id) | ||
state[(EventTypes.Member, s.state_key)] = s | ||
|
||
defer.returnValue(summary) | ||
|
||
@defer.inlineCallbacks | ||
def compute_state_delta(self, room_id, batch, sync_config, since_token, now_token, | ||
full_state): | ||
""" Works out the differnce in state between the start of the timeline | ||
""" Works out the difference in state between the start of the timeline | ||
and the previous sync. | ||
|
||
Args: | ||
|
@@ -511,7 +630,7 @@ def compute_state_delta(self, room_id, batch, sync_config, since_token, now_toke | |
full_state(bool): Whether to force returning the full state. | ||
|
||
Returns: | ||
A deferred new event dictionary | ||
A deferred dict of (type, state_key) -> Event | ||
""" | ||
# TODO(mjark) Check if the state events were received by the server | ||
# after the previous sync, since we need to include those state | ||
|
@@ -1416,7 +1535,6 @@ def _generate_room_entry(self, sync_result_builder, ignored_users, | |
if events == [] and tags is None: | ||
return | ||
|
||
since_token = sync_result_builder.since_token | ||
now_token = sync_result_builder.now_token | ||
sync_config = sync_result_builder.sync_config | ||
|
||
|
@@ -1459,6 +1577,18 @@ def _generate_room_entry(self, sync_result_builder, ignored_users, | |
full_state=full_state | ||
) | ||
|
||
summary = {} | ||
if ( | ||
sync_config.filter_collection.lazy_load_members() and | ||
( | ||
any(ev.type == EventTypes.Member for ev in batch.events) or | ||
since_token is None | ||
) | ||
): | ||
summary = yield self.compute_summary( | ||
room_id, sync_config, batch, state, now_token | ||
) | ||
|
||
if room_builder.rtype == "joined": | ||
unread_notifications = {} | ||
room_sync = JoinedSyncResult( | ||
|
@@ -1468,6 +1598,7 @@ def _generate_room_entry(self, sync_result_builder, ignored_users, | |
ephemeral=ephemeral, | ||
account_data=account_data_events, | ||
unread_notifications=unread_notifications, | ||
summary=summary, | ||
) | ||
|
||
if room_sync or always_include: | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
can we do an early return if
name_id or canonical_alias_id
rather than having this massiveif
block?