mirror of
https://mau.dev/maunium/synapse.git
synced 2024-11-17 07:21:37 +01:00
New test, fix issues
This commit is contained in:
parent
c69df5d5d3
commit
cd36a1283b
2 changed files with 73 additions and 49 deletions
|
@ -194,7 +194,7 @@ class MatrixFederationHttpClient(object):
|
||||||
request,
|
request,
|
||||||
try_trailing_slash_on_400=False,
|
try_trailing_slash_on_400=False,
|
||||||
backoff_on_404=False,
|
backoff_on_404=False,
|
||||||
send_request_args={},
|
**send_request_args
|
||||||
):
|
):
|
||||||
"""Wrapper for _send_request which can optionally retry the request
|
"""Wrapper for _send_request which can optionally retry the request
|
||||||
upon receiving a combination of a 400 HTTP response code and a
|
upon receiving a combination of a 400 HTTP response code and a
|
||||||
|
@ -206,9 +206,8 @@ class MatrixFederationHttpClient(object):
|
||||||
try_trailing_slash_on_400 (bool): Whether on receiving a 400
|
try_trailing_slash_on_400 (bool): Whether on receiving a 400
|
||||||
'M_UNRECOGNIZED' from the server to retry the request with a
|
'M_UNRECOGNIZED' from the server to retry the request with a
|
||||||
trailing slash appended to the request path.
|
trailing slash appended to the request path.
|
||||||
404_backoff (bool): Whether to backoff on 404 when making a
|
backoff_on_404 (bool): Whether to backoff on 404 when making a
|
||||||
request with a trailing slash (only affects request if
|
request with a trailing slash.
|
||||||
try_trailing_slash_on_400 is True).
|
|
||||||
send_request_args (Dict): A dictionary of arguments to pass to
|
send_request_args (Dict): A dictionary of arguments to pass to
|
||||||
`_send_request()`.
|
`_send_request()`.
|
||||||
|
|
||||||
|
@ -220,36 +219,24 @@ class MatrixFederationHttpClient(object):
|
||||||
Deferred[Dict]: Parsed JSON response body.
|
Deferred[Dict]: Parsed JSON response body.
|
||||||
"""
|
"""
|
||||||
try:
|
try:
|
||||||
response = yield self._send_request(**send_request_args)
|
response = yield self._send_request(request, backoff_on_404=backoff_on_404, **send_request_args)
|
||||||
|
|
||||||
# Check if it's necessary to retry with a trailing slash
|
|
||||||
body = yield _handle_json_response(
|
|
||||||
self.hs.get_reactor(), self.default_timeout, request, response,
|
|
||||||
)
|
|
||||||
|
|
||||||
defer.returnValue(body)
|
|
||||||
except HttpResponseException as e:
|
except HttpResponseException as e:
|
||||||
|
# Received an HTTP error > 300. Check if it meets the requirements
|
||||||
|
# to retry with a trailing slash
|
||||||
if not try_trailing_slash_on_400:
|
if not try_trailing_slash_on_400:
|
||||||
# Received an error >= 300. Raise unless we're retrying
|
raise
|
||||||
raise e
|
|
||||||
except Exception as e:
|
|
||||||
raise e
|
|
||||||
|
|
||||||
# Retry with a trailing slash if we received a 400 with
|
if e.code != 400 or e.to_synapse_error().errcode != "M_UNRECOGNIZED":
|
||||||
# 'M_UNRECOGNIZED' which some endpoints can return when omitting a
|
raise
|
||||||
# trailing slash on Synapse <= v0.99.3.
|
|
||||||
# Enable backoff if initially disabled
|
|
||||||
send_request_args["backoff_on_404"] = backoff_on_404
|
|
||||||
|
|
||||||
# Add trailing slash
|
# Retry with a trailing slash if we received a 400 with
|
||||||
send_request_args["request"].path += "/"
|
# 'M_UNRECOGNIZED' which some endpoints can return when omitting a
|
||||||
|
# trailing slash on Synapse <= v0.99.3.
|
||||||
|
request.path += "/"
|
||||||
|
|
||||||
response = yield self._send_request(**send_request_args)
|
response = yield self._send_request(request, backoff_on_404=backoff_on_404, **send_request_args)
|
||||||
body = yield _handle_json_response(
|
|
||||||
self.hs.get_reactor(), self.default_timeout, request, response,
|
|
||||||
)
|
|
||||||
|
|
||||||
defer.returnValue(body)
|
defer.returnValue(response)
|
||||||
|
|
||||||
@defer.inlineCallbacks
|
@defer.inlineCallbacks
|
||||||
def _send_request(
|
def _send_request(
|
||||||
|
@ -587,19 +574,13 @@ class MatrixFederationHttpClient(object):
|
||||||
json=data,
|
json=data,
|
||||||
)
|
)
|
||||||
|
|
||||||
send_request_args = {
|
response = yield self._send_request_with_optional_trailing_slash(
|
||||||
"request": request,
|
request, try_trailing_slash_on_400, backoff_on_404,
|
||||||
"long_retries": long_retries,
|
long_retries=long_retries, timeout=timeout, ignore_backoff=ignore_backoff,
|
||||||
"timeout": timeout,
|
)
|
||||||
"ignore_backoff": ignore_backoff,
|
|
||||||
# Do not backoff on the initial request if we're trying again with
|
|
||||||
# trailing slashes. Otherwise we may end up waiting to contact a
|
|
||||||
# server that is actually up
|
|
||||||
"backoff_on_404": False if try_trailing_slash_on_400 else backoff_on_404,
|
|
||||||
}
|
|
||||||
|
|
||||||
body = yield self._send_request_with_optional_trailing_slash(
|
body = yield _handle_json_response(
|
||||||
request, try_trailing_slash_on_400, backoff_on_404, send_request_args,
|
self.hs.get_reactor(), self.default_timeout, request, response,
|
||||||
)
|
)
|
||||||
|
|
||||||
defer.returnValue(body)
|
defer.returnValue(body)
|
||||||
|
@ -707,16 +688,14 @@ class MatrixFederationHttpClient(object):
|
||||||
query=args,
|
query=args,
|
||||||
)
|
)
|
||||||
|
|
||||||
send_request_args = {
|
response = yield self._send_request_with_optional_trailing_slash(
|
||||||
"request": request,
|
request, try_trailing_slash_on_400, False,
|
||||||
"retry_on_dns_fail": retry_on_dns_fail,
|
retry_on_dns_fail=retry_on_dns_fail, timeout=timeout,
|
||||||
"timeout": timeout,
|
ignore_backoff=ignore_backoff,
|
||||||
"ignore_backoff": ignore_backoff,
|
)
|
||||||
"backoff_on_404": False,
|
|
||||||
}
|
|
||||||
|
|
||||||
body = yield self._send_request_with_optional_trailing_slash(
|
body = yield _handle_json_response(
|
||||||
request, try_trailing_slash_on_400, False, send_request_args,
|
self.hs.get_reactor(), self.default_timeout, request, response,
|
||||||
)
|
)
|
||||||
|
|
||||||
defer.returnValue(body)
|
defer.returnValue(body)
|
||||||
|
|
|
@ -322,6 +322,51 @@ class FederationClientTests(HomeserverTestCase):
|
||||||
r = self.successResultOf(d)
|
r = self.successResultOf(d)
|
||||||
self.assertEqual(r, {})
|
self.assertEqual(r, {})
|
||||||
|
|
||||||
|
def test_client_does_not_retry_on_400_plus(self):
|
||||||
|
"""
|
||||||
|
Another test for trailing slashes but now test that we don't retry on
|
||||||
|
trailing slashes on a non-400/M_UNRECOGNIZED response.
|
||||||
|
|
||||||
|
See test_client_requires_trailing_slashes() for context.
|
||||||
|
"""
|
||||||
|
d = self.cl.get_json(
|
||||||
|
"testserv:8008", "foo/bar", try_trailing_slash_on_400=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
# Send the request
|
||||||
|
self.pump()
|
||||||
|
|
||||||
|
# there should have been a call to connectTCP
|
||||||
|
clients = self.reactor.tcpClients
|
||||||
|
self.assertEqual(len(clients), 1)
|
||||||
|
(_host, _port, factory, _timeout, _bindAddress) = clients[0]
|
||||||
|
|
||||||
|
# complete the connection and wire it up to a fake transport
|
||||||
|
client = factory.buildProtocol(None)
|
||||||
|
conn = StringTransport()
|
||||||
|
client.makeConnection(conn)
|
||||||
|
|
||||||
|
# that should have made it send the request to the connection
|
||||||
|
self.assertRegex(conn.value(), b"^GET /foo/bar")
|
||||||
|
|
||||||
|
# Clear the original request data before sending a response
|
||||||
|
conn.clear()
|
||||||
|
|
||||||
|
# Send the HTTP response
|
||||||
|
client.dataReceived(
|
||||||
|
b"HTTP/1.1 404 Not Found\r\n"
|
||||||
|
b"Content-Type: application/json\r\n"
|
||||||
|
b"Content-Length: 2\r\n"
|
||||||
|
b"\r\n"
|
||||||
|
b"{}"
|
||||||
|
)
|
||||||
|
|
||||||
|
# We should not get another request
|
||||||
|
self.assertEqual(conn.value(), b"")
|
||||||
|
|
||||||
|
# We should get a 404 failure response
|
||||||
|
r = self.failureResultOf(d)
|
||||||
|
|
||||||
def test_client_sends_body(self):
|
def test_client_sends_body(self):
|
||||||
self.cl.post_json(
|
self.cl.post_json(
|
||||||
"testserv:8008", "foo/bar", timeout=10000,
|
"testserv:8008", "foo/bar", timeout=10000,
|
||||||
|
|
Loading…
Reference in a new issue