2020-01-21 07:02:03 +01:00
|
|
|
use crate::{
|
2022-12-16 22:51:14 +01:00
|
|
|
ap_url, instance::Instance, safe_string::SafeString, schema::medias, users::User, Connection,
|
|
|
|
Error, Result, CONFIG,
|
2020-01-21 07:02:03 +01:00
|
|
|
};
|
2022-05-02 17:54:37 +02:00
|
|
|
use activitystreams::{object::Image, prelude::*};
|
2018-11-24 12:44:17 +01:00
|
|
|
use diesel::{self, ExpressionMethods, QueryDsl, RunQueryDsl};
|
2018-10-31 10:40:20 +01:00
|
|
|
use guid_create::GUID;
|
2019-04-17 19:31:47 +02:00
|
|
|
use plume_common::{
|
2022-05-02 17:54:37 +02:00
|
|
|
activity_pub::{inbox::FromId, request, ToAsString, ToAsUri},
|
2022-01-06 21:38:17 +01:00
|
|
|
utils::{escape, MediaProcessor},
|
2019-04-17 19:31:47 +02:00
|
|
|
};
|
2021-02-17 17:23:57 +01:00
|
|
|
use std::{
|
2021-02-18 03:08:40 +01:00
|
|
|
fs::{self, DirBuilder},
|
2021-02-25 03:01:47 +01:00
|
|
|
path::{self, Path, PathBuf},
|
2021-02-17 17:23:57 +01:00
|
|
|
};
|
|
|
|
use tracing::warn;
|
|
|
|
use url::Url;
|
|
|
|
|
2023-05-12 15:40:36 +02:00
|
|
|
#[cfg(feature = "s3")]
|
|
|
|
use crate::config::S3Config;
|
|
|
|
|
2021-02-17 17:23:57 +01:00
|
|
|
const REMOTE_MEDIA_DIRECTORY: &str = "remote";
|
2018-09-02 13:34:48 +02:00
|
|
|
|
2021-02-23 17:05:19 +01:00
|
|
|
#[derive(Clone, Identifiable, Queryable, AsChangeset)]
|
2018-09-02 13:34:48 +02:00
|
|
|
pub struct Media {
|
|
|
|
pub id: i32,
|
|
|
|
pub file_path: String,
|
|
|
|
pub alt_text: String,
|
|
|
|
pub is_remote: bool,
|
|
|
|
pub remote_url: Option<String>,
|
|
|
|
pub sensitive: bool,
|
2018-09-02 22:55:42 +02:00
|
|
|
pub content_warning: Option<String>,
|
2018-11-24 12:44:17 +01:00
|
|
|
pub owner_id: i32,
|
2018-09-02 13:34:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Insertable)]
|
|
|
|
#[table_name = "medias"]
|
|
|
|
pub struct NewMedia {
|
|
|
|
pub file_path: String,
|
|
|
|
pub alt_text: String,
|
|
|
|
pub is_remote: bool,
|
|
|
|
pub remote_url: Option<String>,
|
|
|
|
pub sensitive: bool,
|
2018-09-02 22:55:42 +02:00
|
|
|
pub content_warning: Option<String>,
|
2018-11-24 12:44:17 +01:00
|
|
|
pub owner_id: i32,
|
2018-09-02 13:34:48 +02:00
|
|
|
}
|
|
|
|
|
2023-01-02 18:45:13 +01:00
|
|
|
#[derive(PartialEq, Eq)]
|
2018-12-06 18:54:16 +01:00
|
|
|
pub enum MediaCategory {
|
|
|
|
Image,
|
|
|
|
Audio,
|
|
|
|
Video,
|
|
|
|
Unknown,
|
|
|
|
}
|
|
|
|
|
2019-03-06 14:11:36 +01:00
|
|
|
impl MediaCategory {
|
|
|
|
pub fn to_string(&self) -> &str {
|
|
|
|
match *self {
|
|
|
|
MediaCategory::Image => "image",
|
|
|
|
MediaCategory::Audio => "audio",
|
|
|
|
MediaCategory::Video => "video",
|
|
|
|
MediaCategory::Unknown => "unknown",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-02 13:34:48 +02:00
|
|
|
impl Media {
|
|
|
|
insert!(medias, NewMedia);
|
|
|
|
get!(medias);
|
2021-02-23 17:04:31 +01:00
|
|
|
find_by!(medias, find_by_file_path, file_path as &str);
|
2019-08-28 11:37:03 +02:00
|
|
|
|
|
|
|
pub fn for_user(conn: &Connection, owner: i32) -> Result<Vec<Media>> {
|
|
|
|
medias::table
|
|
|
|
.filter(medias::owner_id.eq(owner))
|
|
|
|
.order(medias::id.desc())
|
|
|
|
.load::<Self>(conn)
|
|
|
|
.map_err(Error::from)
|
|
|
|
}
|
2018-09-02 22:55:42 +02:00
|
|
|
|
2018-12-29 09:36:07 +01:00
|
|
|
pub fn list_all_medias(conn: &Connection) -> Result<Vec<Media>> {
|
2019-03-20 17:56:17 +01:00
|
|
|
medias::table.load::<Media>(conn).map_err(Error::from)
|
2018-11-24 12:44:17 +01:00
|
|
|
}
|
|
|
|
|
2019-03-20 17:56:17 +01:00
|
|
|
pub fn page_for_user(
|
|
|
|
conn: &Connection,
|
|
|
|
user: &User,
|
|
|
|
(min, max): (i32, i32),
|
|
|
|
) -> Result<Vec<Media>> {
|
2019-03-06 14:11:36 +01:00
|
|
|
medias::table
|
|
|
|
.filter(medias::owner_id.eq(user.id))
|
2019-08-28 11:37:03 +02:00
|
|
|
.order(medias::id.desc())
|
2019-03-19 14:37:56 +01:00
|
|
|
.offset(i64::from(min))
|
|
|
|
.limit(i64::from(max - min))
|
2019-03-06 14:11:36 +01:00
|
|
|
.load::<Media>(conn)
|
|
|
|
.map_err(Error::from)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn count_for_user(conn: &Connection, user: &User) -> Result<i64> {
|
|
|
|
medias::table
|
|
|
|
.filter(medias::owner_id.eq(user.id))
|
|
|
|
.count()
|
|
|
|
.get_result(conn)
|
|
|
|
.map_err(Error::from)
|
|
|
|
}
|
|
|
|
|
2018-12-06 18:54:16 +01:00
|
|
|
pub fn category(&self) -> MediaCategory {
|
2019-01-27 13:44:09 +01:00
|
|
|
match &*self
|
2018-11-24 12:44:17 +01:00
|
|
|
.file_path
|
2021-11-27 23:53:13 +01:00
|
|
|
.rsplit_once('.')
|
|
|
|
.map(|x| x.1)
|
2023-05-12 15:40:36 +02:00
|
|
|
.unwrap_or("")
|
2019-01-27 13:44:09 +01:00
|
|
|
.to_lowercase()
|
2018-11-24 12:44:17 +01:00
|
|
|
{
|
2018-12-06 18:54:16 +01:00
|
|
|
"png" | "jpg" | "jpeg" | "gif" | "svg" => MediaCategory::Image,
|
|
|
|
"mp3" | "wav" | "flac" => MediaCategory::Audio,
|
|
|
|
"mp4" | "avi" | "webm" | "mov" => MediaCategory::Video,
|
|
|
|
_ => MediaCategory::Unknown,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-10 22:59:34 +02:00
|
|
|
pub fn html(&self) -> Result<SafeString> {
|
|
|
|
let url = self.url()?;
|
2018-12-29 09:36:07 +01:00
|
|
|
Ok(match self.category() {
|
2019-03-06 14:11:36 +01:00
|
|
|
MediaCategory::Image => SafeString::trusted(&format!(
|
2018-12-06 18:54:16 +01:00
|
|
|
r#"<img src="{}" alt="{}" title="{}">"#,
|
2020-01-19 12:52:32 +01:00
|
|
|
url,
|
|
|
|
escape(&self.alt_text),
|
|
|
|
escape(&self.alt_text)
|
2018-12-06 18:54:16 +01:00
|
|
|
)),
|
2019-03-06 14:11:36 +01:00
|
|
|
MediaCategory::Audio => SafeString::trusted(&format!(
|
|
|
|
r#"<div class="media-preview audio"></div><audio src="{}" title="{}" controls></audio>"#,
|
2020-01-19 12:52:32 +01:00
|
|
|
url,
|
|
|
|
escape(&self.alt_text)
|
2018-12-06 18:54:16 +01:00
|
|
|
)),
|
2019-03-06 14:11:36 +01:00
|
|
|
MediaCategory::Video => SafeString::trusted(&format!(
|
|
|
|
r#"<video src="{}" title="{}" controls></video>"#,
|
2020-01-19 12:52:32 +01:00
|
|
|
url,
|
|
|
|
escape(&self.alt_text)
|
2018-12-06 18:54:16 +01:00
|
|
|
)),
|
2019-03-06 14:11:36 +01:00
|
|
|
MediaCategory::Unknown => SafeString::trusted(&format!(
|
|
|
|
r#"<a href="{}" class="media-preview unknown"></a>"#,
|
|
|
|
url,
|
|
|
|
)),
|
2018-12-29 09:36:07 +01:00
|
|
|
})
|
2018-12-06 18:54:16 +01:00
|
|
|
}
|
|
|
|
|
2019-05-10 22:59:34 +02:00
|
|
|
pub fn markdown(&self) -> Result<SafeString> {
|
2018-12-29 09:36:07 +01:00
|
|
|
Ok(match self.category() {
|
2019-03-20 17:56:17 +01:00
|
|
|
MediaCategory::Image => {
|
2019-04-06 19:20:33 +02:00
|
|
|
SafeString::new(&format!("![{}]({})", escape(&self.alt_text), self.id))
|
2019-03-20 17:56:17 +01:00
|
|
|
}
|
2019-05-10 22:59:34 +02:00
|
|
|
MediaCategory::Audio | MediaCategory::Video => self.html()?,
|
2018-12-06 18:54:16 +01:00
|
|
|
MediaCategory::Unknown => SafeString::new(""),
|
2018-12-29 09:36:07 +01:00
|
|
|
})
|
2018-09-02 22:55:42 +02:00
|
|
|
}
|
|
|
|
|
2023-05-12 15:40:36 +02:00
|
|
|
/// Returns full file path for medias stored in the local media directory.
|
|
|
|
pub fn local_path(&self) -> Option<PathBuf> {
|
|
|
|
if self.file_path.is_empty() {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
if CONFIG.s3.is_some() {
|
|
|
|
#[cfg(feature="s3")]
|
|
|
|
unreachable!("Called Media::local_path() but media are stored on S3");
|
|
|
|
#[cfg(not(feature="s3"))]
|
|
|
|
unreachable!();
|
|
|
|
}
|
|
|
|
|
|
|
|
let relative_path = self
|
|
|
|
.file_path
|
|
|
|
.trim_start_matches(&CONFIG.media_directory)
|
|
|
|
.trim_start_matches(path::MAIN_SEPARATOR)
|
|
|
|
.trim_start_matches("static/media/");
|
|
|
|
|
|
|
|
Some(Path::new(&CONFIG.media_directory).join(relative_path))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the relative URL to access this file, which is also the key at which
|
|
|
|
/// it is stored in the S3 bucket if we are using S3 storage.
|
|
|
|
/// Does not start with a '/', it is of the form "static/media/<...>"
|
|
|
|
pub fn relative_url(&self) -> Option<String> {
|
|
|
|
if self.file_path.is_empty() {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
let relative_path = self
|
|
|
|
.file_path
|
|
|
|
.trim_start_matches(&CONFIG.media_directory)
|
|
|
|
.replace(path::MAIN_SEPARATOR, "/");
|
|
|
|
|
|
|
|
let relative_path = relative_path
|
|
|
|
.trim_start_matches('/')
|
|
|
|
.trim_start_matches("static/media/");
|
|
|
|
|
|
|
|
Some(format!("static/media/{}", relative_path))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns a public URL through which this media file can be accessed
|
2019-05-10 22:59:34 +02:00
|
|
|
pub fn url(&self) -> Result<String> {
|
2018-09-03 13:17:59 +02:00
|
|
|
if self.is_remote {
|
2018-12-29 09:36:07 +01:00
|
|
|
Ok(self.remote_url.clone().unwrap_or_default())
|
2018-09-03 13:17:59 +02:00
|
|
|
} else {
|
2023-05-12 15:40:36 +02:00
|
|
|
let relative_url = self.relative_url().unwrap_or_default();
|
|
|
|
|
|
|
|
#[cfg(feature="s3")]
|
|
|
|
if CONFIG.s3.as_ref().map(|x| x.direct_download).unwrap_or(false) {
|
|
|
|
let s3_url = match CONFIG.s3.as_ref().unwrap() {
|
|
|
|
S3Config { alias: Some(alias), .. } => {
|
|
|
|
format!("https://{}/{}", alias, relative_url)
|
|
|
|
}
|
|
|
|
S3Config { path_style: true, hostname, bucket, .. } => {
|
|
|
|
format!("https://{}/{}/{}",
|
|
|
|
hostname,
|
|
|
|
bucket,
|
|
|
|
relative_url
|
|
|
|
)
|
|
|
|
}
|
|
|
|
S3Config { path_style: false, hostname, bucket, .. } => {
|
|
|
|
format!("https://{}.{}/{}",
|
|
|
|
bucket,
|
|
|
|
hostname,
|
|
|
|
relative_url
|
|
|
|
)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
return Ok(s3_url);
|
|
|
|
}
|
|
|
|
|
2018-12-29 09:36:07 +01:00
|
|
|
Ok(ap_url(&format!(
|
2021-02-25 03:01:47 +01:00
|
|
|
"{}/{}",
|
2019-05-10 22:59:34 +02:00
|
|
|
Instance::get_local()?.public_domain,
|
2023-05-12 15:40:36 +02:00
|
|
|
relative_url
|
2018-12-29 09:36:07 +01:00
|
|
|
)))
|
2018-09-03 13:17:59 +02:00
|
|
|
}
|
2018-09-02 22:55:42 +02:00
|
|
|
}
|
2018-09-02 23:10:15 +02:00
|
|
|
|
2018-12-29 09:36:07 +01:00
|
|
|
pub fn delete(&self, conn: &Connection) -> Result<()> {
|
2018-11-24 12:44:17 +01:00
|
|
|
if !self.is_remote {
|
2023-05-12 12:28:00 +02:00
|
|
|
if CONFIG.s3.is_some() {
|
2023-05-12 13:19:41 +02:00
|
|
|
#[cfg(not(feature="s3"))]
|
|
|
|
unreachable!();
|
|
|
|
|
2023-05-12 12:28:00 +02:00
|
|
|
#[cfg(feature = "s3")]
|
|
|
|
CONFIG.s3.as_ref().unwrap().get_bucket()
|
2023-05-12 15:40:36 +02:00
|
|
|
.delete_object_blocking(&self.relative_url().ok_or(Error::NotFound)?)?;
|
2022-11-13 11:18:13 +01:00
|
|
|
} else {
|
2023-05-12 15:40:36 +02:00
|
|
|
fs::remove_file(self.local_path().ok_or(Error::NotFound)?)?;
|
2022-11-13 11:18:13 +01:00
|
|
|
}
|
2018-11-24 12:44:17 +01:00
|
|
|
}
|
|
|
|
diesel::delete(self)
|
|
|
|
.execute(conn)
|
2018-12-29 09:36:07 +01:00
|
|
|
.map(|_| ())
|
|
|
|
.map_err(Error::from)
|
2018-09-02 23:10:15 +02:00
|
|
|
}
|
2018-09-03 13:17:59 +02:00
|
|
|
|
2018-12-29 09:36:07 +01:00
|
|
|
pub fn save_remote(conn: &Connection, url: String, user: &User) -> Result<Media> {
|
2018-12-02 19:07:36 +01:00
|
|
|
if url.contains(&['<', '>', '"'][..]) {
|
2018-12-29 09:36:07 +01:00
|
|
|
Err(Error::Url)
|
2018-12-02 19:07:36 +01:00
|
|
|
} else {
|
2018-12-29 09:36:07 +01:00
|
|
|
Media::insert(
|
2018-12-02 19:07:36 +01:00
|
|
|
conn,
|
|
|
|
NewMedia {
|
|
|
|
file_path: String::new(),
|
|
|
|
alt_text: String::new(),
|
|
|
|
is_remote: true,
|
|
|
|
remote_url: Some(url),
|
|
|
|
sensitive: false,
|
|
|
|
content_warning: None,
|
|
|
|
owner_id: user.id,
|
|
|
|
},
|
2018-12-29 09:36:07 +01:00
|
|
|
)
|
2018-12-02 19:07:36 +01:00
|
|
|
}
|
2018-09-03 13:17:59 +02:00
|
|
|
}
|
|
|
|
|
2018-12-29 09:36:07 +01:00
|
|
|
pub fn set_owner(&self, conn: &Connection, user: &User) -> Result<()> {
|
2018-09-03 13:17:59 +02:00
|
|
|
diesel::update(self)
|
2018-11-24 12:44:17 +01:00
|
|
|
.set(medias::owner_id.eq(user.id))
|
2018-09-03 13:17:59 +02:00
|
|
|
.execute(conn)
|
2018-12-29 09:36:07 +01:00
|
|
|
.map(|_| ())
|
|
|
|
.map_err(Error::from)
|
2018-09-03 13:17:59 +02:00
|
|
|
}
|
2018-10-31 10:40:20 +01:00
|
|
|
|
|
|
|
// TODO: merge with save_remote?
|
2022-12-16 22:51:14 +01:00
|
|
|
pub fn from_activity(conn: &Connection, image: &Image) -> Result<Media> {
|
2022-02-26 02:58:49 +01:00
|
|
|
let remote_url = image
|
|
|
|
.url()
|
|
|
|
.and_then(|url| url.to_as_uri())
|
|
|
|
.ok_or(Error::MissingApProperty)?;
|
|
|
|
|
2023-05-12 15:54:44 +02:00
|
|
|
let file_path = if CONFIG.s3.is_some() {
|
|
|
|
#[cfg(not(feature="s3"))]
|
|
|
|
unreachable!();
|
|
|
|
|
|
|
|
#[cfg(feature = "s3")]
|
|
|
|
{
|
2023-05-12 16:11:29 +02:00
|
|
|
use rocket::http::ContentType;
|
|
|
|
|
2023-05-12 15:54:44 +02:00
|
|
|
let dest = determine_mirror_s3_path(&remote_url);
|
|
|
|
|
|
|
|
let media = request::get(
|
|
|
|
remote_url.as_str(),
|
|
|
|
User::get_sender(),
|
|
|
|
CONFIG.proxy().cloned(),
|
|
|
|
)?;
|
2023-05-12 16:11:29 +02:00
|
|
|
|
|
|
|
let content_type = media
|
|
|
|
.headers()
|
|
|
|
.get(reqwest::header::CONTENT_TYPE)
|
|
|
|
.and_then(|x| x.to_str().ok())
|
|
|
|
.and_then(ContentType::parse_flexible)
|
|
|
|
.unwrap_or(ContentType::Binary);
|
|
|
|
|
2023-05-12 15:54:44 +02:00
|
|
|
let bytes = media.bytes()?;
|
|
|
|
|
|
|
|
let bucket = CONFIG.s3.as_ref().unwrap().get_bucket();
|
2023-05-12 16:11:29 +02:00
|
|
|
bucket.put_object_with_content_type_blocking(
|
|
|
|
&dest,
|
|
|
|
&bytes,
|
|
|
|
&content_type.to_string()
|
|
|
|
)?;
|
2023-05-12 15:54:44 +02:00
|
|
|
|
|
|
|
dest
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
let path = determine_mirror_file_path(&remote_url);
|
|
|
|
let parent = path.parent().ok_or(Error::InvalidValue)?;
|
|
|
|
if !parent.is_dir() {
|
|
|
|
DirBuilder::new().recursive(true).create(parent)?;
|
|
|
|
}
|
2022-02-26 02:58:49 +01:00
|
|
|
|
2023-05-12 15:54:44 +02:00
|
|
|
let mut dest = fs::File::create(path.clone())?;
|
|
|
|
// TODO: conditional GET
|
|
|
|
request::get(
|
|
|
|
remote_url.as_str(),
|
|
|
|
User::get_sender(),
|
|
|
|
CONFIG.proxy().cloned(),
|
|
|
|
)?
|
|
|
|
.copy_to(&mut dest)?;
|
|
|
|
path.to_str().ok_or(Error::InvalidValue)?.to_string()
|
|
|
|
};
|
|
|
|
|
|
|
|
Media::find_by_file_path(conn, &file_path)
|
2022-02-26 02:58:49 +01:00
|
|
|
.and_then(|mut media| {
|
|
|
|
let mut updated = false;
|
|
|
|
|
|
|
|
let alt_text = image
|
|
|
|
.content()
|
|
|
|
.and_then(|content| content.to_as_string())
|
|
|
|
.ok_or(Error::NotFound)?;
|
|
|
|
let summary = image.summary().and_then(|summary| summary.to_as_string());
|
|
|
|
let sensitive = summary.is_some();
|
|
|
|
let content_warning = summary;
|
|
|
|
if media.alt_text != alt_text {
|
|
|
|
media.alt_text = alt_text;
|
|
|
|
updated = true;
|
|
|
|
}
|
|
|
|
if media.is_remote {
|
|
|
|
media.is_remote = false;
|
|
|
|
updated = true;
|
|
|
|
}
|
2022-05-03 15:25:43 +02:00
|
|
|
if media.remote_url.is_some() {
|
|
|
|
media.remote_url = None;
|
|
|
|
updated = true;
|
|
|
|
}
|
2022-02-26 02:58:49 +01:00
|
|
|
if media.sensitive != sensitive {
|
|
|
|
media.sensitive = sensitive;
|
|
|
|
updated = true;
|
|
|
|
}
|
|
|
|
if media.content_warning != content_warning {
|
|
|
|
media.content_warning = content_warning;
|
|
|
|
updated = true;
|
|
|
|
}
|
|
|
|
if updated {
|
2022-12-16 22:51:14 +01:00
|
|
|
diesel::update(&media).set(&media).execute(conn)?;
|
2022-02-26 02:58:49 +01:00
|
|
|
}
|
|
|
|
Ok(media)
|
|
|
|
})
|
|
|
|
.or_else(|_| {
|
|
|
|
let summary = image.summary().and_then(|summary| summary.to_as_string());
|
|
|
|
Media::insert(
|
|
|
|
conn,
|
|
|
|
NewMedia {
|
2023-05-12 15:54:44 +02:00
|
|
|
file_path,
|
2022-02-26 02:58:49 +01:00
|
|
|
alt_text: image
|
|
|
|
.content()
|
|
|
|
.and_then(|content| content.to_as_string())
|
|
|
|
.ok_or(Error::NotFound)?,
|
|
|
|
is_remote: false,
|
|
|
|
remote_url: None,
|
|
|
|
sensitive: summary.is_some(),
|
|
|
|
content_warning: summary,
|
2022-05-02 12:24:36 +02:00
|
|
|
owner_id: User::from_id(
|
2022-02-26 02:58:49 +01:00
|
|
|
conn,
|
|
|
|
&image
|
|
|
|
.attributed_to()
|
|
|
|
.and_then(|attributed_to| attributed_to.to_as_uri())
|
|
|
|
.ok_or(Error::MissingApProperty)?,
|
|
|
|
None,
|
|
|
|
CONFIG.proxy(),
|
|
|
|
)
|
|
|
|
.map_err(|(_, e)| e)?
|
|
|
|
.id,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-04-06 19:20:33 +02:00
|
|
|
pub fn get_media_processor<'a>(conn: &'a Connection, user: Vec<&User>) -> MediaProcessor<'a> {
|
|
|
|
let uid = user.iter().map(|u| u.id).collect::<Vec<_>>();
|
|
|
|
Box::new(move |id| {
|
|
|
|
let media = Media::get(conn, id).ok()?;
|
|
|
|
// if owner is user or check is disabled
|
|
|
|
if uid.contains(&media.owner_id) || uid.is_empty() {
|
2019-05-10 22:59:34 +02:00
|
|
|
Some((media.url().ok()?, media.content_warning))
|
2019-04-06 19:20:33 +02:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2018-11-24 12:44:17 +01:00
|
|
|
}
|
|
|
|
|
2021-02-17 17:23:57 +01:00
|
|
|
fn determine_mirror_file_path(url: &str) -> PathBuf {
|
2023-05-12 15:40:36 +02:00
|
|
|
let mut file_path = Path::new(&CONFIG.media_directory).join(REMOTE_MEDIA_DIRECTORY);
|
2023-05-12 15:54:44 +02:00
|
|
|
|
2023-05-12 15:40:36 +02:00
|
|
|
match Url::parse(url) {
|
|
|
|
Ok(url) if url.has_host() => {
|
2021-02-17 17:23:57 +01:00
|
|
|
file_path.push(url.host_str().unwrap());
|
|
|
|
for segment in url.path_segments().expect("FIXME") {
|
|
|
|
file_path.push(segment);
|
|
|
|
}
|
|
|
|
// TODO: handle query
|
|
|
|
// HINT: Use characters which must be percent-encoded in path as separator between path and query
|
|
|
|
// HINT: handle extension
|
2023-05-12 15:40:36 +02:00
|
|
|
}
|
|
|
|
other => {
|
|
|
|
if let Err(err) = other {
|
|
|
|
warn!("Failed to parse url: {} {}", &url, err);
|
|
|
|
} else {
|
|
|
|
warn!("Error without a host: {}", &url);
|
|
|
|
}
|
2021-02-17 17:23:57 +01:00
|
|
|
let ext = url
|
|
|
|
.rsplit('.')
|
|
|
|
.next()
|
|
|
|
.map(ToOwned::to_owned)
|
|
|
|
.unwrap_or_else(|| String::from("png"));
|
2021-11-27 23:53:13 +01:00
|
|
|
file_path.push(format!("{}.{}", GUID::rand(), ext));
|
2023-05-12 15:40:36 +02:00
|
|
|
}
|
|
|
|
}
|
2021-02-17 17:23:57 +01:00
|
|
|
file_path
|
|
|
|
}
|
|
|
|
|
2023-05-12 15:54:44 +02:00
|
|
|
#[cfg(feature="s3")]
|
|
|
|
fn determine_mirror_s3_path(url: &str) -> String {
|
|
|
|
match Url::parse(url) {
|
|
|
|
Ok(url) if url.has_host() => {
|
|
|
|
format!("static/media/{}/{}/{}",
|
|
|
|
REMOTE_MEDIA_DIRECTORY,
|
|
|
|
url.host_str().unwrap(),
|
|
|
|
url.path().trim_start_matches('/'),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
other => {
|
|
|
|
if let Err(err) = other {
|
|
|
|
warn!("Failed to parse url: {} {}", &url, err);
|
|
|
|
} else {
|
|
|
|
warn!("Error without a host: {}", &url);
|
|
|
|
}
|
|
|
|
let ext = url
|
|
|
|
.rsplit('.')
|
|
|
|
.next()
|
|
|
|
.map(ToOwned::to_owned)
|
|
|
|
.unwrap_or_else(|| String::from("png"));
|
|
|
|
format!("static/media/{}/{}.{}",
|
|
|
|
REMOTE_MEDIA_DIRECTORY,
|
|
|
|
GUID::rand(),
|
|
|
|
ext,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-24 12:44:17 +01:00
|
|
|
#[cfg(test)]
|
|
|
|
pub(crate) mod tests {
|
|
|
|
use super::*;
|
2020-01-21 07:02:03 +01:00
|
|
|
use crate::{tests::db, users::tests as usersTests, Connection as Conn};
|
2018-11-24 12:44:17 +01:00
|
|
|
use diesel::Connection;
|
|
|
|
use std::env::{current_dir, set_current_dir};
|
|
|
|
use std::fs;
|
|
|
|
use std::path::Path;
|
|
|
|
|
2018-12-09 18:44:26 +01:00
|
|
|
pub(crate) fn fill_database(conn: &Conn) -> (Vec<User>, Vec<Media>) {
|
2023-01-02 18:45:13 +01:00
|
|
|
let mut wd = current_dir().unwrap();
|
2018-11-24 12:44:17 +01:00
|
|
|
while wd.pop() {
|
|
|
|
if wd.join(".git").exists() {
|
|
|
|
set_current_dir(wd).unwrap();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let users = usersTests::fill_database(conn);
|
|
|
|
let user_one = users[0].id;
|
|
|
|
let user_two = users[1].id;
|
|
|
|
let f1 = "static/media/1.png".to_owned();
|
|
|
|
let f2 = "static/media/2.mp3".to_owned();
|
|
|
|
fs::write(f1.clone(), []).unwrap();
|
|
|
|
fs::write(f2.clone(), []).unwrap();
|
2019-03-20 17:56:17 +01:00
|
|
|
(
|
|
|
|
users,
|
|
|
|
vec![
|
|
|
|
NewMedia {
|
|
|
|
file_path: f1,
|
|
|
|
alt_text: "some alt".to_owned(),
|
|
|
|
is_remote: false,
|
|
|
|
remote_url: None,
|
|
|
|
sensitive: false,
|
|
|
|
content_warning: None,
|
|
|
|
owner_id: user_one,
|
|
|
|
},
|
|
|
|
NewMedia {
|
|
|
|
file_path: f2,
|
|
|
|
alt_text: "alt message".to_owned(),
|
|
|
|
is_remote: false,
|
|
|
|
remote_url: None,
|
|
|
|
sensitive: true,
|
|
|
|
content_warning: Some("Content warning".to_owned()),
|
|
|
|
owner_id: user_one,
|
|
|
|
},
|
|
|
|
NewMedia {
|
|
|
|
file_path: "".to_owned(),
|
|
|
|
alt_text: "another alt".to_owned(),
|
|
|
|
is_remote: true,
|
|
|
|
remote_url: Some("https://example.com/".to_owned()),
|
|
|
|
sensitive: false,
|
|
|
|
content_warning: None,
|
|
|
|
owner_id: user_two,
|
|
|
|
},
|
|
|
|
]
|
|
|
|
.into_iter()
|
2018-12-29 09:36:07 +01:00
|
|
|
.map(|nm| Media::insert(conn, nm).unwrap())
|
2019-03-20 17:56:17 +01:00
|
|
|
.collect(),
|
|
|
|
)
|
2018-11-24 12:44:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn clean(conn: &Conn) {
|
|
|
|
//used to remove files generated by tests
|
2018-12-29 09:36:07 +01:00
|
|
|
for media in Media::list_all_medias(conn).unwrap() {
|
2022-01-10 14:04:57 +01:00
|
|
|
if let Some(err) = media.delete(conn).err() {
|
|
|
|
match &err {
|
|
|
|
Error::Io(e) => match e.kind() {
|
|
|
|
std::io::ErrorKind::NotFound => (),
|
|
|
|
_ => panic!("{:?}", err),
|
|
|
|
},
|
|
|
|
_ => panic!("{:?}", err),
|
|
|
|
}
|
|
|
|
}
|
2018-11-24 12:44:17 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn delete() {
|
|
|
|
let conn = &db();
|
|
|
|
conn.test_transaction::<_, (), _>(|| {
|
2018-12-09 18:44:26 +01:00
|
|
|
let user = fill_database(conn).0[0].id;
|
2018-11-24 12:44:17 +01:00
|
|
|
|
|
|
|
let path = "static/media/test_deletion".to_owned();
|
|
|
|
fs::write(path.clone(), []).unwrap();
|
|
|
|
|
|
|
|
let media = Media::insert(
|
|
|
|
conn,
|
|
|
|
NewMedia {
|
|
|
|
file_path: path.clone(),
|
|
|
|
alt_text: "alt message".to_owned(),
|
|
|
|
is_remote: false,
|
|
|
|
remote_url: None,
|
|
|
|
sensitive: false,
|
|
|
|
content_warning: None,
|
|
|
|
owner_id: user,
|
|
|
|
},
|
2019-03-20 17:56:17 +01:00
|
|
|
)
|
|
|
|
.unwrap();
|
2018-11-24 12:44:17 +01:00
|
|
|
|
|
|
|
assert!(Path::new(&path).exists());
|
2018-12-29 09:36:07 +01:00
|
|
|
media.delete(conn).unwrap();
|
2018-11-24 12:44:17 +01:00
|
|
|
assert!(!Path::new(&path).exists());
|
|
|
|
|
|
|
|
clean(conn);
|
|
|
|
Ok(())
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn set_owner() {
|
|
|
|
let conn = &db();
|
|
|
|
conn.test_transaction::<_, (), _>(|| {
|
2018-12-09 18:44:26 +01:00
|
|
|
let (users, _) = fill_database(conn);
|
2018-11-24 12:44:17 +01:00
|
|
|
let u1 = &users[0];
|
|
|
|
let u2 = &users[1];
|
|
|
|
|
|
|
|
let path = "static/media/test_set_owner".to_owned();
|
|
|
|
fs::write(path.clone(), []).unwrap();
|
|
|
|
|
|
|
|
let media = Media::insert(
|
|
|
|
conn,
|
|
|
|
NewMedia {
|
2023-01-02 18:45:13 +01:00
|
|
|
file_path: path,
|
2018-11-24 12:44:17 +01:00
|
|
|
alt_text: "alt message".to_owned(),
|
|
|
|
is_remote: false,
|
|
|
|
remote_url: None,
|
|
|
|
sensitive: false,
|
|
|
|
content_warning: None,
|
|
|
|
owner_id: u1.id,
|
|
|
|
},
|
2019-03-20 17:56:17 +01:00
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
assert!(Media::for_user(conn, u1.id)
|
|
|
|
.unwrap()
|
|
|
|
.iter()
|
|
|
|
.any(|m| m.id == media.id));
|
|
|
|
assert!(!Media::for_user(conn, u2.id)
|
|
|
|
.unwrap()
|
|
|
|
.iter()
|
|
|
|
.any(|m| m.id == media.id));
|
2018-12-29 09:36:07 +01:00
|
|
|
media.set_owner(conn, u2).unwrap();
|
2019-03-20 17:56:17 +01:00
|
|
|
assert!(!Media::for_user(conn, u1.id)
|
|
|
|
.unwrap()
|
|
|
|
.iter()
|
|
|
|
.any(|m| m.id == media.id));
|
|
|
|
assert!(Media::for_user(conn, u2.id)
|
|
|
|
.unwrap()
|
|
|
|
.iter()
|
|
|
|
.any(|m| m.id == media.id));
|
2018-11-24 12:44:17 +01:00
|
|
|
|
|
|
|
clean(conn);
|
|
|
|
Ok(())
|
|
|
|
});
|
2018-10-31 10:40:20 +01:00
|
|
|
}
|
2018-09-02 13:34:48 +02:00
|
|
|
}
|