From f74e8edc129af4e49e2987342de290695ec2de14 Mon Sep 17 00:00:00 2001 From: Sumner Evans Date: Fri, 15 Apr 2022 09:46:17 -0600 Subject: [PATCH] media/create: add MSC2246 create endpoint Signed-off-by: Sumner Evans --- synapse/rest/media/v1/create_resource.py | 84 +++++++++++++++++++ synapse/rest/media/v1/media_repository.py | 40 +++++++++ .../databases/main/media_repository.py | 18 ++++ ...sc2246_add_unused_expires_at_for_media.sql | 20 +++++ 4 files changed, 162 insertions(+) create mode 100644 synapse/rest/media/v1/create_resource.py create mode 100644 synapse/storage/schema/main/delta/68/06_msc2246_add_unused_expires_at_for_media.sql diff --git a/synapse/rest/media/v1/create_resource.py b/synapse/rest/media/v1/create_resource.py new file mode 100644 index 000000000000..1dfe09e3904a --- /dev/null +++ b/synapse/rest/media/v1/create_resource.py @@ -0,0 +1,84 @@ +# Copyright 2014-2016 OpenMarket Ltd +# Copyright 2020-2021 The Matrix.org Foundation C.I.C. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +import logging +from typing import TYPE_CHECKING + +from synapse.api.errors import LimitExceededError +from synapse.api.ratelimiting import Ratelimiter +from synapse.http.server import DirectServeJsonResource, respond_with_json +from synapse.http.site import SynapseRequest + +if TYPE_CHECKING: + from synapse.rest.media.v1.media_repository import MediaRepository + from synapse.server import HomeServer + + +logger = logging.getLogger(__name__) + + +class CreateResource(DirectServeJsonResource): + isLeaf = True + + def __init__(self, hs: "HomeServer", media_repo: "MediaRepository"): + super().__init__() + + self.media_repo = media_repo + self.clock = hs.get_clock() + self.auth = hs.get_auth() + + # A rate limiter for creating new media IDs. + self._create_media_rate_limiter = Ratelimiter( + store=hs.get_datastores().main, + clock=self.clock, + rate_hz=hs.config.ratelimiting.rc_media_create.per_second, + burst_count=hs.config.ratelimiting.rc_media_create.burst_count, + ) + + async def _async_render_OPTIONS(self, request: SynapseRequest) -> None: + respond_with_json(request, 200, {}, send_cors=True) + + async def _async_render_POST(self, request: SynapseRequest) -> None: + requester = await self.auth.get_user_by_req(request) + + # If the create media requests for the user are over the limit, drop + # them. + allowed, time_allowed = await self._create_media_rate_limiter.can_do_action( + requester + ) + if not allowed: + time_now_s = self.clock.time() + raise LimitExceededError( + retry_after_ms=int(1000 * (time_allowed - time_now_s)) + ) + + content_uri, unused_expires_at = await self.media_repo.create_media_id( + requester.user + ) + + logger.info( + "Created Media URI %r that if unused will expire at %d", + content_uri, + unused_expires_at, + ) + respond_with_json( + request, + 200, + { + "content_uri": content_uri, + "unused_expires_at": unused_expires_at, + }, + send_cors=True, + ) diff --git a/synapse/rest/media/v1/media_repository.py b/synapse/rest/media/v1/media_repository.py index e7999e9a7269..bb2408a78e7c 100644 --- a/synapse/rest/media/v1/media_repository.py +++ b/synapse/rest/media/v1/media_repository.py @@ -37,6 +37,7 @@ from synapse.http.site import SynapseRequest from synapse.logging.context import defer_to_thread from synapse.metrics.background_process_metrics import run_as_background_process +from synapse.rest.media.v1.create_resource import CreateResource from synapse.types import UserID from synapse.util.async_helpers import Linearizer from synapse.util.retryutils import NotRetryingDestination @@ -84,6 +85,7 @@ def __init__(self, hs: "HomeServer"): self.store = hs.get_datastores().main self.max_upload_size = hs.config.media.max_upload_size self.max_image_pixels = hs.config.media.max_image_pixels + self.unused_expiration_time = hs.config.media.unused_expiration_time Thumbnailer.set_limits(self.max_image_pixels) @@ -181,6 +183,27 @@ def mark_recently_accessed(self, server_name: Optional[str], media_id: str) -> N else: self.recently_accessed_locals.add(media_id) + async def create_media_id(self, auth_user: UserID) -> Tuple[str, int]: + """Create and store a media ID for a local user and return the mxc URL + + Args: + auth_user: The user_id of the uploader + + Returns: + The mxc url of the stored content + """ + media_id = random_string(24) + now = self.clock.time_msec() + # After the configured amount of time, don't allow the upload to start. + unused_expires_at = now + self.unused_expiration_time + await self.store.store_local_media_id( + media_id=media_id, + time_now_ms=now, + user_id=auth_user, + unused_expires_at=unused_expires_at, + ) + return f"mxc://{self.server_name}/{media_id}", unused_expires_at + async def create_content( self, media_type: str, @@ -1043,6 +1066,20 @@ class MediaVersion(Enum): UNSTABLE = b"unstable" +class MSC2246MediaRepositoryResource(Resource): + """Media creation and asynchronous uploading. + + This resource implements MSC2246 + https://github.com/matrix-org/matrix-spec-proposals/pull/2246 + """ + + def __init__(self, hs: "HomeServer"): + super().__init__() + media_repo = hs.get_media_repository() + + self.putChild(b"create", CreateResource(hs, media_repo)) + + class VersionedMediaRepositoryResource(Resource): """File uploading and downloading. @@ -1108,6 +1145,9 @@ def __init__(self, hs: "HomeServer", version: MediaVersion): ) self.putChild(b"config", MediaConfigResource(hs)) + if version == MediaVersion.UNSTABLE and hs.config.experimental.msc2246_enabled: + self.putChild(b"fi.mau.msc2246", MSC2246MediaRepositoryResource(hs)) + class MediaRepositoryResource(Resource): """ diff --git a/synapse/storage/databases/main/media_repository.py b/synapse/storage/databases/main/media_repository.py index deffdc19ce9f..a451e78c8190 100644 --- a/synapse/storage/databases/main/media_repository.py +++ b/synapse/storage/databases/main/media_repository.py @@ -302,6 +302,24 @@ def _get_local_media_before_txn(txn: LoggingTransaction) -> List[str]: "get_local_media_before", _get_local_media_before_txn ) + async def store_local_media_id( + self, + media_id: str, + time_now_ms: int, + user_id: UserID, + unused_expires_at: int, + ) -> None: + await self.db_pool.simple_insert( + "local_media_repository", + { + "media_id": media_id, + "created_ts": time_now_ms, + "user_id": user_id.to_string(), + "unused_expires_at": unused_expires_at, + }, + desc="store_local_media_id", + ) + async def store_local_media( self, media_id: str, diff --git a/synapse/storage/schema/main/delta/68/06_msc2246_add_unused_expires_at_for_media.sql b/synapse/storage/schema/main/delta/68/06_msc2246_add_unused_expires_at_for_media.sql new file mode 100644 index 000000000000..8e9438bad7a8 --- /dev/null +++ b/synapse/storage/schema/main/delta/68/06_msc2246_add_unused_expires_at_for_media.sql @@ -0,0 +1,20 @@ +/* Copyright 2022 The Matrix.org Foundation C.I.C + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +-- Add new colums to the `local_media_repository` to keep track of when the +-- media ID must be used by. This is to support MSC2246 async uploads. + +ALTER TABLE local_media_repository + ADD COLUMN unused_expires_at BIGINT DEFAULT NULL;