From e5c512bde1a0efaffd742bfd18dc49a0898154a3 Mon Sep 17 00:00:00 2001 From: Patrick Cloke Date: Tue, 20 Feb 2024 12:07:55 -0500 Subject: [PATCH 1/4] Stabilize support for Retry-After header (MSC4014) --- synapse/api/errors.py | 3 ++- synapse/config/experimental.py | 9 --------- tests/api/test_errors.py | 8 ++------ 3 files changed, 4 insertions(+), 16 deletions(-) diff --git a/synapse/api/errors.py b/synapse/api/errors.py index b44088f9b3e..7d9f691677d 100644 --- a/synapse/api/errors.py +++ b/synapse/api/errors.py @@ -526,9 +526,10 @@ def __init__( retry_after_ms: Optional[int] = None, errcode: str = Codes.LIMIT_EXCEEDED, ): + # Use HTTP header Retry-After to enable library-assisted retry handling. headers = ( {"Retry-After": str(math.ceil(retry_after_ms / 1000))} - if self.include_retry_after_header and retry_after_ms is not None + if retry_after_ms is not None else None ) super().__init__(code, "Too Many Requests", errcode, headers=headers) diff --git a/synapse/config/experimental.py b/synapse/config/experimental.py index d43d9da956c..0bd3befdc2e 100644 --- a/synapse/config/experimental.py +++ b/synapse/config/experimental.py @@ -25,7 +25,6 @@ import attr import attr.validators -from synapse.api.errors import LimitExceededError from synapse.api.room_versions import KNOWN_ROOM_VERSIONS, RoomVersions from synapse.config import ConfigError from synapse.config._base import Config, RootConfig @@ -415,14 +414,6 @@ def read_config(self, config: JsonDict, **kwargs: Any) -> None: "msc4010_push_rules_account_data", False ) - # MSC4041: Use HTTP header Retry-After to enable library-assisted retry handling - # - # This is a bit hacky, but the most reasonable way to *alway* include the - # headers. - LimitExceededError.include_retry_after_header = experimental.get( - "msc4041_enabled", False - ) - self.msc4028_push_encrypted_events = experimental.get( "msc4028_push_encrypted_events", False ) diff --git a/tests/api/test_errors.py b/tests/api/test_errors.py index 25fa93b9d86..efa3addf009 100644 --- a/tests/api/test_errors.py +++ b/tests/api/test_errors.py @@ -33,18 +33,14 @@ def test_key_appears_in_context_but_not_error_dict(self) -> None: self.assertIn("needle", err.debug_context) self.assertNotIn("needle", serialised) - # Create a sub-class to avoid mutating the class-level property. - class LimitExceededErrorHeaders(LimitExceededError): - include_retry_after_header = True - def test_limit_exceeded_header(self) -> None: - err = self.LimitExceededErrorHeaders(limiter_name="test", retry_after_ms=100) + err = LimitExceededError(limiter_name="test", retry_after_ms=100) self.assertEqual(err.error_dict(None).get("retry_after_ms"), 100) assert err.headers is not None self.assertEqual(err.headers.get("Retry-After"), "1") def test_limit_exceeded_rounding(self) -> None: - err = self.LimitExceededErrorHeaders(limiter_name="test", retry_after_ms=3001) + err = LimitExceededError(limiter_name="test", retry_after_ms=3001) self.assertEqual(err.error_dict(None).get("retry_after_ms"), 3001) assert err.headers is not None self.assertEqual(err.headers.get("Retry-After"), "4") From c3ebfc7c4300ec5518e8f822dfcf3d6fb7145f79 Mon Sep 17 00:00:00 2001 From: Patrick Cloke Date: Tue, 20 Feb 2024 12:09:38 -0500 Subject: [PATCH 2/4] Newsfragment --- changelog.d/16947.feature | 1 + 1 file changed, 1 insertion(+) create mode 100644 changelog.d/16947.feature diff --git a/changelog.d/16947.feature b/changelog.d/16947.feature new file mode 100644 index 00000000000..efd0dbddb2a --- /dev/null +++ b/changelog.d/16947.feature @@ -0,0 +1 @@ +Include `Retry-After` header by default per [MSC4041](https://github.com/matrix-org/matrix-spec-proposals/pull/4041). Contributed by @clokep. From 7ac3611031866544ec29f32a7903bc7f1ea27d40 Mon Sep 17 00:00:00 2001 From: Patrick Cloke Date: Wed, 6 Mar 2024 09:38:20 -0500 Subject: [PATCH 3/4] Remove unused class property. --- synapse/api/errors.py | 2 -- 1 file changed, 2 deletions(-) diff --git a/synapse/api/errors.py b/synapse/api/errors.py index 7d9f691677d..dd4a1ae7063 100644 --- a/synapse/api/errors.py +++ b/synapse/api/errors.py @@ -517,8 +517,6 @@ def error_dict(self, config: Optional["HomeServerConfig"]) -> "JsonDict": class LimitExceededError(SynapseError): """A client has sent too many requests and is being throttled.""" - include_retry_after_header = False - def __init__( self, limiter_name: str, From 5b4e1a1b444000153ba70f7f2c4139e9149921c1 Mon Sep 17 00:00:00 2001 From: Patrick Cloke Date: Wed, 6 Mar 2024 11:59:39 -0500 Subject: [PATCH 4/4] Remove setting unneeded config flag. --- tests/rest/client/test_login.py | 3 --- 1 file changed, 3 deletions(-) diff --git a/tests/rest/client/test_login.py b/tests/rest/client/test_login.py index 3d3a7b0aa71..3a1f150082e 100644 --- a/tests/rest/client/test_login.py +++ b/tests/rest/client/test_login.py @@ -177,7 +177,6 @@ def make_homeserver(self, reactor: MemoryReactor, clock: Clock) -> HomeServer: # rc_login dict here, we need to set this manually as well "account": {"per_second": 10000, "burst_count": 10000}, }, - "experimental_features": {"msc4041_enabled": True}, } ) def test_POST_ratelimiting_per_address(self) -> None: @@ -229,7 +228,6 @@ def test_POST_ratelimiting_per_address(self) -> None: # rc_login dict here, we need to set this manually as well "address": {"per_second": 10000, "burst_count": 10000}, }, - "experimental_features": {"msc4041_enabled": True}, } ) def test_POST_ratelimiting_per_account(self) -> None: @@ -278,7 +276,6 @@ def test_POST_ratelimiting_per_account(self) -> None: "address": {"per_second": 10000, "burst_count": 10000}, "failed_attempts": {"per_second": 0.17, "burst_count": 5}, }, - "experimental_features": {"msc4041_enabled": True}, } ) def test_POST_ratelimiting_per_account_failed_attempts(self) -> None: