Use the proper serialization format when bundling aggregations. (#12090)

This ensures that the `latest_event` field of the bundled aggregation
for threads uses the same format as the other events in the response.
This commit is contained in:
Patrick Cloke 2022-03-03 10:43:06 -05:00 committed by GitHub
parent a511a890d7
commit 1d11b452b7
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
10 changed files with 130 additions and 143 deletions

1
changelog.d/12090.bugfix Normal file
View File

@ -0,0 +1 @@
Use the proper serialization format for bundled thread aggregations. The bug has existed since Synapse v1.48.0.

View File

@ -25,7 +25,7 @@ from synapse.appservice import (
TransactionUnusedFallbackKeys, TransactionUnusedFallbackKeys,
) )
from synapse.events import EventBase from synapse.events import EventBase
from synapse.events.utils import serialize_event from synapse.events.utils import SerializeEventConfig, serialize_event
from synapse.http.client import SimpleHttpClient from synapse.http.client import SimpleHttpClient
from synapse.types import JsonDict, ThirdPartyInstanceID from synapse.types import JsonDict, ThirdPartyInstanceID
from synapse.util.caches.response_cache import ResponseCache from synapse.util.caches.response_cache import ResponseCache
@ -321,16 +321,18 @@ class ApplicationServiceApi(SimpleHttpClient):
serialize_event( serialize_event(
e, e,
time_now, time_now,
as_client_event=True, config=SerializeEventConfig(
# If this is an invite or a knock membership event, and we're interested as_client_event=True,
# in this user, then include any stripped state alongside the event. # If this is an invite or a knock membership event, and we're interested
include_stripped_room_state=( # in this user, then include any stripped state alongside the event.
e.type == EventTypes.Member include_stripped_room_state=(
and ( e.type == EventTypes.Member
e.membership == Membership.INVITE and (
or e.membership == Membership.KNOCK e.membership == Membership.INVITE
) or e.membership == Membership.KNOCK
and service.is_interested_in_user(e.state_key) )
and service.is_interested_in_user(e.state_key)
),
), ),
) )
for e in events for e in events

View File

