From e9e8b5070df836cd4dc7b2f25f1a4431c8521d36 Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Wed, 8 May 2024 15:10:17 -0700 Subject: [PATCH 01/18] add federation /download servlet --- synapse/federation/transport/server/_base.py | 24 +++++++++-- .../federation/transport/server/federation.py | 42 +++++++++++++++++++ 2 files changed, 62 insertions(+), 4 deletions(-) diff --git a/synapse/federation/transport/server/_base.py b/synapse/federation/transport/server/_base.py index 23d1254127a..f5b358e31aa 100644 --- a/synapse/federation/transport/server/_base.py +++ b/synapse/federation/transport/server/_base.py @@ -356,13 +356,29 @@ async def new_func( "request" ) return None + if ( + func.__self__.__class__.__name__ # type: ignore + == "FederationUnstableMediaDownloadServlet" + ): + response = await func( + origin, content, request, *args, **kwargs + ) + else: + response = await func( + origin, content, request.args, *args, **kwargs + ) + else: + if ( + func.__self__.__class__.__name__ # type: ignore + == "FederationUnstableMediaDownloadServlet" + ): + response = await func( + origin, content, request, *args, **kwargs + ) + else: response = await func( origin, content, request.args, *args, **kwargs ) - else: - response = await func( - origin, content, request.args, *args, **kwargs - ) finally: # if we used the origin's context as the parent, add a new span using # the servlet span as a parent, so that we have a link diff --git a/synapse/federation/transport/server/federation.py b/synapse/federation/transport/server/federation.py index a59734785fa..643e6c87a31 100644 --- a/synapse/federation/transport/server/federation.py +++ b/synapse/federation/transport/server/federation.py @@ -44,10 +44,13 @@ ) from synapse.http.servlet import ( parse_boolean_from_args, + parse_integer, parse_integer_from_args, parse_string_from_args, parse_strings_from_args, ) +from synapse.http.site import SynapseRequest +from synapse.media._base import DEFAULT_MAX_TIMEOUT_MS, MAXIMUM_ALLOWED_MAX_TIMEOUT_MS from synapse.types import JsonDict from synapse.util import SYNAPSE_VERSION from synapse.util.ratelimitutils import FederationRateLimiter @@ -787,6 +790,44 @@ async def on_POST( return 200, {"account_statuses": statuses, "failures": failures} +class FederationUnstableMediaDownloadServlet(BaseFederationServerServlet): + """ + Implementation of new federation media `/download` endpoint outlined in MSC3916. Returns + a multipart/form-data response consisting of a JSON object and the requested media + item. This endpoint only returns local media. + """ + + PATH = "/media/download/(?P[^/]*)/(?P[^/]*)" + PREFIX = FEDERATION_UNSTABLE_PREFIX + "/org.matrix.msc3916" + RATELIMIT = True + + def __init__( + self, + hs: "HomeServer", + ratelimiter: FederationRateLimiter, + authenticator: Authenticator, + server_name: str, + ): + super().__init__(hs, authenticator, ratelimiter, server_name) + self.media_repo = self.hs.get_media_repository() + + async def on_GET( + self, + origin: Optional[str], + content: Literal[None], + request: SynapseRequest, + server_name: str, + media_id: str, + ) -> None: + max_timeout_ms = parse_integer( + request, "timeout_ms", default=DEFAULT_MAX_TIMEOUT_MS + ) + max_timeout_ms = min(max_timeout_ms, MAXIMUM_ALLOWED_MAX_TIMEOUT_MS) + await self.media_repo.get_local_media( + request, media_id, None, max_timeout_ms, federation=True + ) + + FEDERATION_SERVLET_CLASSES: Tuple[Type[BaseFederationServlet], ...] = ( FederationSendServlet, FederationEventServlet, @@ -818,4 +859,5 @@ async def on_POST( FederationV1SendKnockServlet, FederationMakeKnockServlet, FederationAccountStatusServlet, + FederationUnstableMediaDownloadServlet, ) From 913e67e27c20c1429b5a4b78262e23e3dceb3fb8 Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Wed, 8 May 2024 15:13:35 -0700 Subject: [PATCH 02/18] add MultipartResponder and MultipartFileSender classes and use these classes when fetching media for federation download --- synapse/media/media_storage.py | 208 ++++++++++++++++++++++++++++++++- 1 file changed, 202 insertions(+), 6 deletions(-) diff --git a/synapse/media/media_storage.py b/synapse/media/media_storage.py index b45b319f5c9..0ae9f56ad74 100644 --- a/synapse/media/media_storage.py +++ b/synapse/media/media_storage.py @@ -19,9 +19,12 @@ # # import contextlib +import json import logging import os import shutil +from contextlib import closing +from io import BytesIO from types import TracebackType from typing import ( IO, @@ -31,14 +34,19 @@ BinaryIO, Callable, Generator, + List, Optional, Sequence, Tuple, Type, + Union, ) +from uuid import uuid4 import attr +from zope.interface import implementer +from twisted.internet import defer, interfaces from twisted.internet.defer import Deferred from twisted.internet.interfaces import IConsumer from twisted.protocols.basic import FileSender @@ -49,6 +57,8 @@ from synapse.util import Clock from synapse.util.file_consumer import BackgroundFileConsumer +from ..storage.databases.main.media_repository import LocalMedia +from ..types import JsonDict from ._base import FileInfo, Responder from .filepath import MediaFilePaths @@ -58,6 +68,8 @@ logger = logging.getLogger(__name__) +CRLF = b"\r\n" + class MediaStorage: """Responsible for storing/fetching files from local sources. @@ -202,15 +214,23 @@ async def finish() -> None: ) raise exc - async def fetch_media(self, file_info: FileInfo) -> Optional[Responder]: + async def fetch_media( + self, + file_info: FileInfo, + media_info: Optional[LocalMedia] = None, + federation: bool = False, + ) -> Optional[Responder]: """Attempts to fetch media described by file_info from the local cache and configured storage providers. Args: - file_info + file_info: Metadata about the media file + media_info: Metadata about the media item + federation: Whether this file is being fetched for a federation request Returns: - Returns a Responder if the file was found, otherwise None. + If the file was found returns a Responder (a Multipart Responder if the requested + file is for the federation /download endpoint), otherwise None. """ paths = [self._file_info_to_path(file_info)] @@ -230,12 +250,19 @@ async def fetch_media(self, file_info: FileInfo) -> Optional[Responder]: local_path = os.path.join(self.local_media_directory, path) if os.path.exists(local_path): logger.debug("responding with local file %s", local_path) - return FileResponder(open(local_path, "rb")) + if federation: + assert media_info is not None + boundary = uuid4().hex.encode("ascii") + return MultipartResponder( + open(local_path, "rb"), media_info, boundary + ) + else: + return FileResponder(open(local_path, "rb")) logger.debug("local file %s did not exist", local_path) for provider in self.storage_providers: for path in paths: - res: Any = await provider.fetch(path, file_info) + res: Any = await provider.fetch(path, file_info, media_info, federation) if res: logger.debug("Streaming %s from %s", path, provider) return res @@ -349,7 +376,7 @@ class FileResponder(Responder): """Wraps an open file that can be sent to a request. Args: - open_file: A file like object to be streamed ot the client, + open_file: A file like object to be streamed to the client, is closed when finished streaming. """ @@ -370,6 +397,38 @@ def __exit__( self.open_file.close() +class MultipartResponder(Responder): + """Wraps an open file, formats the response according to MSC3916 and sends it to a + federation request. + + Args: + open_file: A file like object to be streamed to the client, + is closed when finished streaming. + media_info: metadata about the media item + boundary: bytes to use for the multipart response boundary + """ + + def __init__(self, open_file: IO, media_info: LocalMedia, boundary: bytes) -> None: + self.open_file = open_file + self.media_info = media_info + self.boundary = boundary + + def write_to_consumer(self, consumer: IConsumer) -> Deferred: + return make_deferred_yieldable( + MultipartFileSender().beginFileTransfer( + self.open_file, consumer, self.media_info.media_type, {}, self.boundary + ) + ) + + def __exit__( + self, + exc_type: Optional[Type[BaseException]], + exc_val: Optional[BaseException], + exc_tb: Optional[TracebackType], + ) -> None: + self.open_file.close() + + class SpamMediaException(NotFoundError): """The media was blocked by a spam checker, so we simply 404 the request (in the same way as if it was quarantined). @@ -403,3 +462,140 @@ async def write_chunks_to(self, callback: Callable[[bytes], object]) -> None: # We yield to the reactor by sleeping for 0 seconds. await self.clock.sleep(0) + + +@implementer(interfaces.IProducer) +class MultipartFileSender: + """ + A producer that sends the contents of a file to a federation request in the format + outlined in MSC3916 - a multipart/format-data response where the first field is a + JSON object and the second is the requested file. + + This is a slight re-writing of twisted.protocols.basic.FileSender to achieve the format + outlined above. + """ + + CHUNK_SIZE = 2**14 + + lastSent = "" + deferred: Optional[defer.Deferred] = None + + def beginFileTransfer( + self, + file: IO, + consumer: IConsumer, + file_content_type: str, + json_object: JsonDict, + boundary: bytes, + ) -> Deferred: + """ + Begin transferring a file + + Args: + file: The file object to read data from + consumer: The synapse request to write the data to + file_content_type: The content-type of the file + json_object: The JSON object to write to the first field of the response + boundary: bytes to be used as the multipart/form-data boundary + + Returns: A deferred whose callback will be invoked when the file has + been completely written to the consumer. The last byte written to the + consumer is passed to the callback. + """ + self.file: Optional[IO] = file + self.consumer = consumer + self.json_field = json_object + self.json_field_written = False + self.content_type_written = False + self.file_content_type = file_content_type + self.boundary = boundary + self.deferred: Deferred = defer.Deferred() + self.consumer.registerProducer(self, False) + + deferred = self.deferred + return deferred + + def resumeProducing(self) -> None: + if not self.json_field_written: + self.consumer.write(CRLF + b"--" + self.boundary + b"" + CRLF) + content_type = Header(b"Content-Type", b"application/json") + self.consumer.write(bytes(content_type) + CRLF) + json_field = json.dumps(self.json_field) + json_bytes = json_field.encode("utf-8") + self.consumer.write(json_bytes) + self.consumer.write(CRLF + b"--" + self.boundary + b"" + CRLF) + self.json_field_written = True + chunk: Any = "" + if self.file: + if not self.content_type_written: + type = self.file_content_type.encode("utf-8") + content_type = Header(b"Content-Type", type) + self.consumer.write(bytes(content_type) + CRLF) + self.content_type_written = True + chunk = self.file.read(self.CHUNK_SIZE) + if not chunk: + self.consumer.write(CRLF + b"--" + self.boundary + b"--" + CRLF) + self.file = None + self.consumer.unregisterProducer() + if self.deferred: + self.deferred.callback(self.lastSent) + self.deferred = None + return + + self.consumer.write(chunk) + self.lastSent = chunk[-1:] + + def pauseProducing(self) -> None: + pass + + def stopProducing(self) -> None: + if self.deferred: + self.deferred.errback(Exception("Consumer asked us to stop producing")) + self.deferred = None + + +class Header: + """ + `Header` This class is a tiny wrapper that produces + request headers. We can't use standard python header + class because it encodes unicode fields using =? bla bla ?= + encoding, which is correct, but no one in HTTP world expects + that, everyone wants utf-8 raw bytes. (stolen from treq.multipart) + + """ + + def __init__( + self, + name: bytes, + value: Any, + params: Optional[List[Tuple[Any, Any]]] = None, + ): + self.name = name + self.value = value + self.params = params or [] + + def add_param(self, name: Any, value: Any) -> None: + self.params.append((name, value)) + + def __bytes__(self) -> bytes: + with closing(BytesIO()) as h: + h.write(self.name + b": " + escape(self.value).encode("us-ascii")) + if self.params: + for name, val in self.params: + h.write(b"; ") + h.write(escape(name).encode("us-ascii")) + h.write(b"=") + h.write(b'"' + escape(val).encode("utf-8") + b'"') + h.seek(0) + return h.read() + + +def escape(value: Union[str, bytes]) -> str: + """ + This function prevents header values from corrupting the request, + a newline in the file name parameter makes form-data request unreadable + for a majority of parsers. (stolen from treq.multipart) + """ + if isinstance(value, bytes): + value = value.decode("utf-8") + return value.replace("\r", "").replace("\n", "").replace('"', '\\"') From 6a8063b3b8b9be7617c0b3d98775228f319c6f2f Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Wed, 8 May 2024 15:14:17 -0700 Subject: [PATCH 03/18] add multipart responder function --- synapse/media/_base.py | 63 +++++++++++++++++++++++++++++++++++++++++- 1 file changed, 62 insertions(+), 1 deletion(-) diff --git a/synapse/media/_base.py b/synapse/media/_base.py index 3fbed6062f8..6b32392bd19 100644 --- a/synapse/media/_base.py +++ b/synapse/media/_base.py @@ -25,7 +25,16 @@ import urllib from abc import ABC, abstractmethod from types import TracebackType -from typing import Awaitable, Dict, Generator, List, Optional, Tuple, Type +from typing import ( + TYPE_CHECKING, + Awaitable, + Dict, + Generator, + List, + Optional, + Tuple, + Type, +) import attr @@ -39,6 +48,11 @@ from synapse.logging.context import make_deferred_yieldable from synapse.util.stringutils import is_ascii +if TYPE_CHECKING: + from synapse.media.media_storage import MultipartResponder + from synapse.storage.databases.main.media_repository import LocalMedia + + logger = logging.getLogger(__name__) # list all text content types that will have the charset default to UTF-8 when @@ -260,6 +274,53 @@ def _can_encode_filename_as_token(x: str) -> bool: return True +async def respond_with_multipart_responder( + request: SynapseRequest, + responder: "Optional[MultipartResponder]", + media_info: "LocalMedia", +) -> None: + """ + Responds via a Multipart responder for the federation media `/download` requests + + Args: + request: the federation request to respond to + responder: the Multipart responder which will send the response + media_info: metadata about the media item + """ + if not responder: + respond_404(request) + return + + # If we have a responder we *must* use it as a context manager. + with responder: + if request._disconnected: + logger.warning( + "Not sending response to request %s, already disconnected.", request + ) + return + + logger.debug("Responding to media request with responder %s", responder) + if media_info.media_length is not None: + request.setHeader(b"Content-Length", b"%d" % (media_info.media_length,)) + request.setHeader( + b"Content-Type", b"multipart/form-data; boundary=%s" % responder.boundary + ) + + try: + await responder.write_to_consumer(request) + except Exception as e: + # The majority of the time this will be due to the client having gone + # away. Unfortunately, Twisted simply throws a generic exception at us + # in that case. + logger.warning("Failed to write to consumer: %s %s", type(e), e) + + # Unregister the producer, if it has one, so Twisted doesn't complain + if request.producer: + request.unregisterProducer() + + finish_request(request) + + async def respond_with_responder( request: SynapseRequest, responder: "Optional[Responder]", From 447cbcc408a71deaead089fa3fde50ae4f515306 Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Wed, 8 May 2024 15:14:47 -0700 Subject: [PATCH 04/18] use multipart responder when fetching media for download --- synapse/media/media_repository.py | 19 +++++++++++---- synapse/media/storage_provider.py | 40 +++++++++++++++++++++++++++---- 2 files changed, 50 insertions(+), 9 deletions(-) diff --git a/synapse/media/media_repository.py b/synapse/media/media_repository.py index 0e875132f6f..84f9e5681e7 100644 --- a/synapse/media/media_repository.py +++ b/synapse/media/media_repository.py @@ -53,10 +53,11 @@ ThumbnailInfo, get_filename_from_headers, respond_404, + respond_with_multipart_responder, respond_with_responder, ) from synapse.media.filepath import MediaFilePaths -from synapse.media.media_storage import MediaStorage +from synapse.media.media_storage import MediaStorage, MultipartResponder from synapse.media.storage_provider import StorageProviderWrapper from synapse.media.thumbnailer import Thumbnailer, ThumbnailError from synapse.media.url_previewer import UrlPreviewer @@ -70,6 +71,7 @@ if TYPE_CHECKING: from synapse.server import HomeServer + logger = logging.getLogger(__name__) # How often to run the background job to update the "recently accessed" @@ -422,6 +424,7 @@ async def get_local_media( media_id: str, name: Optional[str], max_timeout_ms: int, + federation: bool = False, ) -> None: """Responds to requests for local media, if exists, or returns 404. @@ -433,6 +436,7 @@ async def get_local_media( the filename in the Content-Disposition header of the response. max_timeout_ms: the maximum number of milliseconds to wait for the media to be uploaded. + federation: whether the local media being fetched is for a federation request Returns: Resolves once a response has successfully been written to request @@ -452,10 +456,17 @@ async def get_local_media( file_info = FileInfo(None, media_id, url_cache=bool(url_cache)) - responder = await self.media_storage.fetch_media(file_info) - await respond_with_responder( - request, responder, media_type, media_length, upload_name + responder = await self.media_storage.fetch_media( + file_info, media_info, federation ) + if federation: + # this really should be a Multipart responder but just in case + assert isinstance(responder, MultipartResponder) + await respond_with_multipart_responder(request, responder, media_info) + else: + await respond_with_responder( + request, responder, media_type, media_length, upload_name + ) async def get_remote_media( self, diff --git a/synapse/media/storage_provider.py b/synapse/media/storage_provider.py index 06e5d27a53a..a2d50adf658 100644 --- a/synapse/media/storage_provider.py +++ b/synapse/media/storage_provider.py @@ -24,14 +24,16 @@ import os import shutil from typing import TYPE_CHECKING, Callable, Optional +from uuid import uuid4 from synapse.config._base import Config from synapse.logging.context import defer_to_thread, run_in_background from synapse.logging.opentracing import start_active_span, trace_with_opname from synapse.util.async_helpers import maybe_awaitable +from ..storage.databases.main.media_repository import LocalMedia from ._base import FileInfo, Responder -from .media_storage import FileResponder +from .media_storage import FileResponder, MultipartResponder logger = logging.getLogger(__name__) @@ -55,13 +57,21 @@ async def store_file(self, path: str, file_info: FileInfo) -> None: """ @abc.abstractmethod - async def fetch(self, path: str, file_info: FileInfo) -> Optional[Responder]: + async def fetch( + self, + path: str, + file_info: FileInfo, + media_info: Optional[LocalMedia] = None, + federation: bool = False, + ) -> Optional[Responder]: """Attempt to fetch the file described by file_info and stream it into writer. Args: path: Relative path of file in local cache file_info: The metadata of the file. + media_info: metadata of the media item + federation: Whether the requested media is for a federation request Returns: Returns a Responder if the provider has the file, otherwise returns None. @@ -124,7 +134,13 @@ async def store() -> None: run_in_background(store) @trace_with_opname("StorageProviderWrapper.fetch") - async def fetch(self, path: str, file_info: FileInfo) -> Optional[Responder]: + async def fetch( + self, + path: str, + file_info: FileInfo, + media_info: Optional[LocalMedia] = None, + federation: bool = False, + ) -> Optional[Responder]: if file_info.url_cache: # Files in the URL preview cache definitely aren't stored here, # so avoid any potentially slow I/O or network access. @@ -132,7 +148,9 @@ async def fetch(self, path: str, file_info: FileInfo) -> Optional[Responder]: # store_file is supposed to return an Awaitable, but guard # against improper implementations. - return await maybe_awaitable(self.backend.fetch(path, file_info)) + return await maybe_awaitable( + self.backend.fetch(path, file_info, media_info, federation) + ) class FileStorageProviderBackend(StorageProvider): @@ -172,11 +190,23 @@ async def store_file(self, path: str, file_info: FileInfo) -> None: ) @trace_with_opname("FileStorageProviderBackend.fetch") - async def fetch(self, path: str, file_info: FileInfo) -> Optional[Responder]: + async def fetch( + self, + path: str, + file_info: FileInfo, + media_info: Optional[LocalMedia] = None, + federation: bool = False, + ) -> Optional[Responder]: """See StorageProvider.fetch""" backup_fname = os.path.join(self.base_directory, path) if os.path.isfile(backup_fname): + if federation: + assert media_info is not None + boundary = uuid4().hex.encode("ascii") + return MultipartResponder( + open(backup_fname, "rb"), media_info, boundary + ) return FileResponder(open(backup_fname, "rb")) return None From d317c5e61d58fd94fb8bc20629344530cc97414b Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Wed, 8 May 2024 15:14:54 -0700 Subject: [PATCH 05/18] tests --- tests/federation/test_federation_media.py | 128 ++++++++++++++++++++++ 1 file changed, 128 insertions(+) create mode 100644 tests/federation/test_federation_media.py diff --git a/tests/federation/test_federation_media.py b/tests/federation/test_federation_media.py new file mode 100644 index 00000000000..89af495f35a --- /dev/null +++ b/tests/federation/test_federation_media.py @@ -0,0 +1,128 @@ +# +# This file is licensed under the Affero General Public License (AGPL) version 3. +# +# Copyright (C) 2024 New Vector, Ltd +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU Affero General Public License as +# published by the Free Software Foundation, either version 3 of the +# License, or (at your option) any later version. +# +# See the GNU Affero General Public License for more details: +# . +# +# Originally licensed under the Apache License, Version 2.0: +# . +# +# [This file includes modifications made by New Vector Limited] +# +# +import io +import os +import shutil +import tempfile + +from twisted.internet.testing import MemoryReactor + +from synapse.media.filepath import MediaFilePaths +from synapse.media.media_storage import MediaStorage +from synapse.media.storage_provider import FileStorageProviderBackend +from synapse.server import HomeServer +from synapse.types import UserID +from synapse.util import Clock + +from tests import unittest +from tests.test_utils import SMALL_PNG + + +class FederationUnstableMediaDownloads(unittest.FederatingHomeserverTestCase): + def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None: + super().prepare(reactor, clock, hs) + self.test_dir = tempfile.mkdtemp(prefix="synapse-tests-") + self.addCleanup(shutil.rmtree, self.test_dir) + + self.primary_base_path = os.path.join(self.test_dir, "primary") + self.secondary_base_path = os.path.join(self.test_dir, "secondary") + + hs.config.media.media_store_path = self.primary_base_path + + storage_providers = [FileStorageProviderBackend(hs, self.secondary_base_path)] + + self.filepaths = MediaFilePaths(self.primary_base_path) + self.media_storage = MediaStorage( + hs, self.primary_base_path, self.filepaths, storage_providers + ) + self.media_repo = hs.get_media_repository() + + def test_file_download(self) -> None: + content = io.BytesIO(b"file_to_stream") + content_uri = self.get_success( + self.media_repo.create_content( + "text/plain", + "test_upload", + content, + 46, + UserID.from_string("@user_id:whatever.org"), + ) + ) + # test with a text file + channel = self.make_signed_federation_request( + "GET", + f"/_matrix/federation/unstable/org.matrix.msc3916/media/download/{self.hs.hostname}/{content_uri.media_id}", + ) + self.pump() + self.assertEqual(200, channel.code) + content_type = channel.headers.getRawHeaders("content-type") + assert content_type is not None + assert "multipart/form-data" in content_type[0] + assert "boundary" in content_type[0] + # extract boundary + boundary = content_type[0].split("boundary=")[1] + # split on boundary and check that json field and expected value exist + stripped = channel.text_body.split("\r\n" + "--" + boundary) + found_json = any( + "\r\nContent-Type: application/json\r\n{}" in field for field in stripped + ) + self.assertTrue(found_json) + # check that text file and expected value exist + found_file = any( + "\r\nContent-Type: text/plain\r\nfile_to_stream" in field + for field in stripped + ) + self.assertTrue(found_file) + + content = io.BytesIO(SMALL_PNG) + content_uri = self.get_success( + self.media_repo.create_content( + "image/png", + "test_png_upload", + content, + 67, + UserID.from_string("@user_id:whatever.org"), + ) + ) + # test with an image file + channel = self.make_signed_federation_request( + "GET", + f"/_matrix/federation/unstable/org.matrix.msc3916/media/download/{self.hs.hostname}/{content_uri.media_id}", + ) + self.pump() + self.assertEqual(200, channel.code) + content_type = channel.headers.getRawHeaders("content-type") + assert content_type is not None + assert "multipart/form-data" in content_type[0] + assert "boundary" in content_type[0] + # extract boundary + boundary = content_type[0].split("boundary=")[1] + # split on boundary and check that json field and expected value exist + body = channel.result.get("body") + assert body is not None + stripped_bytes = body.split(b"\r\n" + b"--" + boundary.encode("utf-8")) + found_json = any( + b"\r\nContent-Type: application/json\r\n{}" in field + for field in stripped_bytes + ) + self.assertTrue(found_json) + # check that png file exists and matches what was uploaded + found_file = any(SMALL_PNG in field for field in stripped_bytes) + self.assertTrue(found_file) From cbf9154737595a51c9e007c18ed6883512e12503 Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Wed, 8 May 2024 15:37:07 -0700 Subject: [PATCH 06/18] cleanup + note --- synapse/media/media_repository.py | 1 - tests/federation/test_federation_media.py | 2 ++ 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/synapse/media/media_repository.py b/synapse/media/media_repository.py index 84f9e5681e7..52711bbc06f 100644 --- a/synapse/media/media_repository.py +++ b/synapse/media/media_repository.py @@ -71,7 +71,6 @@ if TYPE_CHECKING: from synapse.server import HomeServer - logger = logging.getLogger(__name__) # How often to run the background job to update the "recently accessed" diff --git a/tests/federation/test_federation_media.py b/tests/federation/test_federation_media.py index 89af495f35a..67af44c8d5d 100644 --- a/tests/federation/test_federation_media.py +++ b/tests/federation/test_federation_media.py @@ -80,6 +80,8 @@ def test_file_download(self) -> None: boundary = content_type[0].split("boundary=")[1] # split on boundary and check that json field and expected value exist stripped = channel.text_body.split("\r\n" + "--" + boundary) + # TODO: the json object expected will change once MSC3911 is implemented, currently + # {} is returned for all requests as a placeholder (per MSC3196) found_json = any( "\r\nContent-Type: application/json\r\n{}" in field for field in stripped ) From b6803f6049a823ca2c76da129c790728257e4ef4 Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Wed, 8 May 2024 15:40:19 -0700 Subject: [PATCH 07/18] newsfragment --- changelog.d/17172.feature | 2 ++ 1 file changed, 2 insertions(+) create mode 100644 changelog.d/17172.feature diff --git a/changelog.d/17172.feature b/changelog.d/17172.feature new file mode 100644 index 00000000000..245dea815cb --- /dev/null +++ b/changelog.d/17172.feature @@ -0,0 +1,2 @@ +Support [MSC3916](https://github.com/matrix-org/matrix-spec-proposals/blob/rav/authentication-for-media/proposals/3916-authentication-for-media.md) +by adding a federation /download endpoint (#17172). \ No newline at end of file From 522e752f7fe3d079124ecba392ef1bdfdd4e61e4 Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Wed, 8 May 2024 15:58:20 -0700 Subject: [PATCH 08/18] maybe fix olddeps error?? --- tests/federation/test_federation_media.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/federation/test_federation_media.py b/tests/federation/test_federation_media.py index 67af44c8d5d..bd9abf9edf9 100644 --- a/tests/federation/test_federation_media.py +++ b/tests/federation/test_federation_media.py @@ -22,7 +22,7 @@ import shutil import tempfile -from twisted.internet.testing import MemoryReactor +from twisted.test.proto_helpers import MemoryReactor from synapse.media.filepath import MediaFilePaths from synapse.media.media_storage import MediaStorage From a01553233e13c3557a2884596ec88629a89c8399 Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Thu, 9 May 2024 14:35:54 -0700 Subject: [PATCH 09/18] dont load federation download servlet if the media repo is not enabled --- synapse/federation/transport/server/__init__.py | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/synapse/federation/transport/server/__init__.py b/synapse/federation/transport/server/__init__.py index bac569e9770..25638f63a2a 100644 --- a/synapse/federation/transport/server/__init__.py +++ b/synapse/federation/transport/server/__init__.py @@ -33,6 +33,7 @@ FEDERATION_SERVLET_CLASSES, FederationAccountStatusServlet, FederationUnstableClientKeysClaimServlet, + FederationUnstableMediaDownloadServlet, ) from synapse.http.server import HttpServer, JsonResource from synapse.http.servlet import ( @@ -315,6 +316,12 @@ def register_servlets( ): continue + if ( + servletclass == FederationUnstableMediaDownloadServlet + and not hs.config.server.enable_media_repo + ): + continue + servletclass( hs=hs, authenticator=authenticator, From 85bd3a501329c2abf823e797bd2e73ed7f4e40eb Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Mon, 13 May 2024 15:04:17 -0700 Subject: [PATCH 10/18] improve readability --- synapse/media/media_storage.py | 14 ++++++++++++-- 1 file changed, 12 insertions(+), 2 deletions(-) diff --git a/synapse/media/media_storage.py b/synapse/media/media_storage.py index 0ae9f56ad74..2b9a3e4eee7 100644 --- a/synapse/media/media_storage.py +++ b/synapse/media/media_storage.py @@ -516,27 +516,37 @@ def beginFileTransfer( return deferred def resumeProducing(self) -> None: + # write the first field, which will always be a json field if not self.json_field_written: - self.consumer.write(CRLF + b"--" + self.boundary + b"" + CRLF) + self.consumer.write(CRLF + b"--" + self.boundary + CRLF) + content_type = Header(b"Content-Type", b"application/json") self.consumer.write(bytes(content_type) + CRLF) + json_field = json.dumps(self.json_field) json_bytes = json_field.encode("utf-8") self.consumer.write(json_bytes) - self.consumer.write(CRLF + b"--" + self.boundary + b"" + CRLF) + self.consumer.write(CRLF + b"--" + self.boundary + CRLF) + self.json_field_written = True + chunk: Any = "" if self.file: + # if we haven't written the content type yet, do so if not self.content_type_written: type = self.file_content_type.encode("utf-8") content_type = Header(b"Content-Type", type) self.consumer.write(bytes(content_type) + CRLF) self.content_type_written = True + chunk = self.file.read(self.CHUNK_SIZE) + if not chunk: + # we've reached the end of the file self.consumer.write(CRLF + b"--" + self.boundary + b"--" + CRLF) self.file = None self.consumer.unregisterProducer() + if self.deferred: self.deferred.callback(self.lastSent) self.deferred = None From ef41492fa197281b744f0fb48f0ab5695d162128 Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Mon, 13 May 2024 16:25:04 -0700 Subject: [PATCH 11/18] gate endpoint behind experimental config option --- synapse/config/experimental.py | 4 ++++ synapse/federation/transport/server/__init__.py | 11 ++++++----- tests/federation/test_federation_media.py | 4 ++++ 3 files changed, 14 insertions(+), 5 deletions(-) diff --git a/synapse/config/experimental.py b/synapse/config/experimental.py index 749452ce934..5eb71db958e 100644 --- a/synapse/config/experimental.py +++ b/synapse/config/experimental.py @@ -436,3 +436,7 @@ def read_config(self, config: JsonDict, **kwargs: Any) -> None: self.msc4115_membership_on_events = experimental.get( "msc4115_membership_on_events", False ) + + self.msc3916_authenticated_media_enabled = experimental.get( + "msc3916_authenticated_media_enabled", False + ) diff --git a/synapse/federation/transport/server/__init__.py b/synapse/federation/transport/server/__init__.py index 25638f63a2a..edaf0196d67 100644 --- a/synapse/federation/transport/server/__init__.py +++ b/synapse/federation/transport/server/__init__.py @@ -316,11 +316,12 @@ def register_servlets( ): continue - if ( - servletclass == FederationUnstableMediaDownloadServlet - and not hs.config.server.enable_media_repo - ): - continue + if servletclass == FederationUnstableMediaDownloadServlet: + if ( + not hs.config.server.enable_media_repo + or not hs.config.experimental.msc3916_authenticated_media_enabled + ): + continue servletclass( hs=hs, diff --git a/tests/federation/test_federation_media.py b/tests/federation/test_federation_media.py index bd9abf9edf9..ed6ca7e88d5 100644 --- a/tests/federation/test_federation_media.py +++ b/tests/federation/test_federation_media.py @@ -33,6 +33,7 @@ from tests import unittest from tests.test_utils import SMALL_PNG +from tests.unittest import override_config class FederationUnstableMediaDownloads(unittest.FederatingHomeserverTestCase): @@ -54,6 +55,9 @@ def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None: ) self.media_repo = hs.get_media_repository() + @override_config( + {"experimental_features": {"msc3916_authenticated_media_enabled": True}} + ) def test_file_download(self) -> None: content = io.BytesIO(b"file_to_stream") content_uri = self.get_success( From cbac3f5250ab10299257e4bfdf57f5a0009737b1 Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Tue, 14 May 2024 13:44:55 -0700 Subject: [PATCH 12/18] add comment and test --- synapse/media/media_storage.py | 3 ++- tests/federation/test_federation_media.py | 29 +++++++++++++++++++++++ 2 files changed, 31 insertions(+), 1 deletion(-) diff --git a/synapse/media/media_storage.py b/synapse/media/media_storage.py index 2b9a3e4eee7..5ce91706964 100644 --- a/synapse/media/media_storage.py +++ b/synapse/media/media_storage.py @@ -511,7 +511,8 @@ def beginFileTransfer( self.boundary = boundary self.deferred: Deferred = defer.Deferred() self.consumer.registerProducer(self, False) - + # while it's not entirely clear why this assignment is necessary, it mirrors + # the behavior in FileSender.beginFileTransfer and thus is preserved here deferred = self.deferred return deferred diff --git a/tests/federation/test_federation_media.py b/tests/federation/test_federation_media.py index ed6ca7e88d5..88aad03ea40 100644 --- a/tests/federation/test_federation_media.py +++ b/tests/federation/test_federation_media.py @@ -76,10 +76,12 @@ def test_file_download(self) -> None: ) self.pump() self.assertEqual(200, channel.code) + content_type = channel.headers.getRawHeaders("content-type") assert content_type is not None assert "multipart/form-data" in content_type[0] assert "boundary" in content_type[0] + # extract boundary boundary = content_type[0].split("boundary=")[1] # split on boundary and check that json field and expected value exist @@ -90,6 +92,7 @@ def test_file_download(self) -> None: "\r\nContent-Type: application/json\r\n{}" in field for field in stripped ) self.assertTrue(found_json) + # check that text file and expected value exist found_file = any( "\r\nContent-Type: text/plain\r\nfile_to_stream" in field @@ -114,10 +117,12 @@ def test_file_download(self) -> None: ) self.pump() self.assertEqual(200, channel.code) + content_type = channel.headers.getRawHeaders("content-type") assert content_type is not None assert "multipart/form-data" in content_type[0] assert "boundary" in content_type[0] + # extract boundary boundary = content_type[0].split("boundary=")[1] # split on boundary and check that json field and expected value exist @@ -129,6 +134,30 @@ def test_file_download(self) -> None: for field in stripped_bytes ) self.assertTrue(found_json) + # check that png file exists and matches what was uploaded found_file = any(SMALL_PNG in field for field in stripped_bytes) self.assertTrue(found_file) + + + @override_config( + {"experimental_features": {"msc3916_authenticated_media_enabled": False}} + ) + def test_disable_config(self) -> None: + content = io.BytesIO(b"file_to_stream") + content_uri = self.get_success( + self.media_repo.create_content( + "text/plain", + "test_upload", + content, + 46, + UserID.from_string("@user_id:whatever.org"), + ) + ) + channel = self.make_signed_federation_request( + "GET", + f"/_matrix/federation/unstable/org.matrix.msc3916/media/download/{self.hs.hostname}/{content_uri.media_id}", + ) + self.pump() + self.assertEqual(404, channel.code) + self.assertEqual(channel.json_body.get("errcode"), "M_UNRECOGNIZED") \ No newline at end of file From faa346f378185ce9053cbec79340cc1183e20b1d Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Tue, 14 May 2024 13:51:55 -0700 Subject: [PATCH 13/18] lint --- tests/federation/test_federation_media.py | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/tests/federation/test_federation_media.py b/tests/federation/test_federation_media.py index 88aad03ea40..4f9ed87a086 100644 --- a/tests/federation/test_federation_media.py +++ b/tests/federation/test_federation_media.py @@ -139,7 +139,6 @@ def test_file_download(self) -> None: found_file = any(SMALL_PNG in field for field in stripped_bytes) self.assertTrue(found_file) - @override_config( {"experimental_features": {"msc3916_authenticated_media_enabled": False}} ) @@ -160,4 +159,4 @@ def test_disable_config(self) -> None: ) self.pump() self.assertEqual(404, channel.code) - self.assertEqual(channel.json_body.get("errcode"), "M_UNRECOGNIZED") \ No newline at end of file + self.assertEqual(channel.json_body.get("errcode"), "M_UNRECOGNIZED") From 66993ddb69475b12b51c0834de32daf2bce38a6d Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Wed, 5 Jun 2024 12:26:04 -0700 Subject: [PATCH 14/18] use multipart-mixed --- synapse/media/_base.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/synapse/media/_base.py b/synapse/media/_base.py index 6b32392bd19..19bca94170c 100644 --- a/synapse/media/_base.py +++ b/synapse/media/_base.py @@ -303,7 +303,7 @@ async def respond_with_multipart_responder( if media_info.media_length is not None: request.setHeader(b"Content-Length", b"%d" % (media_info.media_length,)) request.setHeader( - b"Content-Type", b"multipart/form-data; boundary=%s" % responder.boundary + b"Content-Type", b"multipart/mixed; boundary=%s" % responder.boundary ) try: From dbf803d6a0b0f12ae51e7ab9c5d7d276c0c3a8e8 Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Wed, 5 Jun 2024 12:26:31 -0700 Subject: [PATCH 15/18] don't load endpoint if storage provider implementation is incompatible --- synapse/federation/transport/server/__init__.py | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/synapse/federation/transport/server/__init__.py b/synapse/federation/transport/server/__init__.py index edaf0196d67..c68ad5f64dd 100644 --- a/synapse/federation/transport/server/__init__.py +++ b/synapse/federation/transport/server/__init__.py @@ -19,6 +19,7 @@ # [This file includes modifications made by New Vector Limited] # # +import inspect import logging from typing import TYPE_CHECKING, Dict, Iterable, List, Optional, Tuple, Type @@ -323,6 +324,21 @@ def register_servlets( ): continue + # don't load the endpoint if the storage provider is incompatible + media_repo = hs.get_media_repository() + load_download_endpoint = True + for provider in media_repo.media_storage.storage_providers: + signature = inspect.signature(provider.backend.fetch) + if "federation" not in signature.parameters: + logger.warning( + "Federation `/download` enpoint will not be enabled as your storage " + "provider is not compatible with this endpoint." + ) + load_download_endpoint = False + + if not load_download_endpoint: + continue + servletclass( hs=hs, authenticator=authenticator, From 0305870483acc0f5ae809373b2daea484bc2649f Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Wed, 5 Jun 2024 12:26:37 -0700 Subject: [PATCH 16/18] tests --- synapse/media/media_storage.py | 4 +- tests/federation/test_federation_media.py | 86 +++++++++++++++++++++-- tests/media/test_media_storage.py | 14 +++- 3 files changed, 93 insertions(+), 11 deletions(-) diff --git a/synapse/media/media_storage.py b/synapse/media/media_storage.py index 5ce91706964..3d2fed117af 100644 --- a/synapse/media/media_storage.py +++ b/synapse/media/media_storage.py @@ -63,7 +63,7 @@ from .filepath import MediaFilePaths if TYPE_CHECKING: - from synapse.media.storage_provider import StorageProvider + from synapse.media.storage_provider import StorageProviderWrapper from synapse.server import HomeServer logger = logging.getLogger(__name__) @@ -86,7 +86,7 @@ def __init__( hs: "HomeServer", local_media_directory: str, filepaths: MediaFilePaths, - storage_providers: Sequence["StorageProvider"], + storage_providers: Sequence["StorageProviderWrapper"], ): self.hs = hs self.reactor = hs.get_reactor() diff --git a/tests/federation/test_federation_media.py b/tests/federation/test_federation_media.py index 4f9ed87a086..1e9ae98595b 100644 --- a/tests/federation/test_federation_media.py +++ b/tests/federation/test_federation_media.py @@ -21,14 +21,20 @@ import os import shutil import tempfile +from typing import Optional from twisted.test.proto_helpers import MemoryReactor +from synapse.media._base import FileInfo, Responder from synapse.media.filepath import MediaFilePaths from synapse.media.media_storage import MediaStorage -from synapse.media.storage_provider import FileStorageProviderBackend +from synapse.media.storage_provider import ( + FileStorageProviderBackend, + StorageProviderWrapper, +) from synapse.server import HomeServer -from synapse.types import UserID +from synapse.storage.databases.main.media_repository import LocalMedia +from synapse.types import JsonDict, UserID from synapse.util import Clock from tests import unittest @@ -36,18 +42,25 @@ from tests.unittest import override_config -class FederationUnstableMediaDownloads(unittest.FederatingHomeserverTestCase): +class FederationUnstableMediaDownloadsTest(unittest.FederatingHomeserverTestCase): + def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None: super().prepare(reactor, clock, hs) self.test_dir = tempfile.mkdtemp(prefix="synapse-tests-") self.addCleanup(shutil.rmtree, self.test_dir) - self.primary_base_path = os.path.join(self.test_dir, "primary") self.secondary_base_path = os.path.join(self.test_dir, "secondary") hs.config.media.media_store_path = self.primary_base_path - storage_providers = [FileStorageProviderBackend(hs, self.secondary_base_path)] + storage_providers = [ + StorageProviderWrapper( + FileStorageProviderBackend(hs, self.secondary_base_path), + store_local=True, + store_remote=False, + store_synchronous=True, + ) + ] self.filepaths = MediaFilePaths(self.primary_base_path) self.media_storage = MediaStorage( @@ -79,7 +92,7 @@ def test_file_download(self) -> None: content_type = channel.headers.getRawHeaders("content-type") assert content_type is not None - assert "multipart/form-data" in content_type[0] + assert "multipart/mixed" in content_type[0] assert "boundary" in content_type[0] # extract boundary @@ -120,7 +133,7 @@ def test_file_download(self) -> None: content_type = channel.headers.getRawHeaders("content-type") assert content_type is not None - assert "multipart/form-data" in content_type[0] + assert "multipart/mixed" in content_type[0] assert "boundary" in content_type[0] # extract boundary @@ -160,3 +173,62 @@ def test_disable_config(self) -> None: self.pump() self.assertEqual(404, channel.code) self.assertEqual(channel.json_body.get("errcode"), "M_UNRECOGNIZED") + + +class FakeFileStorageProviderBackend: + """ + Fake storage provider stub with incompatible `fetch` signature for testing + """ + + def __init__(self, hs: "HomeServer", config: str): + self.hs = hs + self.cache_directory = hs.config.media.media_store_path + self.base_directory = config + + def __str__(self) -> str: + return "FakeFileStorageProviderBackend[%s]" % (self.base_directory,) + + async def fetch( + self, path: str, file_info: FileInfo, media_info: Optional[LocalMedia] = None + ) -> Optional[Responder]: + pass + + +TEST_DIR = tempfile.mkdtemp(prefix="synapse-tests-") + + +class FederationUnstableMediaEndpointCompatibilityTest( + unittest.FederatingHomeserverTestCase +): + + def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None: + super().prepare(reactor, clock, hs) + self.test_dir = TEST_DIR + self.addCleanup(shutil.rmtree, self.test_dir) + self.media_repo = hs.get_media_repository() + + def default_config(self) -> JsonDict: + config = super().default_config() + primary_base_path = os.path.join(TEST_DIR, "primary") + config["media_storage_providers"] = [ + { + "module": "tests.federation.test_federation_media.FakeFileStorageProviderBackend", + "store_local": "True", + "store_remote": "False", + "store_synchronous": "False", + "config": {"directory": primary_base_path}, + } + ] + return config + + @override_config( + {"experimental_features": {"msc3916_authenticated_media_enabled": True}} + ) + def test_incompatible_storage_provider_fails_to_load_endpoint(self) -> None: + channel = self.make_signed_federation_request( + "GET", + f"/_matrix/federation/unstable/org.matrix.msc3916/media/download/{self.hs.hostname}/xyz", + ) + self.pump() + self.assertEqual(404, channel.code) + self.assertEqual(channel.json_body.get("errcode"), "M_UNRECOGNIZED") diff --git a/tests/media/test_media_storage.py b/tests/media/test_media_storage.py index cae67e11c81..5942722c2f4 100644 --- a/tests/media/test_media_storage.py +++ b/tests/media/test_media_storage.py @@ -45,7 +45,10 @@ from synapse.media._base import FileInfo, ThumbnailInfo from synapse.media.filepath import MediaFilePaths from synapse.media.media_storage import MediaStorage, ReadableFileWrapper -from synapse.media.storage_provider import FileStorageProviderBackend +from synapse.media.storage_provider import ( + FileStorageProviderBackend, + StorageProviderWrapper, +) from synapse.module_api import ModuleApi from synapse.module_api.callbacks.spamchecker_callbacks import load_legacy_spam_checkers from synapse.rest import admin @@ -73,7 +76,14 @@ def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None: hs.config.media.media_store_path = self.primary_base_path - storage_providers = [FileStorageProviderBackend(hs, self.secondary_base_path)] + storage_providers = [ + StorageProviderWrapper( + FileStorageProviderBackend(hs, self.secondary_base_path), + store_local=True, + store_remote=False, + store_synchronous=True, + ) + ] self.filepaths = MediaFilePaths(self.primary_base_path) self.media_storage = MediaStorage( From 157e5d7a6e4fa76c4db043ef43b7847da6175335 Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Wed, 5 Jun 2024 15:01:55 -0700 Subject: [PATCH 17/18] remove servername from endpoint --- synapse/federation/transport/server/federation.py | 3 +-- tests/federation/test_federation_media.py | 8 ++++---- 2 files changed, 5 insertions(+), 6 deletions(-) diff --git a/synapse/federation/transport/server/federation.py b/synapse/federation/transport/server/federation.py index 643e6c87a31..1f02451efa5 100644 --- a/synapse/federation/transport/server/federation.py +++ b/synapse/federation/transport/server/federation.py @@ -797,7 +797,7 @@ class FederationUnstableMediaDownloadServlet(BaseFederationServerServlet): item. This endpoint only returns local media. """ - PATH = "/media/download/(?P[^/]*)/(?P[^/]*)" + PATH = "/media/download/(?P[^/]*)" PREFIX = FEDERATION_UNSTABLE_PREFIX + "/org.matrix.msc3916" RATELIMIT = True @@ -816,7 +816,6 @@ async def on_GET( origin: Optional[str], content: Literal[None], request: SynapseRequest, - server_name: str, media_id: str, ) -> None: max_timeout_ms = parse_integer( diff --git a/tests/federation/test_federation_media.py b/tests/federation/test_federation_media.py index 1e9ae98595b..1c89d19e99e 100644 --- a/tests/federation/test_federation_media.py +++ b/tests/federation/test_federation_media.py @@ -85,7 +85,7 @@ def test_file_download(self) -> None: # test with a text file channel = self.make_signed_federation_request( "GET", - f"/_matrix/federation/unstable/org.matrix.msc3916/media/download/{self.hs.hostname}/{content_uri.media_id}", + f"/_matrix/federation/unstable/org.matrix.msc3916/media/download/{content_uri.media_id}", ) self.pump() self.assertEqual(200, channel.code) @@ -126,7 +126,7 @@ def test_file_download(self) -> None: # test with an image file channel = self.make_signed_federation_request( "GET", - f"/_matrix/federation/unstable/org.matrix.msc3916/media/download/{self.hs.hostname}/{content_uri.media_id}", + f"/_matrix/federation/unstable/org.matrix.msc3916/media/download/{content_uri.media_id}", ) self.pump() self.assertEqual(200, channel.code) @@ -168,7 +168,7 @@ def test_disable_config(self) -> None: ) channel = self.make_signed_federation_request( "GET", - f"/_matrix/federation/unstable/org.matrix.msc3916/media/download/{self.hs.hostname}/{content_uri.media_id}", + f"/_matrix/federation/unstable/org.matrix.msc3916/media/download/{content_uri.media_id}", ) self.pump() self.assertEqual(404, channel.code) @@ -227,7 +227,7 @@ def default_config(self) -> JsonDict: def test_incompatible_storage_provider_fails_to_load_endpoint(self) -> None: channel = self.make_signed_federation_request( "GET", - f"/_matrix/federation/unstable/org.matrix.msc3916/media/download/{self.hs.hostname}/xyz", + "/_matrix/federation/unstable/org.matrix.msc3916/media/download/xyz", ) self.pump() self.assertEqual(404, channel.code) From fb7ce78ffe75895946df7de656f78853bbcf8117 Mon Sep 17 00:00:00 2001 From: "H. Shay" Date: Thu, 6 Jun 2024 10:19:46 -0700 Subject: [PATCH 18/18] requested changes --- synapse/federation/transport/server/__init__.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/synapse/federation/transport/server/__init__.py b/synapse/federation/transport/server/__init__.py index c68ad5f64dd..266675c9b85 100644 --- a/synapse/federation/transport/server/__init__.py +++ b/synapse/federation/transport/server/__init__.py @@ -331,10 +331,10 @@ def register_servlets( signature = inspect.signature(provider.backend.fetch) if "federation" not in signature.parameters: logger.warning( - "Federation `/download` enpoint will not be enabled as your storage " - "provider is not compatible with this endpoint." + f"Federation media `/download` endpoint will not be enabled as storage provider {provider.backend} is not compatible with this endpoint." ) load_download_endpoint = False + break if not load_download_endpoint: continue