Don't pull out the full state when storing state (#13274)

This commit is contained in:
Erik Johnston 2022-07-15 13:59:45 +01:00 committed by GitHub
parent 3343035a06
commit 0731e0829c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 130 additions and 69 deletions

1
changelog.d/13274.misc Normal file
View File

@ -0,0 +1 @@
Don't pull out state in `compute_event_context` for unconflicted state.

View File

@ -298,12 +298,18 @@ class StateHandler:
state_group_before_event_prev_group = entry.prev_group state_group_before_event_prev_group = entry.prev_group
deltas_to_state_group_before_event = entry.delta_ids deltas_to_state_group_before_event = entry.delta_ids
state_ids_before_event = None
# We make sure that we have a state group assigned to the state. # We make sure that we have a state group assigned to the state.
if entry.state_group is None: if entry.state_group is None:
# store_state_group requires us to have either a previous state group
# (with deltas) or the complete state map. So, if we don't have a
# previous state group, load the complete state map now.
if state_group_before_event_prev_group is None:
state_ids_before_event = await entry.get_state( state_ids_before_event = await entry.get_state(
self._state_storage_controller, StateFilter.all() self._state_storage_controller, StateFilter.all()
) )
state_group_before_event = ( state_group_before_event = (
await self._state_storage_controller.store_state_group( await self._state_storage_controller.store_state_group(
event.event_id, event.event_id,
@ -316,7 +322,6 @@ class StateHandler:
entry.state_group = state_group_before_event entry.state_group = state_group_before_event
else: else:
state_group_before_event = entry.state_group state_group_before_event = entry.state_group
state_ids_before_event = None
# #
# now if it's not a state event, we're done # now if it's not a state event, we're done
@ -336,19 +341,20 @@ class StateHandler:
# #
# otherwise, we'll need to create a new state group for after the event # otherwise, we'll need to create a new state group for after the event
# #
if state_ids_before_event is None:
state_ids_before_event = await entry.get_state(
self._state_storage_controller, StateFilter.all()
)
key = (event.type, event.state_key) key = (event.type, event.state_key)
if key in state_ids_before_event:
replaces = state_ids_before_event[key] if state_ids_before_event is not None:
if replaces != event.event_id: replaces = state_ids_before_event.get(key)
else:
replaces_state_map = await entry.get_state(
self._state_storage_controller, StateFilter.from_types([key])
)
replaces = replaces_state_map.get(key)
if replaces and replaces != event.event_id:
event.unsigned["replaces_state"] = replaces event.unsigned["replaces_state"] = replaces
state_ids_after_event = dict(state_ids_before_event)
state_ids_after_event[key] = event.event_id
delta_ids = {key: event.event_id} delta_ids = {key: event.event_id}
state_group_after_event = ( state_group_after_event = (
@ -357,7 +363,7 @@ class StateHandler:
event.room_id, event.room_id,
prev_group=state_group_before_event, prev_group=state_group_before_event,
delta_ids=delta_ids, delta_ids=delta_ids,
current_state_ids=state_ids_after_event, current_state_ids=None,
) )
) )

View File

@ -346,7 +346,7 @@ class StateStorageController:
room_id: str, room_id: str,
prev_group: Optional[int], prev_group: Optional[int],
delta_ids: Optional[StateMap[str]], delta_ids: Optional[StateMap[str]],
current_state_ids: StateMap[str], current_state_ids: Optional[StateMap[str]],
) -> int: ) -> int:
"""Store a new set of state, returning a newly assigned state group. """Store a new set of state, returning a newly assigned state group.

View File

