Skip to content
This repository was archived by the owner on Apr 26, 2024. It is now read-only.

Commit 9d8a323

Browse files
authored
Respond with proper error responses on unknown paths. (#14621)
Returns a proper 404 with an errcode of M_RECOGNIZED for unknown endpoints per MSC3743.
1 parent da77720 commit 9d8a323

File tree

9 files changed

+32
-14
lines changed

9 files changed

+32
-14
lines changed

changelog.d/14621.bugfix

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
Return spec-compliant JSON errors when unknown endpoints are requested.

synapse/api/errors.py

Lines changed: 2 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -300,10 +300,8 @@ def __init__(self, session_id: str, result: "JsonDict"):
300300
class UnrecognizedRequestError(SynapseError):
301301
"""An error indicating we don't understand the request you're trying to make"""
302302

303-
def __init__(
304-
self, msg: str = "Unrecognized request", errcode: str = Codes.UNRECOGNIZED
305-
):
306-
super().__init__(400, msg, errcode)
303+
def __init__(self, msg: str = "Unrecognized request", code: int = 400):
304+
super().__init__(code, msg, Codes.UNRECOGNIZED)
307305

308306

309307
class NotFoundError(SynapseError):

synapse/http/server.py

Lines changed: 18 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -577,7 +577,24 @@ def _unrecognised_request_handler(request: Request) -> NoReturn:
577577
Args:
578578
request: Unused, but passed in to match the signature of ServletCallback.
579579
"""
580-
raise UnrecognizedRequestError()
580+
raise UnrecognizedRequestError(code=404)
581+
582+
583+
class UnrecognizedRequestResource(resource.Resource):
584+
"""
585+
Similar to twisted.web.resource.NoResource, but returns a JSON 404 with an
586+
errcode of M_UNRECOGNIZED.
587+
"""
588+
589+
def render(self, request: SynapseRequest) -> int:
590+
f = failure.Failure(UnrecognizedRequestError(code=404))
591+
return_json_error(f, request, None)
592+
# A response has already been sent but Twisted requires either NOT_DONE_YET
593+
# or the response bytes as a return value.
594+
return NOT_DONE_YET
595+
596+
def getChild(self, name: str, request: Request) -> resource.Resource:
597+
return self
581598

582599

583600
class RootRedirect(resource.Resource):

synapse/rest/media/v1/media_repository.py

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,6 @@
2424
import twisted.internet.error
2525
import twisted.web.http
2626
from twisted.internet.defer import Deferred
27-
from twisted.web.resource import Resource
2827

2928
from synapse.api.errors import (
3029
FederationDeniedError,
@@ -35,6 +34,7 @@
3534
)
3635
from synapse.config._base import ConfigError
3736
from synapse.config.repository import ThumbnailRequirement
37+
from synapse.http.server import UnrecognizedRequestResource
3838
from synapse.http.site import SynapseRequest
3939
from synapse.logging.context import defer_to_thread
4040
from synapse.metrics.background_process_metrics import run_as_background_process
@@ -1046,7 +1046,7 @@ async def _remove_local_media_from_disk(
10461046
return removed_media, len(removed_media)
10471047

10481048

1049-
class MediaRepositoryResource(Resource):
1049+
class MediaRepositoryResource(UnrecognizedRequestResource):
10501050
"""File uploading and downloading.
10511051
10521052
Uploads are POSTed to a resource which returns a token which is used to GET

synapse/util/httpresourcetree.py

Lines changed: 4 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,9 @@
1515
import logging
1616
from typing import Dict
1717

18-
from twisted.web.resource import NoResource, Resource
18+
from twisted.web.resource import Resource
19+
20+
from synapse.http.server import UnrecognizedRequestResource
1921

2022
logger = logging.getLogger(__name__)
2123

@@ -49,7 +51,7 @@ def create_resource_tree(
4951
for path_seg in full_path.split(b"/")[1:-1]:
5052
if path_seg not in last_resource.listNames():
5153
# resource doesn't exist, so make a "dummy resource"
52-
child_resource: Resource = NoResource()
54+
child_resource: Resource = UnrecognizedRequestResource()
5355
last_resource.putChild(path_seg, child_resource)
5456
res_id = _resource_id(last_resource, path_seg)
5557
resource_mappings[res_id] = child_resource

tests/rest/admin/test_user.py

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -3994,7 +3994,7 @@ def test_user_is_not_local(self, method: str) -> None:
39943994
"""
39953995
Tests that shadow-banning for a user that is not a local returns a 400
39963996
"""
3997-
url = "/_synapse/admin/v1/whois/@unknown_person:unknown_domain"
3997+
url = "/_synapse/admin/v1/users/@unknown_person:unknown_domain/shadow_ban"
39983998

39993999
channel = self.make_request(method, url, access_token=self.admin_user_tok)
40004000
self.assertEqual(400, channel.code, msg=channel.json_body)

tests/rest/client/test_login_token_request.py

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -48,13 +48,13 @@ def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None:
4848

4949
def test_disabled(self) -> None:
5050
channel = self.make_request("POST", endpoint, {}, access_token=None)
51-
self.assertEqual(channel.code, 400)
51+
self.assertEqual(channel.code, 404)
5252

5353
self.register_user(self.user, self.password)
5454
token = self.login(self.user, self.password)
5555

5656
channel = self.make_request("POST", endpoint, {}, access_token=token)
57-
self.assertEqual(channel.code, 400)
57+
self.assertEqual(channel.code, 404)
5858

5959
@override_config({"experimental_features": {"msc3882_enabled": True}})
6060
def test_require_auth(self) -> None:

tests/rest/client/test_rendezvous.py

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,7 @@ def make_homeserver(self, reactor: MemoryReactor, clock: Clock) -> HomeServer:
3636

3737
def test_disabled(self) -> None:
3838
channel = self.make_request("POST", endpoint, {}, access_token=None)
39-
self.assertEqual(channel.code, 400)
39+
self.assertEqual(channel.code, 404)
4040

4141
@override_config({"experimental_features": {"msc3886_endpoint": "/asd"}})
4242
def test_redirect(self) -> None:

tests/test_server.py

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -174,7 +174,7 @@ def _callback(request: SynapseRequest, **kwargs: object) -> None:
174174
self.reactor, FakeSite(res, self.reactor), b"GET", b"/_matrix/foobar"
175175
)
176176

177-
self.assertEqual(channel.code, 400)
177+
self.assertEqual(channel.code, 404)
178178
self.assertEqual(channel.json_body["error"], "Unrecognized request")
179179
self.assertEqual(channel.json_body["errcode"], "M_UNRECOGNIZED")
180180

0 commit comments

Comments
 (0)