@ -26,6 +26,7 @@ from typing import (
Union, Union,
) )
import attr
from frozendict import frozendict from frozendict import frozendict
from synapse.api.constants import EventContentFields, EventTypes, RelationTypes from synapse.api.constants import EventContentFields, EventTypes, RelationTypes
@ -303,29 +304,37 @@ def format_event_for_client_v2_without_room_id(d: JsonDict) -> JsonDict:
return d return d
@attr.s(slots=True, frozen=True, auto_attribs=True)
class SerializeEventConfig:
as_client_event: bool = True
# Function to convert from federation format to client format
event_format: Callable[[JsonDict], JsonDict] = format_event_for_client_v1
# ID of the user's auth token - used for namespacing of transaction IDs
token_id: Optional[int] = None
# List of event fields to include. If empty, all fields will be returned.
only_event_fields: Optional[List[str]] = None
# Some events can have stripped room state stored in the `unsigned` field.
# This is required for invite and knock functionality. If this option is
# False, that state will be removed from the event before it is returned.
# Otherwise, it will be kept.
include_stripped_room_state: bool = False
_DEFAULT_SERIALIZE_EVENT_CONFIG = SerializeEventConfig()
def serialize_event( def serialize_event(
e: Union[JsonDict, EventBase], e: Union[JsonDict, EventBase],
time_now_ms: int, time_now_ms: int,
*, *,
as_client_event: bool = True, config: SerializeEventConfig = _DEFAULT_SERIALIZE_EVENT_CONFIG,
event_format: Callable[[JsonDict], JsonDict] = format_event_for_client_v1,
token_id: Optional[str] = None,
only_event_fields: Optional[List[str]] = None,
include_stripped_room_state: bool = False,
) -> JsonDict: ) -> JsonDict:
"""Serialize event for clients """Serialize event for clients
Args: Args:
e e
time_now_ms time_now_ms
as_client_event config: Event serialization config
event_format
token_id
only_event_fields
include_stripped_room_state: Some events can have stripped room state
stored in the `unsigned` field. This is required for invite and knock
functionality. If this option is False, that state will be removed from the
event before it is returned. Otherwise, it will be kept.
Returns: Returns:
The serialized event dictionary. The serialized event dictionary.
@ -348,11 +357,11 @@ def serialize_event(
if "redacted_because" in e.unsigned: if "redacted_because" in e.unsigned:
d["unsigned"]["redacted_because"] = serialize_event( d["unsigned"]["redacted_because"] = serialize_event(
e.unsigned["redacted_because"], time_now_ms, event_format=event_format e.unsigned["redacted_because"], time_now_ms, config=config
) )
if token_id is not None: if config.token_id is not None:
if token_id == getattr(e.internal_metadata, "token_id", None): if config.token_id == getattr(e.internal_metadata, "token_id", None):
txn_id = getattr(e.internal_metadata, "txn_id", None) txn_id = getattr(e.internal_metadata, "txn_id", None)
if txn_id is not None: if txn_id is not None:
d["unsigned"]["transaction_id"] = txn_id d["unsigned"]["transaction_id"] = txn_id
@ -361,13 +370,14 @@ def serialize_event(
# that are meant to provide metadata about a room to an invitee/knocker. They are # that are meant to provide metadata about a room to an invitee/knocker. They are
# intended to only be included in specific circumstances, such as down sync, and # intended to only be included in specific circumstances, such as down sync, and
# should not be included in any other case. # should not be included in any other case.
if not include_stripped_room_state: if not config.include_stripped_room_state:
d["unsigned"].pop("invite_room_state", None) d["unsigned"].pop("invite_room_state", None)
d["unsigned"].pop("knock_room_state", None) d["unsigned"].pop("knock_room_state", None)
if as_client_event: if config.as_client_event:
d = event_format(d) d = config.event_format(d)
only_event_fields = config.only_event_fields
if only_event_fields: if only_event_fields:
if not isinstance(only_event_fields, list) or not all( if not isinstance(only_event_fields, list) or not all(
isinstance(f, str) for f in only_event_fields isinstance(f, str) for f in only_event_fields
@ -390,18 +400,18 @@ class EventClientSerializer:
event: Union[JsonDict, EventBase], event: Union[JsonDict, EventBase],
time_now: int, time_now: int,
*, *,
config: SerializeEventConfig = _DEFAULT_SERIALIZE_EVENT_CONFIG,
bundle_aggregations: Optional[Dict[str, "BundledAggregations"]] = None, bundle_aggregations: Optional[Dict[str, "BundledAggregations"]] = None,
**kwargs: Any,
) -> JsonDict: ) -> JsonDict:
"""Serializes a single event. """Serializes a single event.
Args: Args:
event: The event being serialized. event: The event being serialized.
time_now: The current time in milliseconds time_now: The current time in milliseconds
config: Event serialization config
bundle_aggregations: Whether to include the bundled aggregations for this bundle_aggregations: Whether to include the bundled aggregations for this
event. Only applies to non-state events. (State events never include event. Only applies to non-state events. (State events never include
bundled aggregations.) bundled aggregations.)
**kwargs: Arguments to pass to `serialize_event`
Returns: Returns:
The serialized event The serialized event
@ -410,7 +420,7 @@ class EventClientSerializer:
if not isinstance(event, EventBase): if not isinstance(event, EventBase):
return event return event
serialized_event = serialize_event(event, time_now, **kwargs) serialized_event = serialize_event(event, time_now, config=config)
# Check if there are any bundled aggregations to include with the event. # Check if there are any bundled aggregations to include with the event.
if bundle_aggregations: if bundle_aggregations:
@ -419,6 +429,7 @@ class EventClientSerializer:
self._inject_bundled_aggregations( self._inject_bundled_aggregations(
event, event,
time_now, time_now,
config,
bundle_aggregations[event.event_id], bundle_aggregations[event.event_id],
serialized_event, serialized_event,
) )
@ -456,6 +467,7 @@ class EventClientSerializer:
self, self,
event: EventBase, event: EventBase,
time_now: int, time_now: int,
config: SerializeEventConfig,
aggregations: "BundledAggregations", aggregations: "BundledAggregations",
serialized_event: JsonDict, serialized_event: JsonDict,
) -> None: ) -> None:
@ -466,6 +478,7 @@ class EventClientSerializer:
time_now: The current time in milliseconds time_now: The current time in milliseconds
aggregations: The bundled aggregation to serialize. aggregations: The bundled aggregation to serialize.
serialized_event: The serialized event which may be modified. serialized_event: The serialized event which may be modified.
config: Event serialization config
""" """
serialized_aggregations = {} serialized_aggregations = {}
@ -493,8 +506,8 @@ class EventClientSerializer:
thread = aggregations.thread thread = aggregations.thread
# Don't bundle aggregations as this could recurse forever. # Don't bundle aggregations as this could recurse forever.
serialized_latest_event = self.serialize_event( serialized_latest_event = serialize_event(
thread.latest_event, time_now, bundle_aggregations=None thread.latest_event, time_now, config=config
) )
# Manually apply an edit, if one exists. # Manually apply an edit, if one exists.
if thread.latest_edit: if thread.latest_edit:
@ -515,20 +528,34 @@ class EventClientSerializer:
) )
def serialize_events( def serialize_events(
self, events: Iterable[Union[JsonDict, EventBase]], time_now: int, **kwargs: Any self,
events: Iterable[Union[JsonDict, EventBase]],
time_now: int,
*,
config: SerializeEventConfig = _DEFAULT_SERIALIZE_EVENT_CONFIG,
bundle_aggregations: Optional[Dict[str, "BundledAggregations"]] = None,
) -> List[JsonDict]: ) -> List[JsonDict]:
"""Serializes multiple events. """Serializes multiple events.
Args: Args:
event event
time_now: The current time in milliseconds time_now: The current time in milliseconds
**kwargs: Arguments to pass to `serialize_event` config: Event serialization config
bundle_aggregations: Whether to include the bundled aggregations for this
event. Only applies to non-state events. (State events never include
bundled aggregations.)
Returns: Returns:
The list of serialized events The list of serialized events
""" """
return [ return [
self.serialize_event(event, time_now=time_now, **kwargs) for event in events self.serialize_event(
event,
time_now,
config=config,
bundle_aggregations=bundle_aggregations,
)
for event in events
] ]

