1//! `GET /_matrix/media/*/download/{serverName}/{mediaId}`
2//!
3//! Retrieve content from the media store.
45pub mod v3 {
6//! `/v3/` ([spec])
7 //!
8 //! [spec]: https://spec.matrix.org/latest/client-server-api/#get_matrixmediav3downloadservernamemediaid
910use std::time::Duration;
1112use http::header::{CONTENT_DISPOSITION, CONTENT_TYPE};
13use ruma_common::{
14 api::{request, response, Metadata},
15 http_headers::ContentDisposition,
16 metadata, IdParseError, MxcUri, OwnedServerName,
17 };
1819use crate::http_headers::CROSS_ORIGIN_RESOURCE_POLICY;
2021const METADATA: Metadata = metadata! {
22 method: GET,
23 rate_limited: false,
24 authentication: None,
25 history: {
261.0 => "/_matrix/media/r0/download/:server_name/:media_id",
271.1 => "/_matrix/media/v3/download/:server_name/:media_id",
281.11 => deprecated,
29 }
30 };
3132/// Request type for the `get_media_content` endpoint.
33#[request(error = crate::Error)]
34 #[deprecated = "\
35 Since Matrix 1.11, clients should use `authenticated_media::get_content::v1::Request` \
36 instead if the homeserver supports it.\
37 "]
38pub struct Request {
39/// The server name from the mxc:// URI (the authoritory component).
40#[ruma_api(path)]
41pub server_name: OwnedServerName,
4243/// The media ID from the mxc:// URI (the path component).
44#[ruma_api(path)]
45pub media_id: String,
4647/// Whether to fetch media deemed remote.
48 ///
49 /// Used to prevent routing loops. Defaults to `true`.
50#[ruma_api(query)]
51 #[serde(
52 default = "ruma_common::serde::default_true",
53 skip_serializing_if = "ruma_common::serde::is_true"
54)]
55pub allow_remote: bool,
5657/// The maximum duration that the client is willing to wait to start receiving data, in the
58 /// case that the content has not yet been uploaded.
59 ///
60 /// The default value is 20 seconds.
61#[ruma_api(query)]
62 #[serde(
63 with = "ruma_common::serde::duration::ms",
64 default = "ruma_common::media::default_download_timeout",
65 skip_serializing_if = "ruma_common::media::is_default_download_timeout"
66)]
67pub timeout_ms: Duration,
6869/// Whether the server may return a 307 or 308 redirect response that points at the
70 /// relevant media content.
71 ///
72 /// Unless explicitly set to `true`, the server must return the media content itself.
73#[ruma_api(query)]
74 #[serde(default, skip_serializing_if = "ruma_common::serde::is_default")]
75pub allow_redirect: bool,
76 }
7778/// Response type for the `get_media_content` endpoint.
79#[response(error = crate::Error)]
80pub struct Response {
81/// The content that was previously uploaded.
82#[ruma_api(raw_body)]
83pub file: Vec<u8>,
8485/// The content type of the file that was previously uploaded.
86#[ruma_api(header = CONTENT_TYPE)]
87pub content_type: Option<String>,
8889/// The value of the `Content-Disposition` HTTP header, possibly containing the name of the
90 /// file that was previously uploaded.
91#[ruma_api(header = CONTENT_DISPOSITION)]
92pub content_disposition: Option<ContentDisposition>,
9394/// The value of the `Cross-Origin-Resource-Policy` HTTP header.
95 ///
96 /// See [MDN] for the syntax.
97 ///
98 /// [MDN]: https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Cross-Origin-Resource-Policy#syntax
99#[ruma_api(header = CROSS_ORIGIN_RESOURCE_POLICY)]
100pub cross_origin_resource_policy: Option<String>,
101 }
102103#[allow(deprecated)]
104impl Request {
105/// Creates a new `Request` with the given media ID and server name.
106pub fn new(media_id: String, server_name: OwnedServerName) -> Self {
107Self {
108 media_id,
109 server_name,
110 allow_remote: true,
111 timeout_ms: ruma_common::media::default_download_timeout(),
112 allow_redirect: false,
113 }
114 }
115116/// Creates a new `Request` with the given url.
117pub fn from_url(url: &MxcUri) -> Result<Self, IdParseError> {
118let (server_name, media_id) = url.parts()?;
119120Ok(Self::new(media_id.to_owned(), server_name.to_owned()))
121 }
122 }
123124impl Response {
125/// Creates a new `Response` with the given file contents.
126 ///
127 /// The Cross-Origin Resource Policy defaults to `cross-origin`.
128pub fn new(
129 file: Vec<u8>,
130 content_type: String,
131 content_disposition: ContentDisposition,
132 ) -> Self {
133Self {
134 file,
135 content_type: Some(content_type),
136 content_disposition: Some(content_disposition),
137 cross_origin_resource_policy: Some("cross-origin".to_owned()),
138 }
139 }
140 }
141}