From e8ba0c038178fa756f65f0361b071b83ffb32a51 Mon Sep 17 00:00:00 2001 From: Nick Mills-Barrett Date: Wed, 2 Aug 2023 14:00:58 +0100 Subject: [PATCH 1/3] Add linearizer on user ID to push rule PUT/DELETE requests --- synapse/rest/client/push_rule.py | 28 ++++++++++++++++++++++------ 1 file changed, 22 insertions(+), 6 deletions(-) diff --git a/synapse/rest/client/push_rule.py b/synapse/rest/client/push_rule.py index 5c9fece3ba33..5ed3b83a03e2 100644 --- a/synapse/rest/client/push_rule.py +++ b/synapse/rest/client/push_rule.py @@ -32,6 +32,7 @@ from synapse.rest.client._base import client_patterns from synapse.storage.push_rule import InconsistentRuleException, RuleNotFoundException from synapse.types import JsonDict +from synapse.util.async_helpers import Linearizer if TYPE_CHECKING: from synapse.server import HomeServer @@ -53,26 +54,32 @@ def __init__(self, hs: "HomeServer"): self.notifier = hs.get_notifier() self._is_worker = hs.config.worker.worker_app is not None self._push_rules_handler = hs.get_push_rules_handler() + self._push_rule_linearizer = Linearizer(name="push_rules") async def on_PUT(self, request: SynapseRequest, path: str) -> Tuple[int, JsonDict]: if self._is_worker: raise Exception("Cannot handle PUT /push_rules on worker") + requester = await self.auth.get_user_by_req(request) + user_id = requester.user.to_string() + + async with self._push_rule_linearizer.queue(user_id): + return await self.handle_put(request, path, user_id) + + async def handle_put( + self, request: SynapseRequest, path: str, user_id: str + ) -> Tuple[int, JsonDict]: spec = _rule_spec_from_path(path.split("/")) try: priority_class = _priority_class_from_spec(spec) except InvalidRuleException as e: raise SynapseError(400, str(e)) - requester = await self.auth.get_user_by_req(request) - if "/" in spec.rule_id or "\\" in spec.rule_id: raise SynapseError(400, "rule_id may not contain slashes") content = parse_json_value_from_request(request) - user_id = requester.user.to_string() - if spec.attr: try: await self._push_rules_handler.set_rule_attr(user_id, spec, content) @@ -126,11 +133,20 @@ async def on_DELETE( if self._is_worker: raise Exception("Cannot handle DELETE /push_rules on worker") - spec = _rule_spec_from_path(path.split("/")) - requester = await self.auth.get_user_by_req(request) user_id = requester.user.to_string() + async with self._push_rule_linearizer.queue(user_id): + return await self.handle_delete(request, path, user_id) + + async def handle_delete( + self, + request: SynapseRequest, + path: str, + user_id: str, + ) -> Tuple[int, JsonDict]: + spec = _rule_spec_from_path(path.split("/")) + namespaced_rule_id = f"global/{spec.template}/{spec.rule_id}" try: From 8c0c38aa239006f8131dc567da83c16640159562 Mon Sep 17 00:00:00 2001 From: Nick Mills-Barrett Date: Wed, 2 Aug 2023 14:08:13 +0100 Subject: [PATCH 2/3] Add changelog --- changelog.d/16052.bugfix | 1 + 1 file changed, 1 insertion(+) create mode 100644 changelog.d/16052.bugfix diff --git a/changelog.d/16052.bugfix b/changelog.d/16052.bugfix new file mode 100644 index 000000000000..440d3b08ccb5 --- /dev/null +++ b/changelog.d/16052.bugfix @@ -0,0 +1 @@ +Linearize push rule modification endpoints to prevent deadlocks. Contributed by Nick @ Beeper (@fizzadar). From 50504a5f5bde8dde30c495b3e205c5e91c67a168 Mon Sep 17 00:00:00 2001 From: Nick Mills-Barrett Date: Fri, 11 Aug 2023 11:50:08 +0100 Subject: [PATCH 3/3] Update changelog.d/16052.bugfix Co-authored-by: Erik Johnston --- changelog.d/16052.bugfix | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/changelog.d/16052.bugfix b/changelog.d/16052.bugfix index 440d3b08ccb5..3c7a60f226a7 100644 --- a/changelog.d/16052.bugfix +++ b/changelog.d/16052.bugfix @@ -1 +1 @@ -Linearize push rule modification endpoints to prevent deadlocks. Contributed by Nick @ Beeper (@fizzadar). +Fix long-standing bug where concurrent requests to change a user's push rules could cause a deadlock. Contributed by Nick @ Beeper (@fizzadar).