View File

@ -19,6 +19,7 @@ from typing import TYPE_CHECKING, Iterable, List, Optional
from synapse.api.constants import EduTypes, EventTypes, Membership from synapse.api.constants import EduTypes, EventTypes, Membership
from synapse.api.errors import AuthError, SynapseError from synapse.api.errors import AuthError, SynapseError
from synapse.events import EventBase from synapse.events import EventBase
from synapse.events.utils import SerializeEventConfig
from synapse.handlers.presence import format_user_presence_state from synapse.handlers.presence import format_user_presence_state
from synapse.streams.config import PaginationConfig from synapse.streams.config import PaginationConfig
from synapse.types import JsonDict, UserID from synapse.types import JsonDict, UserID
@ -120,7 +121,7 @@ class EventStreamHandler:
chunks = self._event_serializer.serialize_events( chunks = self._event_serializer.serialize_events(
events, events,
time_now, time_now,
as_client_event=as_client_event, config=SerializeEventConfig(as_client_event=as_client_event),
) )
chunk = { chunk = {

View File

@ -18,6 +18,7 @@ from typing import TYPE_CHECKING, List, Optional, Tuple, cast
from synapse.api.constants import EduTypes, EventTypes, Membership from synapse.api.constants import EduTypes, EventTypes, Membership
from synapse.api.errors import SynapseError from synapse.api.errors import SynapseError
from synapse.events import EventBase from synapse.events import EventBase
from synapse.events.utils import SerializeEventConfig
from synapse.events.validator import EventValidator from synapse.events.validator import EventValidator
from synapse.handlers.presence import format_user_presence_state from synapse.handlers.presence import format_user_presence_state
from synapse.handlers.receipts import ReceiptEventSource from synapse.handlers.receipts import ReceiptEventSource
@ -156,6 +157,8 @@ class InitialSyncHandler:
if limit is None: if limit is None:
limit = 10 limit = 10
serializer_options = SerializeEventConfig(as_client_event=as_client_event)
async def handle_room(event: RoomsForUser) -> None: async def handle_room(event: RoomsForUser) -> None:
d: JsonDict = { d: JsonDict = {
"room_id": event.room_id, "room_id": event.room_id,
@ -173,7 +176,7 @@ class InitialSyncHandler:
d["invite"] = self._event_serializer.serialize_event( d["invite"] = self._event_serializer.serialize_event(
invite_event, invite_event,
time_now, time_now,
as_client_event=as_client_event, config=serializer_options,
) )
rooms_ret.append(d) rooms_ret.append(d)
@ -225,7 +228,7 @@ class InitialSyncHandler:
self._event_serializer.serialize_events( self._event_serializer.serialize_events(
messages, messages,
time_now=time_now, time_now=time_now,
as_client_event=as_client_event, config=serializer_options,
) )
), ),
"start": await start_token.to_string(self.store), "start": await start_token.to_string(self.store),
@ -235,7 +238,7 @@ class InitialSyncHandler:
d["state"] = self._event_serializer.serialize_events( d["state"] = self._event_serializer.serialize_events(
current_state.values(), current_state.values(),
time_now=time_now, time_now=time_now,
as_client_event=as_client_event, config=serializer_options,
) )
account_data_events = [] account_data_events = []

View File

@ -22,6 +22,7 @@ from twisted.python.failure import Failure
from synapse.api.constants import EventTypes, Membership from synapse.api.constants import EventTypes, Membership
from synapse.api.errors import SynapseError from synapse.api.errors import SynapseError
from synapse.api.filtering import Filter from synapse.api.filtering import Filter
from synapse.events.utils import SerializeEventConfig
from synapse.handlers.room import ShutdownRoomResponse from synapse.handlers.room import ShutdownRoomResponse
from synapse.metrics.background_process_metrics import run_as_background_process from synapse.metrics.background_process_metrics import run_as_background_process
from synapse.storage.state import StateFilter from synapse.storage.state import StateFilter
@ -541,13 +542,15 @@ class PaginationHandler:
time_now = self.clock.time_msec() time_now = self.clock.time_msec()
serialize_options = SerializeEventConfig(as_client_event=as_client_event)
chunk = { chunk = {
"chunk": ( "chunk": (
self._event_serializer.serialize_events( self._event_serializer.serialize_events(
events, events,
time_now, time_now,
config=serialize_options,
bundle_aggregations=aggregations, bundle_aggregations=aggregations,
as_client_event=as_client_event,
) )
), ),
"start": await from_token.to_string(self.store), "start": await from_token.to_string(self.store),
@ -556,7 +559,7 @@ class PaginationHandler:
if state: if state:
chunk["state"] = self._event_serializer.serialize_events( chunk["state"] = self._event_serializer.serialize_events(
state, time_now, as_client_event=as_client_event state, time_now, config=serialize_options
) )
return chunk return chunk

View File

@ -16,7 +16,10 @@ import logging
from typing import TYPE_CHECKING, Tuple from typing import TYPE_CHECKING, Tuple
from synapse.api.constants import ReceiptTypes from synapse.api.constants import ReceiptTypes
from synapse.events.utils import format_event_for_client_v2_without_room_id from synapse.events.utils import (
SerializeEventConfig,
format_event_for_client_v2_without_room_id,
)
from synapse.http.server import HttpServer from synapse.http.server import HttpServer
from synapse.http.servlet import RestServlet, parse_integer, parse_string from synapse.http.servlet import RestServlet, parse_integer, parse_string
from synapse.http.site import SynapseRequest from synapse.http.site import SynapseRequest
@ -75,7 +78,9 @@ class NotificationsServlet(RestServlet):
self._event_serializer.serialize_event( self._event_serializer.serialize_event(
notif_events[pa.event_id], notif_events[pa.event_id],
self.clock.time_msec(), self.clock.time_msec(),
event_format=format_event_for_client_v2_without_room_id, config=SerializeEventConfig(
event_format=format_event_for_client_v2_without_room_id
),
) )
), ),
} }

