mirror of
https://mau.dev/maunium/synapse.git
synced 2024-12-13 17:23:21 +01:00
Add more tests for in-flight state query duplication. (#12033)
This commit is contained in:
parent
af2c1e3d2a
commit
546b9c9e64
2 changed files with 174 additions and 23 deletions
1
changelog.d/12033.misc
Normal file
1
changelog.d/12033.misc
Normal file
|
@ -0,0 +1 @@
|
|||
Deduplicate in-flight requests in `_get_state_for_groups`.
|
|
@ -18,8 +18,9 @@ from unittest.mock import patch
|
|||
from twisted.internet.defer import Deferred, ensureDeferred
|
||||
from twisted.test.proto_helpers import MemoryReactor
|
||||
|
||||
from synapse.api.constants import EventTypes
|
||||
from synapse.storage.state import StateFilter
|
||||
from synapse.types import MutableStateMap, StateMap
|
||||
from synapse.types import StateMap
|
||||
from synapse.util import Clock
|
||||
|
||||
from tests.unittest import HomeserverTestCase
|
||||
|
@ -27,6 +28,21 @@ from tests.unittest import HomeserverTestCase
|
|||
if typing.TYPE_CHECKING:
|
||||
from synapse.server import HomeServer
|
||||
|
||||
# StateFilter for ALL non-m.room.member state events
|
||||
ALL_NON_MEMBERS_STATE_FILTER = StateFilter.freeze(
|
||||
types={EventTypes.Member: set()},
|
||||
include_others=True,
|
||||
)
|
||||
|
||||
FAKE_STATE = {
|
||||
(EventTypes.Member, "@alice:test"): "join",
|
||||
(EventTypes.Member, "@bob:test"): "leave",
|
||||
(EventTypes.Member, "@charlie:test"): "invite",
|
||||
("test.type", "a"): "AAA",
|
||||
("test.type", "b"): "BBB",
|
||||
("other.event.type", "state.key"): "123",
|
||||
}
|
||||
|
||||
|
||||
class StateGroupInflightCachingTestCase(HomeserverTestCase):
|
||||
def prepare(
|
||||
|
@ -65,24 +81,8 @@ class StateGroupInflightCachingTestCase(HomeserverTestCase):
|
|||
Assemble a fake database response and complete the database request.
|
||||
"""
|
||||
|
||||
result: Dict[int, StateMap[str]] = {}
|
||||
|
||||
for group in groups:
|
||||
group_result: MutableStateMap[str] = {}
|
||||
result[group] = group_result
|
||||
|
||||
for state_type, state_keys in state_filter.types.items():
|
||||
if state_keys is None:
|
||||
group_result[(state_type, "a")] = "xyz"
|
||||
group_result[(state_type, "b")] = "xyz"
|
||||
else:
|
||||
for state_key in state_keys:
|
||||
group_result[(state_type, state_key)] = "abc"
|
||||
|
||||
if state_filter.include_others:
|
||||
group_result[("other.event.type", "state.key")] = "123"
|
||||
|
||||
d.callback(result)
|
||||
# Return a filtered copy of the fake state
|
||||
d.callback({group: state_filter.filter_state(FAKE_STATE) for group in groups})
|
||||
|
||||
def test_duplicate_requests_deduplicated(self) -> None:
|
||||
"""
|
||||
|
@ -125,9 +125,159 @@ class StateGroupInflightCachingTestCase(HomeserverTestCase):
|
|||
# Now we can complete the request
|
||||
self._complete_request_fake(groups, sf, d)
|
||||
|
||||
self.assertEqual(
|
||||
self.get_success(req1), {("other.event.type", "state.key"): "123"}
|
||||
self.assertEqual(self.get_success(req1), FAKE_STATE)
|
||||
self.assertEqual(self.get_success(req2), FAKE_STATE)
|
||||
|
||||
def test_smaller_request_deduplicated(self) -> None:
|
||||
"""
|
||||
Tests that duplicate requests for state are deduplicated.
|
||||
|
||||
This test:
|
||||
- requests some state (state group 42, 'all' state filter)
|
||||
- requests a subset of that state, before the first request finishes
|
||||
- checks to see that only one database query was made
|
||||
- completes the database query
|
||||
- checks that both requests see the correct retrieved state
|
||||
"""
|
||||
req1 = ensureDeferred(
|
||||
self.state_datastore._get_state_for_group_using_inflight_cache(
|
||||
42, StateFilter.from_types((("test.type", None),))
|
||||
)
|
||||
)
|
||||
self.assertEqual(
|
||||
self.get_success(req2), {("other.event.type", "state.key"): "123"}
|
||||
self.pump(by=0.1)
|
||||
|
||||
# This should have gone to the database
|
||||
self.assertEqual(len(self.get_state_group_calls), 1)
|
||||
self.assertFalse(req1.called)
|
||||
|
||||
req2 = ensureDeferred(
|
||||
self.state_datastore._get_state_for_group_using_inflight_cache(
|
||||
42, StateFilter.from_types((("test.type", "b"),))
|
||||
)
|
||||
)
|
||||
self.pump(by=0.1)
|
||||
|
||||
# No more calls should have gone to the database, because the second
|
||||
# request was already in the in-flight cache!
|
||||
self.assertEqual(len(self.get_state_group_calls), 1)
|
||||
self.assertFalse(req1.called)
|
||||
self.assertFalse(req2.called)
|
||||
|
||||
groups, sf, d = self.get_state_group_calls[0]
|
||||
self.assertEqual(groups, (42,))
|
||||
# The state filter is expanded internally for increased cache hit rate,
|
||||
# so we the database sees a wider state filter than requested.
|
||||
self.assertEqual(sf, ALL_NON_MEMBERS_STATE_FILTER)
|
||||
|
||||
# Now we can complete the request
|
||||
self._complete_request_fake(groups, sf, d)
|
||||
|
||||
self.assertEqual(
|
||||
self.get_success(req1),
|
||||
{("test.type", "a"): "AAA", ("test.type", "b"): "BBB"},
|
||||
)
|
||||
self.assertEqual(self.get_success(req2), {("test.type", "b"): "BBB"})
|
||||
|
||||
def test_partially_overlapping_request_deduplicated(self) -> None:
|
||||
"""
|
||||
Tests that partially-overlapping requests are partially deduplicated.
|
||||
|
||||
This test:
|
||||
- requests a single type of wildcard state
|
||||
(This is internally expanded to be all non-member state)
|
||||
- requests the entire state in parallel
|
||||
- checks to see that two database queries were made, but that the second
|
||||
one is only for member state.
|
||||
- completes the database queries
|
||||
- checks that both requests have the correct result.
|
||||
"""
|
||||
|
||||
req1 = ensureDeferred(
|
||||
self.state_datastore._get_state_for_group_using_inflight_cache(
|
||||
42, StateFilter.from_types((("test.type", None),))
|
||||
)
|
||||
)
|
||||
self.pump(by=0.1)
|
||||
|
||||
# This should have gone to the database
|
||||
self.assertEqual(len(self.get_state_group_calls), 1)
|
||||
self.assertFalse(req1.called)
|
||||
|
||||
req2 = ensureDeferred(
|
||||
self.state_datastore._get_state_for_group_using_inflight_cache(
|
||||
42, StateFilter.all()
|
||||
)
|
||||
)
|
||||
self.pump(by=0.1)
|
||||
|
||||
# Because it only partially overlaps, this also went to the database
|
||||
self.assertEqual(len(self.get_state_group_calls), 2)
|
||||
self.assertFalse(req1.called)
|
||||
self.assertFalse(req2.called)
|
||||
|
||||
# First request:
|
||||
groups, sf, d = self.get_state_group_calls[0]
|
||||
self.assertEqual(groups, (42,))
|
||||
# The state filter is expanded internally for increased cache hit rate,
|
||||
# so we the database sees a wider state filter than requested.
|
||||
self.assertEqual(sf, ALL_NON_MEMBERS_STATE_FILTER)
|
||||
self._complete_request_fake(groups, sf, d)
|
||||
|
||||
# Second request:
|
||||
groups, sf, d = self.get_state_group_calls[1]
|
||||
self.assertEqual(groups, (42,))
|
||||
# The state filter is narrowed to only request membership state, because
|
||||
# the remainder of the state is already being queried in the first request!
|
||||
self.assertEqual(
|
||||
sf, StateFilter.freeze({EventTypes.Member: None}, include_others=False)
|
||||
)
|
||||
self._complete_request_fake(groups, sf, d)
|
||||
|
||||
# Check the results are correct
|
||||
self.assertEqual(
|
||||
self.get_success(req1),
|
||||
{("test.type", "a"): "AAA", ("test.type", "b"): "BBB"},
|
||||
)
|
||||
self.assertEqual(self.get_success(req2), FAKE_STATE)
|
||||
|
||||
def test_in_flight_requests_stop_being_in_flight(self) -> None:
|
||||
"""
|
||||
Tests that in-flight request deduplication doesn't somehow 'hold on'
|
||||
to completed requests: once they're done, they're taken out of the
|
||||
in-flight cache.
|
||||
"""
|
||||
req1 = ensureDeferred(
|
||||
self.state_datastore._get_state_for_group_using_inflight_cache(
|
||||
42, StateFilter.all()
|
||||
)
|
||||
)
|
||||
self.pump(by=0.1)
|
||||
|
||||
# This should have gone to the database
|
||||
self.assertEqual(len(self.get_state_group_calls), 1)
|
||||
self.assertFalse(req1.called)
|
||||
|
||||
# Complete the request right away.
|
||||
self._complete_request_fake(*self.get_state_group_calls[0])
|
||||
self.assertTrue(req1.called)
|
||||
|
||||
# Send off another request
|
||||
req2 = ensureDeferred(
|
||||
self.state_datastore._get_state_for_group_using_inflight_cache(
|
||||
42, StateFilter.all()
|
||||
)
|
||||
)
|
||||
self.pump(by=0.1)
|
||||
|
||||
# It should have gone to the database again, because the previous request
|
||||
# isn't in-flight and therefore isn't available for deduplication.
|
||||
self.assertEqual(len(self.get_state_group_calls), 2)
|
||||
self.assertFalse(req2.called)
|
||||
|
||||
# Complete the request right away.
|
||||
self._complete_request_fake(*self.get_state_group_calls[1])
|
||||
self.assertTrue(req2.called)
|
||||
groups, sf, d = self.get_state_group_calls[0]
|
||||
|
||||
self.assertEqual(self.get_success(req1), FAKE_STATE)
|
||||
self.assertEqual(self.get_success(req2), FAKE_STATE)
|
||||
|
|
Loading…
Reference in a new issue