mirror of
https://git.asonix.dog/asonix/pict-rs.git
synced 2025-01-07 10:05:28 +00:00
Start introducing more video controls
This commit is contained in:
parent
c9a74a73ca
commit
c57a48db8a
6 changed files with 45 additions and 2 deletions
|
@ -51,7 +51,9 @@ impl Args {
|
||||||
media_max_height,
|
media_max_height,
|
||||||
media_max_area,
|
media_max_area,
|
||||||
media_max_file_size,
|
media_max_file_size,
|
||||||
|
media_max_frame_count,
|
||||||
media_enable_silent_video,
|
media_enable_silent_video,
|
||||||
|
media_enable_full_video,
|
||||||
media_filters,
|
media_filters,
|
||||||
media_format,
|
media_format,
|
||||||
media_cache_duration,
|
media_cache_duration,
|
||||||
|
@ -69,7 +71,9 @@ impl Args {
|
||||||
max_height: media_max_height,
|
max_height: media_max_height,
|
||||||
max_area: media_max_area,
|
max_area: media_max_area,
|
||||||
max_file_size: media_max_file_size,
|
max_file_size: media_max_file_size,
|
||||||
|
max_frame_count: media_max_frame_count,
|
||||||
enable_silent_video: media_enable_silent_video,
|
enable_silent_video: media_enable_silent_video,
|
||||||
|
enable_full_video: media_enable_full_video,
|
||||||
filters: media_filters,
|
filters: media_filters,
|
||||||
format: media_format,
|
format: media_format,
|
||||||
cache_duration: media_cache_duration,
|
cache_duration: media_cache_duration,
|
||||||
|
@ -312,8 +316,12 @@ struct Media {
|
||||||
#[serde(skip_serializing_if = "Option::is_none")]
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
max_file_size: Option<usize>,
|
max_file_size: Option<usize>,
|
||||||
#[serde(skip_serializing_if = "Option::is_none")]
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
max_frame_count: Option<usize>,
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
enable_silent_video: Option<bool>,
|
enable_silent_video: Option<bool>,
|
||||||
#[serde(skip_serializing_if = "Option::is_none")]
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
enable_full_video: Option<bool>,
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
filters: Option<Vec<String>>,
|
filters: Option<Vec<String>>,
|
||||||
#[serde(skip_serializing_if = "Option::is_none")]
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
format: Option<ImageFormat>,
|
format: Option<ImageFormat>,
|
||||||
|
@ -412,9 +420,15 @@ struct Run {
|
||||||
/// The maximum size, in megabytes, for uploaded media
|
/// The maximum size, in megabytes, for uploaded media
|
||||||
#[clap(long)]
|
#[clap(long)]
|
||||||
media_max_file_size: Option<usize>,
|
media_max_file_size: Option<usize>,
|
||||||
/// Whether to enable GIF and silent MP4 uploads. Full videos are unsupported
|
/// The maximum number of frames allowed for uploaded GIF and MP4s.
|
||||||
|
#[clap(long)]
|
||||||
|
media_max_frame_count: Option<usize>,
|
||||||
|
/// Whether to enable GIF and silent MP4 uploads
|
||||||
#[clap(long)]
|
#[clap(long)]
|
||||||
media_enable_silent_video: Option<bool>,
|
media_enable_silent_video: Option<bool>,
|
||||||
|
/// Whether to enable full MP4 uploads
|
||||||
|
#[clap(long)]
|
||||||
|
media_enable_full_video: Option<bool>,
|
||||||
/// Which media filters should be enabled on the `process` endpoint
|
/// Which media filters should be enabled on the `process` endpoint
|
||||||
#[clap(long)]
|
#[clap(long)]
|
||||||
media_filters: Option<Vec<String>>,
|
media_filters: Option<Vec<String>>,
|
||||||
|
|
|
@ -65,7 +65,9 @@ struct MediaDefaults {
|
||||||
max_height: usize,
|
max_height: usize,
|
||||||
max_area: usize,
|
max_area: usize,
|
||||||
max_file_size: usize,
|
max_file_size: usize,
|
||||||
|
max_frame_count: usize,
|
||||||
enable_silent_video: bool,
|
enable_silent_video: bool,
|
||||||
|
enable_full_video: bool,
|
||||||
filters: Vec<String>,
|
filters: Vec<String>,
|
||||||
skip_validate_imports: bool,
|
skip_validate_imports: bool,
|
||||||
cache_duration: i64,
|
cache_duration: i64,
|
||||||
|
@ -150,7 +152,9 @@ impl Default for MediaDefaults {
|
||||||
max_height: 10_000,
|
max_height: 10_000,
|
||||||
max_area: 40_000_000,
|
max_area: 40_000_000,
|
||||||
max_file_size: 40,
|
max_file_size: 40,
|
||||||
|
max_frame_count: 3_600,
|
||||||
enable_silent_video: true,
|
enable_silent_video: true,
|
||||||
|
enable_full_video: false,
|
||||||
filters: vec![
|
filters: vec![
|
||||||
"blur".into(),
|
"blur".into(),
|
||||||
"crop".into(),
|
"crop".into(),
|
||||||
|
|
|
@ -98,8 +98,12 @@ pub(crate) struct Media {
|
||||||
|
|
||||||
pub(crate) max_file_size: usize,
|
pub(crate) max_file_size: usize,
|
||||||
|
|
||||||
|
pub(crate) max_frame_count: usize,
|
||||||
|
|
||||||
pub(crate) enable_silent_video: bool,
|
pub(crate) enable_silent_video: bool,
|
||||||
|
|
||||||
|
pub(crate) enable_full_video: bool,
|
||||||
|
|
||||||
pub(crate) filters: BTreeSet<String>,
|
pub(crate) filters: BTreeSet<String>,
|
||||||
|
|
||||||
#[serde(skip_serializing_if = "Option::is_none")]
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
|
|
@ -12,6 +12,7 @@ enum MaybeHumanDate {
|
||||||
pub(crate) struct Details {
|
pub(crate) struct Details {
|
||||||
width: usize,
|
width: usize,
|
||||||
height: usize,
|
height: usize,
|
||||||
|
frames: Option<usize>,
|
||||||
content_type: Serde<mime::Mime>,
|
content_type: Serde<mime::Mime>,
|
||||||
created_at: MaybeHumanDate,
|
created_at: MaybeHumanDate,
|
||||||
}
|
}
|
||||||
|
@ -33,6 +34,7 @@ impl Details {
|
||||||
details.width,
|
details.width,
|
||||||
details.height,
|
details.height,
|
||||||
details.mime_type,
|
details.mime_type,
|
||||||
|
details.frames,
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -48,13 +50,15 @@ impl Details {
|
||||||
details.width,
|
details.width,
|
||||||
details.height,
|
details.height,
|
||||||
details.mime_type,
|
details.mime_type,
|
||||||
|
details.frames,
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn now(width: usize, height: usize, content_type: mime::Mime) -> Self {
|
pub(crate) fn now(width: usize, height: usize, content_type: mime::Mime, frames: Option<usize>) -> Self {
|
||||||
Details {
|
Details {
|
||||||
width,
|
width,
|
||||||
height,
|
height,
|
||||||
|
frames,
|
||||||
content_type: Serde::new(content_type),
|
content_type: Serde::new(content_type),
|
||||||
created_at: MaybeHumanDate::HumanDate(time::OffsetDateTime::now_utc()),
|
created_at: MaybeHumanDate::HumanDate(time::OffsetDateTime::now_utc()),
|
||||||
}
|
}
|
||||||
|
|
|
@ -99,6 +99,9 @@ pub(crate) enum UploadError {
|
||||||
#[error("Invalid media dimensions")]
|
#[error("Invalid media dimensions")]
|
||||||
Dimensions,
|
Dimensions,
|
||||||
|
|
||||||
|
#[error("Too many frames")]
|
||||||
|
Frames,
|
||||||
|
|
||||||
#[error("Unable to download image, bad response {0}")]
|
#[error("Unable to download image, bad response {0}")]
|
||||||
Download(actix_web::http::StatusCode),
|
Download(actix_web::http::StatusCode),
|
||||||
|
|
||||||
|
|
|
@ -86,6 +86,7 @@ pub(crate) struct Details {
|
||||||
pub(crate) mime_type: mime::Mime,
|
pub(crate) mime_type: mime::Mime,
|
||||||
pub(crate) width: usize,
|
pub(crate) width: usize,
|
||||||
pub(crate) height: usize,
|
pub(crate) height: usize,
|
||||||
|
pub(crate) frames: Option<usize>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tracing::instrument(name = "Clear Metadata", skip(input))]
|
#[tracing::instrument(name = "Clear Metadata", skip(input))]
|
||||||
|
@ -210,6 +211,8 @@ pub(crate) async fn details_file(path_str: &str) -> Result<Details, Error> {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_details(s: std::borrow::Cow<'_, str>) -> Result<Details, Error> {
|
fn parse_details(s: std::borrow::Cow<'_, str>) -> Result<Details, Error> {
|
||||||
|
let frames = s.lines().count();
|
||||||
|
|
||||||
let mut lines = s.lines();
|
let mut lines = s.lines();
|
||||||
let first = lines.next().ok_or(UploadError::UnsupportedFormat)?;
|
let first = lines.next().ok_or(UploadError::UnsupportedFormat)?;
|
||||||
|
|
||||||
|
@ -257,6 +260,11 @@ fn parse_details(s: std::borrow::Cow<'_, str>) -> Result<Details, Error> {
|
||||||
mime_type,
|
mime_type,
|
||||||
width,
|
width,
|
||||||
height,
|
height,
|
||||||
|
frames: if frames > 1 {
|
||||||
|
Some(frames)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
},
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -307,6 +315,12 @@ impl Details {
|
||||||
return Err(UploadError::Dimensions.into());
|
return Err(UploadError::Dimensions.into());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let Some(frames) = self.frames {
|
||||||
|
if frames > crate::CONFIG.media.max_frame_count {
|
||||||
|
return Err(UploadError::Frames.into());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let input_type = match (self.mime_type.type_(), self.mime_type.subtype()) {
|
let input_type = match (self.mime_type.type_(), self.mime_type.subtype()) {
|
||||||
(mime::VIDEO, mime::MP4 | mime::MPEG) => ValidInputType::Mp4,
|
(mime::VIDEO, mime::MP4 | mime::MPEG) => ValidInputType::Mp4,
|
||||||
(mime::IMAGE, mime::GIF) => ValidInputType::Gif,
|
(mime::IMAGE, mime::GIF) => ValidInputType::Gif,
|
||||||
|
|
Loading…
Reference in a new issue