View File

@ -14,24 +14,14 @@
import itertools import itertools
import logging import logging
from collections import defaultdict from collections import defaultdict
from typing import ( from typing import TYPE_CHECKING, Any, Dict, List, Optional, Tuple, Union
TYPE_CHECKING,
Any,
Callable,
Dict,
Iterable,
List,
Optional,
Tuple,
Union,
)
from synapse.api.constants import Membership, PresenceState from synapse.api.constants import Membership, PresenceState
from synapse.api.errors import Codes, StoreError, SynapseError from synapse.api.errors import Codes, StoreError, SynapseError
from synapse.api.filtering import FilterCollection from synapse.api.filtering import FilterCollection
from synapse.api.presence import UserPresenceState from synapse.api.presence import UserPresenceState
from synapse.events import EventBase
from synapse.events.utils import ( from synapse.events.utils import (
SerializeEventConfig,
format_event_for_client_v2_without_room_id, format_event_for_client_v2_without_room_id,
format_event_raw, format_event_raw,
) )
@ -48,7 +38,6 @@ from synapse.http.server import HttpServer
from synapse.http.servlet import RestServlet, parse_boolean, parse_integer, parse_string from synapse.http.servlet import RestServlet, parse_boolean, parse_integer, parse_string
from synapse.http.site import SynapseRequest from synapse.http.site import SynapseRequest
from synapse.logging.opentracing import trace from synapse.logging.opentracing import trace
from synapse.storage.databases.main.relations import BundledAggregations
from synapse.types import JsonDict, StreamToken from synapse.types import JsonDict, StreamToken
from synapse.util import json_decoder from synapse.util import json_decoder
@ -239,28 +228,31 @@ class SyncRestServlet(RestServlet):
else: else:
raise Exception("Unknown event format %s" % (filter.event_format,)) raise Exception("Unknown event format %s" % (filter.event_format,))
serialize_options = SerializeEventConfig(
event_format=event_formatter,
token_id=access_token_id,
only_event_fields=filter.event_fields,
)
stripped_serialize_options = SerializeEventConfig(
event_format=event_formatter,
token_id=access_token_id,
include_stripped_room_state=True,
)
joined = await self.encode_joined( joined = await self.encode_joined(
sync_result.joined, sync_result.joined, time_now, serialize_options
time_now,
access_token_id,
filter.event_fields,
event_formatter,
) )
invited = await self.encode_invited( invited = await self.encode_invited(
sync_result.invited, time_now, access_token_id, event_formatter sync_result.invited, time_now, stripped_serialize_options
) )
knocked = await self.encode_knocked( knocked = await self.encode_knocked(
sync_result.knocked, time_now, access_token_id, event_formatter sync_result.knocked, time_now, stripped_serialize_options
) )
archived = await self.encode_archived( archived = await self.encode_archived(
sync_result.archived, sync_result.archived, time_now, serialize_options
time_now,
access_token_id,
filter.event_fields,
event_formatter,
) )
logger.debug("building sync response dict") logger.debug("building sync response dict")
@ -339,9 +331,7 @@ class SyncRestServlet(RestServlet):
self, self,
rooms: List[JoinedSyncResult], rooms: List[JoinedSyncResult],
time_now: int, time_now: int,
token_id: Optional[int], serialize_options: SerializeEventConfig,
event_fields: List[str],
event_formatter: Callable[[JsonDict], JsonDict],
) -> JsonDict: ) -> JsonDict:
""" """
Encode the joined rooms in a sync result Encode the joined rooms in a sync result
@ -349,24 +339,14 @@ class SyncRestServlet(RestServlet):
Args: Args:
rooms: list of sync results for rooms this user is joined to rooms: list of sync results for rooms this user is joined to
time_now: current time - used as a baseline for age calculations time_now: current time - used as a baseline for age calculations
token_id: ID of the user's auth token - used for namespacing serialize_options: Event serializer options
of transaction IDs
event_fields: List of event fields to include. If empty,
all fields will be returned.
event_formatter: function to convert from federation format
to client format
Returns: Returns:
The joined rooms list, in our response format The joined rooms list, in our response format
""" """
joined = {} joined = {}
for room in rooms: for room in rooms:
joined[room.room_id] = await self.encode_room( joined[room.room_id] = await self.encode_room(
room, room, time_now, joined=True, serialize_options=serialize_options
time_now,
token_id,
joined=True,
only_fields=event_fields,
event_formatter=event_formatter,
) )
return joined return joined
@ -376,8 +356,7 @@ class SyncRestServlet(RestServlet):
self, self,
rooms: List[InvitedSyncResult], rooms: List[InvitedSyncResult],
time_now: int, time_now: int,
token_id: Optional[int], serialize_options: SerializeEventConfig,
event_formatter: Callable[[JsonDict], JsonDict],
) -> JsonDict: ) -> JsonDict:
""" """
Encode the invited rooms in a sync result Encode the invited rooms in a sync result
@ -385,10 +364,7 @@ class SyncRestServlet(RestServlet):
Args: Args:
rooms: list of sync results for rooms this user is invited to rooms: list of sync results for rooms this user is invited to
time_now: current time - used as a baseline for age calculations time_now: current time - used as a baseline for age calculations
token_id: ID of the user's auth token - used for namespacing serialize_options: Event serializer options
of transaction IDs
event_formatter: function to convert from federation format
to client format
Returns: Returns:
The invited rooms list, in our response format The invited rooms list, in our response format
@ -396,11 +372,7 @@ class SyncRestServlet(RestServlet):
invited = {} invited = {}
for room in rooms: for room in rooms:
invite = self._event_serializer.serialize_event( invite = self._event_serializer.serialize_event(
room.invite, room.invite, time_now, config=serialize_options
time_now,
token_id=token_id,
event_format=event_formatter,
include_stripped_room_state=True,
) )
unsigned = dict(invite.get("unsigned", {})) unsigned = dict(invite.get("unsigned", {}))
invite["unsigned"] = unsigned invite["unsigned"] = unsigned
@ -415,8 +387,7 @@ class SyncRestServlet(RestServlet):
self, self,
rooms: List[KnockedSyncResult], rooms: List[KnockedSyncResult],
time_now: int, time_now: int,
token_id: Optional[int], serialize_options: SerializeEventConfig,
event_formatter: Callable[[Dict], Dict],
) -> Dict[str, Dict[str, Any]]: ) -> Dict[str, Dict[str, Any]]:
""" """
Encode the rooms we've knocked on in a sync result. Encode the rooms we've knocked on in a sync result.
@ -424,8 +395,7 @@ class SyncRestServlet(RestServlet):
Args: Args:
rooms: list of sync results for rooms this user is knocking on rooms: list of sync results for rooms this user is knocking on
time_now: current time - used as a baseline for age calculations time_now: current time - used as a baseline for age calculations
token_id: ID of the user's auth token - used for namespacing of transaction IDs serialize_options: Event serializer options
event_formatter: function to convert from federation format to client format
Returns: Returns:
The list of rooms the user has knocked on, in our response format. The list of rooms the user has knocked on, in our response format.
@ -433,11 +403,7 @@ class SyncRestServlet(RestServlet):
knocked = {} knocked = {}
for room in rooms: for room in rooms:
knock = self._event_serializer.serialize_event( knock = self._event_serializer.serialize_event(
room.knock, room.knock, time_now, config=serialize_options
time_now,
token_id=token_id,
event_format=event_formatter,
include_stripped_room_state=True,
) )
# Extract the `unsigned` key from the knock event. # Extract the `unsigned` key from the knock event.
@ -470,9 +436,7 @@ class SyncRestServlet(RestServlet):
self, self,
rooms: List[ArchivedSyncResult], rooms: List[ArchivedSyncResult],
time_now: int, time_now: int,
token_id: Optional[int], serialize_options: SerializeEventConfig,
event_fields: List[str],
event_formatter: Callable[[JsonDict], JsonDict],
) -> JsonDict: ) -> JsonDict:
""" """
Encode the archived rooms in a sync result Encode the archived rooms in a sync result
@ -480,23 +444,14 @@ class SyncRestServlet(RestServlet):
Args: Args:
rooms: list of sync results for rooms this user is joined to rooms: list of sync results for rooms this user is joined to
time_now: current time - used as a baseline for age calculations time_now: current time - used as a baseline for age calculations
token_id: ID of the user's auth token - used for namespacing serialize_options: Event serializer options
of transaction IDs
event_fields: List of event fields to include. If empty,
all fields will be returned.
event_formatter: function to convert from federation format to client format
Returns: Returns:
The archived rooms list, in our response format The archived rooms list, in our response format
""" """
joined = {} joined = {}
for room in rooms: for room in rooms:
joined[room.room_id] = await self.encode_room( joined[room.room_id] = await self.encode_room(
room, room, time_now, joined=False, serialize_options=serialize_options
time_now,
token_id,
joined=False,
only_fields=event_fields,
event_formatter=event_formatter,
) )
return joined return joined
@ -505,10 +460,8 @@ class SyncRestServlet(RestServlet):
self, self,
room: Union[JoinedSyncResult, ArchivedSyncResult], room: Union[JoinedSyncResult, ArchivedSyncResult],
time_now: int, time_now: int,
token_id: Optional[int],
joined: bool, joined: bool,
only_fields: Optional[List[str]], serialize_options: SerializeEventConfig,
event_formatter: Callable[[JsonDict], JsonDict],
) -> JsonDict: ) -> JsonDict:
""" """
Args: Args:
@ -524,20 +477,6 @@ class SyncRestServlet(RestServlet):
Returns: Returns:
The room, encoded in our response format The room, encoded in our response format
""" """
def serialize(
events: Iterable[EventBase],
aggregations: Optional[Dict[str, BundledAggregations]] = None,
) -> List[JsonDict]:
return self._event_serializer.serialize_events(
events,
time_now=time_now,
bundle_aggregations=aggregations,
token_id=token_id,
event_format=event_formatter,
only_event_fields=only_fields,
)
state_dict = room.state state_dict = room.state
timeline_events = room.timeline.events timeline_events = room.timeline.events
@ -554,9 +493,14 @@ class SyncRestServlet(RestServlet):
event.room_id, event.room_id,
) )
serialized_state = serialize(state_events) serialized_state = self._event_serializer.serialize_events(
serialized_timeline = serialize( state_events, time_now, config=serialize_options
timeline_events, room.timeline.bundled_aggregations )
serialized_timeline = self._event_serializer.serialize_events(
timeline_events,
time_now,
config=serialize_options,
bundle_aggregations=room.timeline.bundled_aggregations,
) )
account_data = room.account_data account_data = room.account_data

