conduit/src/client_server/media.rs

178 lines
5.2 KiB
Rust
Raw Normal View History

2020-07-30 17:14:47 +01:00
use super::State;
2020-12-19 15:00:11 +00:00
use crate::{database::media::FileMeta, utils, ConduitResult, Database, Error, Ruma};
2020-07-30 17:14:47 +01:00
use ruma::api::client::{
error::ErrorKind,
r0::media::{create_content, get_content, get_content_thumbnail, get_media_config},
};
#[cfg(feature = "conduit_bin")]
use rocket::{get, post};
2021-06-08 17:10:00 +01:00
use std::{convert::TryInto, sync::Arc};
2020-07-30 17:14:47 +01:00
const MXC_LENGTH: usize = 32;
2020-07-30 17:14:47 +01:00
#[cfg_attr(feature = "conduit_bin", get("/_matrix/media/r0/config"))]
2021-02-28 11:41:03 +00:00
#[tracing::instrument(skip(db))]
pub async fn get_media_config_route(
2021-06-08 17:10:00 +01:00
db: State<'_, Arc<Database>>,
2020-07-30 17:14:47 +01:00
) -> ConduitResult<get_media_config::Response> {
Ok(get_media_config::Response {
upload_size: db.globals.max_request_size().into(),
}
.into())
}
#[cfg_attr(
feature = "conduit_bin",
post("/_matrix/media/r0/upload", data = "<body>")
)]
2021-02-28 11:41:03 +00:00
#[tracing::instrument(skip(db, body))]
pub async fn create_content_route(
2021-06-08 17:10:00 +01:00
db: State<'_, Arc<Database>>,
2020-09-08 16:32:03 +01:00
body: Ruma<create_content::Request<'_>>,
2020-07-30 17:14:47 +01:00
) -> ConduitResult<create_content::Response> {
let mxc = format!(
"mxc://{}/{}",
db.globals.server_name(),
utils::random_string(MXC_LENGTH)
);
2021-06-06 13:28:32 +01:00
db.media
.create(
mxc.clone(),
&db.globals,
&body
.filename
.as_ref()
.map(|filename| "inline; filename=".to_owned() + filename)
.as_deref(),
&body.content_type.as_deref(),
&body.file,
)
.await?;
2020-07-30 17:14:47 +01:00
db.flush().await?;
2020-12-19 15:00:11 +00:00
Ok(create_content::Response {
content_uri: mxc.try_into().expect("Invalid mxc:// URI"),
2020-12-19 15:00:11 +00:00
blurhash: None,
}
.into())
2020-07-30 17:14:47 +01:00
}
#[cfg_attr(
feature = "conduit_bin",
get("/_matrix/media/r0/download/<_>/<_>", data = "<body>")
2020-07-30 17:14:47 +01:00
)]
2021-02-28 11:41:03 +00:00
#[tracing::instrument(skip(db, body))]
pub async fn get_content_route(
2021-06-08 17:10:00 +01:00
db: State<'_, Arc<Database>>,
2020-09-08 16:32:03 +01:00
body: Ruma<get_content::Request<'_>>,
2020-07-30 17:14:47 +01:00
) -> ConduitResult<get_content::Response> {
let mxc = format!("mxc://{}/{}", body.server_name, body.media_id);
2020-07-30 17:14:47 +01:00
if let Some(FileMeta {
content_disposition,
2020-07-30 17:14:47 +01:00
content_type,
file,
2021-06-04 04:36:12 +01:00
}) = db.media.get(&db.globals, &mxc).await?
2020-07-30 17:14:47 +01:00
{
Ok(get_content::Response {
file,
content_type,
content_disposition,
2020-07-30 17:14:47 +01:00
}
.into())
} else if &*body.server_name != db.globals.server_name() && body.allow_remote {
2020-12-19 15:00:11 +00:00
let get_content_response = db
.sending
.send_federation_request(
&db.globals,
&body.server_name,
2020-12-19 15:00:11 +00:00
get_content::Request {
allow_remote: false,
server_name: &body.server_name,
media_id: &body.media_id,
},
)
.await?;
2021-06-06 13:28:32 +01:00
db.media
.create(
mxc,
&db.globals,
&get_content_response.content_disposition.as_deref(),
&get_content_response.content_type.as_deref(),
&get_content_response.file,
)
.await?;
Ok(get_content_response.into())
2020-07-30 17:14:47 +01:00
} else {
Err(Error::BadRequest(ErrorKind::NotFound, "Media not found."))
}
}
#[cfg_attr(
feature = "conduit_bin",
get("/_matrix/media/r0/thumbnail/<_>/<_>", data = "<body>")
2020-07-30 17:14:47 +01:00
)]
2021-02-28 11:41:03 +00:00
#[tracing::instrument(skip(db, body))]
pub async fn get_content_thumbnail_route(
2021-06-08 17:10:00 +01:00
db: State<'_, Arc<Database>>,
2020-09-08 16:32:03 +01:00
body: Ruma<get_content_thumbnail::Request<'_>>,
2020-07-30 17:14:47 +01:00
) -> ConduitResult<get_content_thumbnail::Response> {
let mxc = format!("mxc://{}/{}", body.server_name, body.media_id);
2020-07-30 17:14:47 +01:00
if let Some(FileMeta {
content_type, file, ..
2021-06-06 13:28:32 +01:00
}) = db
.media
.get_thumbnail(
mxc.clone(),
&db.globals,
body.width
.try_into()
.map_err(|_| Error::BadRequest(ErrorKind::InvalidParam, "Width is invalid."))?,
body.height
.try_into()
.map_err(|_| Error::BadRequest(ErrorKind::InvalidParam, "Width is invalid."))?,
)
.await?
{
Ok(get_content_thumbnail::Response { file, content_type }.into())
} else if &*body.server_name != db.globals.server_name() && body.allow_remote {
2020-12-19 15:00:11 +00:00
let get_thumbnail_response = db
.sending
.send_federation_request(
&db.globals,
&body.server_name,
2020-12-19 15:00:11 +00:00
get_content_thumbnail::Request {
allow_remote: false,
height: body.height,
width: body.width,
method: body.method.clone(),
2020-12-19 15:00:11 +00:00
server_name: &body.server_name,
media_id: &body.media_id,
},
)
.await?;
2021-06-06 13:28:32 +01:00
db.media
.upload_thumbnail(
mxc,
&db.globals,
&None,
&get_thumbnail_response.content_type,
body.width.try_into().expect("all UInts are valid u32s"),
body.height.try_into().expect("all UInts are valid u32s"),
&get_thumbnail_response.file,
)
.await?;
Ok(get_thumbnail_response.into())
2020-07-30 17:14:47 +01:00
} else {
Err(Error::BadRequest(ErrorKind::NotFound, "Media not found."))
}
}