@ -400,14 +400,17 @@ class StateGroupDataStore(StateBackgroundUpdateStore, SQLBaseStore):
room_id: str, room_id: str,
prev_group: Optional[int], prev_group: Optional[int],
delta_ids: Optional[StateMap[str]], delta_ids: Optional[StateMap[str]],
current_state_ids: StateMap[str], current_state_ids: Optional[StateMap[str]],
) -> int: ) -> int:
"""Store a new set of state, returning a newly assigned state group. """Store a new set of state, returning a newly assigned state group.
At least one of `current_state_ids` and `prev_group` must be provided. Whenever
`prev_group` is not None, `delta_ids` must also not be None.
Args: Args:
event_id: The event ID for which the state was calculated event_id: The event ID for which the state was calculated
room_id room_id
prev_group: A previous state group for the room, optional. prev_group: A previous state group for the room.
delta_ids: The delta between state at `prev_group` and delta_ids: The delta between state at `prev_group` and
`current_state_ids`, if `prev_group` was given. Same format as `current_state_ids`, if `prev_group` was given. Same format as
`current_state_ids`. `current_state_ids`.
@ -418,22 +421,23 @@ class StateGroupDataStore(StateBackgroundUpdateStore, SQLBaseStore):
The state group ID The state group ID
""" """
def _store_state_group_txn(txn: LoggingTransaction) -> int: if prev_group is None and current_state_ids is None:
if current_state_ids is None: raise Exception("current_state_ids and prev_group can't both be None")
# AFAIK, this can never happen
raise Exception("current_state_ids cannot be None")
state_group = self._state_group_seq_gen.get_next_id_txn(txn) if prev_group is not None and delta_ids is None:
raise Exception("delta_ids is None when prev_group is not None")
self.db_pool.simple_insert_txn( def insert_delta_group_txn(
txn, txn: LoggingTransaction, prev_group: int, delta_ids: StateMap[str]
table="state_groups", ) -> Optional[int]:
values={"id": state_group, "room_id": room_id, "event_id": event_id}, """Try and persist the new group as a delta.
)
# We persist as a delta if we can, while also ensuring the chain Requires that we have the state as a delta from a previous state group.
# of deltas isn't tooo long, as otherwise read performance degrades.
if prev_group: Returns:
The state group if successfully created, or None if the state
needs to be persisted as a full state.
"""
is_in_db = self.db_pool.simple_select_one_onecol_txn( is_in_db = self.db_pool.simple_select_one_onecol_txn(
txn, txn,
table="state_groups", table="state_groups",
@ -447,9 +451,19 @@ class StateGroupDataStore(StateBackgroundUpdateStore, SQLBaseStore):
% (prev_group,) % (prev_group,)
) )
# if the chain of state group deltas is going too long, we fall back to
# persisting a complete state group.
potential_hops = self._count_state_group_hops_txn(txn, prev_group) potential_hops = self._count_state_group_hops_txn(txn, prev_group)
if prev_group and potential_hops < MAX_STATE_DELTA_HOPS: if potential_hops >= MAX_STATE_DELTA_HOPS:
assert delta_ids is not None return None
state_group = self._state_group_seq_gen.get_next_id_txn(txn)
self.db_pool.simple_insert_txn(
txn,
table="state_groups",
values={"id": state_group, "room_id": room_id, "event_id": event_id},
)
self.db_pool.simple_insert_txn( self.db_pool.simple_insert_txn(
txn, txn,
@ -466,7 +480,21 @@ class StateGroupDataStore(StateBackgroundUpdateStore, SQLBaseStore):
for key, state_id in delta_ids.items() for key, state_id in delta_ids.items()
], ],
) )
else:
return state_group
def insert_full_state_txn(
txn: LoggingTransaction, current_state_ids: StateMap[str]
) -> int:
"""Persist the full state, returning the new state group."""
state_group = self._state_group_seq_gen.get_next_id_txn(txn)
self.db_pool.simple_insert_txn(
txn,
table="state_groups",
values={"id": state_group, "room_id": room_id, "event_id": event_id},
)
self.db_pool.simple_insert_many_txn( self.db_pool.simple_insert_many_txn(
txn, txn,
table="state_groups_state", table="state_groups_state",
@ -491,7 +519,7 @@ class StateGroupDataStore(StateBackgroundUpdateStore, SQLBaseStore):
self._state_group_members_cache.update, self._state_group_members_cache.update,
self._state_group_members_cache.sequence, self._state_group_members_cache.sequence,
key=state_group, key=state_group,
value=dict(current_member_state_ids), value=current_member_state_ids,
) )
current_non_member_state_ids = { current_non_member_state_ids = {
@ -503,13 +531,35 @@ class StateGroupDataStore(StateBackgroundUpdateStore, SQLBaseStore):
self._state_group_cache.update, self._state_group_cache.update,
self._state_group_cache.sequence, self._state_group_cache.sequence,
key=state_group, key=state_group,
value=dict(current_non_member_state_ids), value=current_non_member_state_ids,
) )
return state_group return state_group
if prev_group is not None:
state_group = await self.db_pool.runInteraction(
"store_state_group.insert_delta_group",
insert_delta_group_txn,
prev_group,
delta_ids,
)
if state_group is not None:
return state_group
# We're going to persist the state as a complete group rather than
# a delta, so first we need to ensure we have loaded the state map
# from the database.
if current_state_ids is None:
assert prev_group is not None
assert delta_ids is not None
groups = await self._get_state_for_groups([prev_group])
current_state_ids = dict(groups[prev_group])
current_state_ids.update(delta_ids)
return await self.db_pool.runInteraction( return await self.db_pool.runInteraction(
"store_state_group", _store_state_group_txn "store_state_group.insert_full_state",
insert_full_state_txn,
current_state_ids,
) )
async def purge_unreferenced_state_groups( async def purge_unreferenced_state_groups(

View File

@ -709,7 +709,7 @@ class RoomsCreateTestCase(RoomBase):
self.assertEqual(200, channel.code, channel.result) self.assertEqual(200, channel.code, channel.result)
self.assertTrue("room_id" in channel.json_body) self.assertTrue("room_id" in channel.json_body)
assert channel.resource_usage is not None assert channel.resource_usage is not None
self.assertEqual(32, channel.resource_usage.db_txn_count) self.assertEqual(36, channel.resource_usage.db_txn_count)
def test_post_room_initial_state(self) -> None: def test_post_room_initial_state(self) -> None:
# POST with initial_state config key, expect new room id # POST with initial_state config key, expect new room id
@ -722,7 +722,7 @@ class RoomsCreateTestCase(RoomBase):
self.assertEqual(200, channel.code, channel.result) self.assertEqual(200, channel.code, channel.result)
self.assertTrue("room_id" in channel.json_body) self.assertTrue("room_id" in channel.json_body)
assert channel.resource_usage is not None assert channel.resource_usage is not None
self.assertEqual(35, channel.resource_usage.db_txn_count) self.assertEqual(40, channel.resource_usage.db_txn_count)
def test_post_room_visibility_key(self) -> None: def test_post_room_visibility_key(self) -> None:
# POST with visibility config key, expect new room id # POST with visibility config key, expect new room id

View File

@ -99,6 +99,10 @@ class _DummyStore:
state_group = self._next_group state_group = self._next_group
self._next_group += 1 self._next_group += 1
if current_state_ids is None:
current_state_ids = dict(self._group_to_state[prev_group])
current_state_ids.update(delta_ids)
self._group_to_state[state_group] = dict(current_state_ids) self._group_to_state[state_group] = dict(current_state_ids)
return state_group return state_group