2022-10-05 12:45:54 +02:00
|
|
|
use ruma::api::client::error::ErrorKind;
|
|
|
|
|
2024-02-09 23:16:06 -05:00
|
|
|
use crate::{
|
|
|
|
database::KeyValueDatabase,
|
|
|
|
service::{self, media::UrlPreviewData},
|
|
|
|
utils, Error, Result,
|
|
|
|
};
|
2020-10-19 15:29:36 +02:00
|
|
|
|
2022-10-05 18:36:12 +02:00
|
|
|
impl service::media::Data for KeyValueDatabase {
|
2022-10-05 20:34:31 +02:00
|
|
|
fn create_file_metadata(
|
|
|
|
&self,
|
|
|
|
mxc: String,
|
|
|
|
width: u32,
|
|
|
|
height: u32,
|
|
|
|
content_disposition: Option<&str>,
|
|
|
|
content_type: Option<&str>,
|
|
|
|
) -> Result<Vec<u8>> {
|
2020-09-14 14:20:38 +02:00
|
|
|
let mut key = mxc.as_bytes().to_vec();
|
|
|
|
key.push(0xff);
|
|
|
|
key.extend_from_slice(&width.to_be_bytes());
|
|
|
|
key.extend_from_slice(&height.to_be_bytes());
|
|
|
|
key.push(0xff);
|
2021-05-30 21:55:43 +02:00
|
|
|
key.extend_from_slice(
|
|
|
|
content_disposition
|
|
|
|
.as_ref()
|
|
|
|
.map(|f| f.as_bytes())
|
|
|
|
.unwrap_or_default(),
|
|
|
|
);
|
2020-09-14 14:20:38 +02:00
|
|
|
key.push(0xff);
|
2020-11-11 14:30:12 -05:00
|
|
|
key.extend_from_slice(
|
|
|
|
content_type
|
|
|
|
.as_ref()
|
|
|
|
.map(|c| c.as_bytes())
|
|
|
|
.unwrap_or_default(),
|
|
|
|
);
|
2020-09-14 14:20:38 +02:00
|
|
|
|
2021-06-04 08:06:12 +04:30
|
|
|
self.mediaid_file.insert(&key, &[])?;
|
2020-09-14 14:20:38 +02:00
|
|
|
|
2022-09-07 13:25:51 +02:00
|
|
|
Ok(key)
|
2020-09-14 14:20:38 +02:00
|
|
|
}
|
|
|
|
|
2022-10-05 20:34:31 +02:00
|
|
|
fn search_file_metadata(
|
|
|
|
&self,
|
|
|
|
mxc: String,
|
|
|
|
width: u32,
|
|
|
|
height: u32,
|
|
|
|
) -> Result<(Option<String>, Option<String>, Vec<u8>)> {
|
2020-05-18 17:53:34 +02:00
|
|
|
let mut prefix = mxc.as_bytes().to_vec();
|
|
|
|
prefix.push(0xff);
|
2022-10-08 13:02:52 +02:00
|
|
|
prefix.extend_from_slice(&width.to_be_bytes());
|
|
|
|
prefix.extend_from_slice(&height.to_be_bytes());
|
2020-05-19 18:31:34 +02:00
|
|
|
prefix.push(0xff);
|
2020-05-18 17:53:34 +02:00
|
|
|
|
2022-10-05 20:34:31 +02:00
|
|
|
let (key, _) = self
|
|
|
|
.mediaid_file
|
|
|
|
.scan_prefix(prefix)
|
|
|
|
.next()
|
|
|
|
.ok_or(Error::BadRequest(ErrorKind::NotFound, "Media not found"))?;
|
2021-06-06 16:58:32 +04:30
|
|
|
|
2022-09-07 13:25:51 +02:00
|
|
|
let mut parts = key.rsplit(|&b| b == 0xff);
|
2020-05-19 18:31:34 +02:00
|
|
|
|
2022-09-07 13:25:51 +02:00
|
|
|
let content_type = parts
|
|
|
|
.next()
|
|
|
|
.map(|bytes| {
|
|
|
|
utils::string_from_bytes(bytes).map_err(|_| {
|
|
|
|
Error::bad_database("Content type in mediaid_file is invalid unicode.")
|
2020-11-18 08:36:12 -05:00
|
|
|
})
|
2022-09-07 13:25:51 +02:00
|
|
|
})
|
|
|
|
.transpose()?;
|
2021-06-06 16:58:32 +04:30
|
|
|
|
2022-09-07 13:25:51 +02:00
|
|
|
let content_disposition_bytes = parts
|
|
|
|
.next()
|
|
|
|
.ok_or_else(|| Error::bad_database("Media ID in db is invalid."))?;
|
2020-05-19 18:31:34 +02:00
|
|
|
|
2022-09-07 13:25:51 +02:00
|
|
|
let content_disposition = if content_disposition_bytes.is_empty() {
|
|
|
|
None
|
2020-05-18 17:53:34 +02:00
|
|
|
} else {
|
2022-09-07 13:25:51 +02:00
|
|
|
Some(
|
|
|
|
utils::string_from_bytes(content_disposition_bytes).map_err(|_| {
|
2022-10-05 20:34:31 +02:00
|
|
|
Error::bad_database("Content Disposition in mediaid_file is invalid unicode.")
|
2022-09-07 13:25:51 +02:00
|
|
|
})?,
|
|
|
|
)
|
|
|
|
};
|
|
|
|
Ok((content_disposition, content_type, key))
|
2020-05-18 17:53:34 +02:00
|
|
|
}
|
2024-02-09 23:16:06 -05:00
|
|
|
|
|
|
|
fn remove_url_preview(&self, url: &str) -> Result<()> {
|
|
|
|
self.url_previews.remove(url.as_bytes())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn set_url_preview(
|
|
|
|
&self,
|
|
|
|
url: &str,
|
|
|
|
data: &UrlPreviewData,
|
|
|
|
timestamp: std::time::Duration,
|
|
|
|
) -> Result<()> {
|
|
|
|
let mut value = Vec::<u8>::new();
|
|
|
|
value.extend_from_slice(×tamp.as_secs().to_be_bytes());
|
|
|
|
value.push(0xff);
|
|
|
|
value.extend_from_slice(
|
|
|
|
data.title
|
|
|
|
.as_ref()
|
|
|
|
.map(|t| t.as_bytes())
|
|
|
|
.unwrap_or_default(),
|
|
|
|
);
|
|
|
|
value.push(0xff);
|
|
|
|
value.extend_from_slice(
|
|
|
|
data.description
|
|
|
|
.as_ref()
|
|
|
|
.map(|d| d.as_bytes())
|
|
|
|
.unwrap_or_default(),
|
|
|
|
);
|
|
|
|
value.push(0xff);
|
|
|
|
value.extend_from_slice(
|
|
|
|
data.image
|
|
|
|
.as_ref()
|
|
|
|
.map(|i| i.as_bytes())
|
|
|
|
.unwrap_or_default(),
|
|
|
|
);
|
|
|
|
value.push(0xff);
|
|
|
|
value.extend_from_slice(&data.image_size.unwrap_or(0).to_be_bytes());
|
|
|
|
value.push(0xff);
|
|
|
|
value.extend_from_slice(&data.image_width.unwrap_or(0).to_be_bytes());
|
|
|
|
value.push(0xff);
|
|
|
|
value.extend_from_slice(&data.image_height.unwrap_or(0).to_be_bytes());
|
|
|
|
|
|
|
|
self.url_previews.insert(url.as_bytes(), &value)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_url_preview(&self, url: &str) -> Option<UrlPreviewData> {
|
|
|
|
let values = self.url_previews.get(url.as_bytes()).ok()??;
|
|
|
|
|
|
|
|
let mut values = values.split(|&b| b == 0xff);
|
|
|
|
|
|
|
|
let _ts = match values
|
|
|
|
.next()
|
|
|
|
.map(|b| u64::from_be_bytes(b.try_into().expect("valid BE array")))
|
|
|
|
{
|
|
|
|
Some(0) => None,
|
|
|
|
x => x,
|
|
|
|
};
|
|
|
|
let title = match values
|
|
|
|
.next()
|
|
|
|
.and_then(|b| String::from_utf8(b.to_vec()).ok())
|
|
|
|
{
|
|
|
|
Some(s) if s.is_empty() => None,
|
|
|
|
x => x,
|
|
|
|
};
|
|
|
|
let description = match values
|
|
|
|
.next()
|
|
|
|
.and_then(|b| String::from_utf8(b.to_vec()).ok())
|
|
|
|
{
|
|
|
|
Some(s) if s.is_empty() => None,
|
|
|
|
x => x,
|
|
|
|
};
|
|
|
|
let image = match values
|
|
|
|
.next()
|
|
|
|
.and_then(|b| String::from_utf8(b.to_vec()).ok())
|
|
|
|
{
|
|
|
|
Some(s) if s.is_empty() => None,
|
|
|
|
x => x,
|
|
|
|
};
|
|
|
|
let image_size = match values
|
|
|
|
.next()
|
|
|
|
.map(|b| usize::from_be_bytes(b.try_into().expect("valid BE array")))
|
|
|
|
{
|
|
|
|
Some(0) => None,
|
|
|
|
x => x,
|
|
|
|
};
|
|
|
|
let image_width = match values
|
|
|
|
.next()
|
|
|
|
.map(|b| u32::from_be_bytes(b.try_into().expect("valid BE array")))
|
|
|
|
{
|
|
|
|
Some(0) => None,
|
|
|
|
x => x,
|
|
|
|
};
|
|
|
|
let image_height = match values
|
|
|
|
.next()
|
|
|
|
.map(|b| u32::from_be_bytes(b.try_into().expect("valid BE array")))
|
|
|
|
{
|
|
|
|
Some(0) => None,
|
|
|
|
x => x,
|
|
|
|
};
|
|
|
|
|
|
|
|
Some(UrlPreviewData {
|
|
|
|
title,
|
|
|
|
description,
|
|
|
|
image,
|
|
|
|
image_size,
|
|
|
|
image_width,
|
|
|
|
image_height,
|
|
|
|
})
|
|
|
|
}
|
2020-05-18 17:53:34 +02:00
|
|
|
}
|