View File

@ -16,6 +16,7 @@ from synapse.api.constants import EventContentFields
from synapse.api.room_versions import RoomVersions from synapse.api.room_versions import RoomVersions
from synapse.events import make_event_from_dict from synapse.events import make_event_from_dict
from synapse.events.utils import ( from synapse.events.utils import (
SerializeEventConfig,
copy_power_levels_contents, copy_power_levels_contents,
prune_event, prune_event,
serialize_event, serialize_event,
@ -392,7 +393,9 @@ class PruneEventTestCase(unittest.TestCase):
class SerializeEventTestCase(unittest.TestCase): class SerializeEventTestCase(unittest.TestCase):
def serialize(self, ev, fields): def serialize(self, ev, fields):
return serialize_event(ev, 1479807801915, only_event_fields=fields) return serialize_event(
ev, 1479807801915, config=SerializeEventConfig(only_event_fields=fields)
)
def test_event_fields_works_with_keys(self): def test_event_fields_works_with_keys(self):
self.assertEqual( self.assertEqual(

View File

@ -704,10 +704,8 @@ class RelationsTestCase(BaseRelationsTestCase):
} }
}, },
"event_id": thread_2, "event_id": thread_2,
"room_id": self.room,
"sender": self.user_id, "sender": self.user_id,
"type": "m.room.test", "type": "m.room.test",
"user_id": self.user_id,
}, },
relations_dict[RelationTypes.THREAD].get("latest_event"), relations_dict[RelationTypes.THREAD].get("latest_event"),
) )