2
0
Fork 0
mirror of https://git.asonix.dog/asonix/pict-rs synced 2025-01-03 16:31:25 +00:00
pict-rs/src/validate.rs

294 lines
7.5 KiB
Rust
Raw Normal View History

mod exiftool;
mod ffmpeg;
mod magick;
2021-10-21 01:13:39 +00:00
use crate::{
2023-07-13 22:42:21 +00:00
discover::Discovery,
error::Error,
2023-09-02 01:50:10 +00:00
error_code::ErrorCode,
2023-07-13 22:42:21 +00:00
formats::{
AnimationFormat, AnimationOutput, ImageInput, ImageOutput, InputFile, InputVideoFormat,
InternalFormat, Validations,
2023-07-13 22:42:21 +00:00
},
magick::PolicyDir,
process::ProcessRead,
tmp_file::TmpDir,
2021-10-21 01:13:39 +00:00
};
use actix_web::web::Bytes;
2023-07-13 22:42:21 +00:00
#[derive(Debug, thiserror::Error)]
pub(crate) enum ValidationError {
#[error("Too wide")]
Width,
#[error("Too tall")]
Height,
#[error("Too many pixels")]
Area,
#[error("Too many frames")]
Frames,
#[error("Uploaded file is empty")]
Empty,
2023-07-13 22:42:21 +00:00
#[error("Filesize too large")]
Filesize,
#[error("Video is disabled")]
VideoDisabled,
}
2023-09-02 01:50:10 +00:00
impl ValidationError {
pub(crate) const fn error_code(&self) -> ErrorCode {
match self {
Self::Width => ErrorCode::VALIDATE_WIDTH,
Self::Height => ErrorCode::VALIDATE_HEIGHT,
Self::Area => ErrorCode::VALIDATE_AREA,
Self::Frames => ErrorCode::VALIDATE_FRAMES,
Self::Empty => ErrorCode::VALIDATE_FILE_EMPTY,
Self::Filesize => ErrorCode::VALIDATE_FILE_SIZE,
Self::VideoDisabled => ErrorCode::VIDEO_DISABLED,
}
}
}
2023-07-13 22:42:21 +00:00
const MEGABYTES: usize = 1024 * 1024;
#[tracing::instrument(skip_all)]
pub(crate) async fn validate_bytes(
tmp_dir: &TmpDir,
policy_dir: &PolicyDir,
bytes: Bytes,
2023-07-13 22:42:21 +00:00
validations: Validations<'_>,
2023-08-05 17:41:06 +00:00
timeout: u64,
) -> Result<(InternalFormat, ProcessRead), Error> {
if bytes.is_empty() {
return Err(ValidationError::Empty.into());
}
2023-07-13 22:42:21 +00:00
let Discovery {
input,
width,
height,
frames,
} = crate::discover::discover_bytes(tmp_dir, policy_dir, timeout, bytes.clone()).await?;
2023-07-13 22:42:21 +00:00
match &input {
InputFile::Image(input) => {
let (format, process_read) = process_image(
tmp_dir,
policy_dir,
bytes,
*input,
width,
height,
validations.image,
timeout,
)
.await?;
Ok((format, process_read))
}
InputFile::Animation(input) => {
let (format, process_read) = process_animation(
tmp_dir,
policy_dir,
2023-07-13 22:42:21 +00:00
bytes,
*input,
width,
height,
frames.unwrap_or(1),
validations.animation,
2023-08-05 17:41:06 +00:00
timeout,
2023-07-13 22:42:21 +00:00
)
.await?;
Ok((format, process_read))
2023-07-13 22:42:21 +00:00
}
InputFile::Video(input) => {
let (format, process_read) = process_video(
tmp_dir,
2023-07-13 22:42:21 +00:00
bytes,
2023-08-31 02:00:15 +00:00
*input,
2023-07-13 22:42:21 +00:00
width,
height,
frames.unwrap_or(1),
validations.video,
2023-08-05 17:41:06 +00:00
timeout,
2023-07-13 22:42:21 +00:00
)
.await?;
Ok((format, process_read))
2023-07-13 22:42:21 +00:00
}
}
}
#[allow(clippy::too_many_arguments)]
#[tracing::instrument(skip(tmp_dir, policy_dir, bytes, validations))]
2023-07-13 22:42:21 +00:00
async fn process_image(
tmp_dir: &TmpDir,
policy_dir: &PolicyDir,
2023-07-13 22:42:21 +00:00
bytes: Bytes,
input: ImageInput,
width: u16,
height: u16,
validations: &crate::config::Image,
2023-08-05 17:41:06 +00:00
timeout: u64,
) -> Result<(InternalFormat, ProcessRead), Error> {
2023-07-13 22:42:21 +00:00
if width > validations.max_width {
return Err(ValidationError::Width.into());
}
if height > validations.max_height {
return Err(ValidationError::Height.into());
}
if u32::from(width) * u32::from(height) > validations.max_area {
return Err(ValidationError::Area.into());
}
if bytes.len() > validations.max_file_size * MEGABYTES {
return Err(ValidationError::Filesize.into());
}
let ImageOutput {
format,
needs_transcode,
} = input.build_output(validations.format);
let process_read = if needs_transcode {
let quality = validations.quality_for(format);
magick::convert_image(
tmp_dir,
policy_dir,
input.format,
format,
quality,
timeout,
bytes,
)
.await?
2023-07-13 22:42:21 +00:00
} else {
exiftool::clear_metadata_bytes_read(bytes, timeout)?
2023-07-13 22:42:21 +00:00
};
Ok((InternalFormat::Image(format), process_read))
2023-07-13 22:42:21 +00:00
}
fn validate_animation(
size: usize,
width: u16,
height: u16,
frames: u32,
validations: &crate::config::Animation,
) -> Result<(), ValidationError> {
if width > validations.max_width {
return Err(ValidationError::Width);
}
if height > validations.max_height {
return Err(ValidationError::Height);
}
if u32::from(width) * u32::from(height) > validations.max_area {
return Err(ValidationError::Area);
}
if frames > validations.max_frame_count {
return Err(ValidationError::Frames);
}
if size > validations.max_file_size * MEGABYTES {
return Err(ValidationError::Filesize);
}
Ok(())
}
2023-10-07 16:32:36 +00:00
#[allow(clippy::too_many_arguments)]
#[tracing::instrument(skip(tmp_dir, policy_dir, bytes, validations))]
2023-07-13 22:42:21 +00:00
async fn process_animation(
tmp_dir: &TmpDir,
policy_dir: &PolicyDir,
2023-07-13 22:42:21 +00:00
bytes: Bytes,
input: AnimationFormat,
width: u16,
height: u16,
frames: u32,
validations: &crate::config::Animation,
2023-08-05 17:41:06 +00:00
timeout: u64,
) -> Result<(InternalFormat, ProcessRead), Error> {
validate_animation(bytes.len(), width, height, frames, validations)?;
let AnimationOutput {
format,
needs_transcode,
} = input.build_output(validations.format);
let process_read = if needs_transcode {
let quality = validations.quality_for(format);
magick::convert_animation(tmp_dir, policy_dir, input, format, quality, timeout, bytes)
.await?
} else {
exiftool::clear_metadata_bytes_read(bytes, timeout)?
};
Ok((InternalFormat::Animation(format), process_read))
2023-07-13 22:42:21 +00:00
}
fn validate_video(
size: usize,
width: u16,
height: u16,
frames: u32,
validations: &crate::config::Video,
) -> Result<(), ValidationError> {
if !validations.enable {
return Err(ValidationError::VideoDisabled);
}
if width > validations.max_width {
return Err(ValidationError::Width);
}
if height > validations.max_height {
return Err(ValidationError::Height);
}
2023-07-13 22:42:21 +00:00
if u32::from(width) * u32::from(height) > validations.max_area {
return Err(ValidationError::Area);
}
if frames > validations.max_frame_count {
return Err(ValidationError::Frames);
}
if size > validations.max_file_size * MEGABYTES {
return Err(ValidationError::Filesize);
}
Ok(())
}
2023-10-07 16:32:36 +00:00
#[allow(clippy::too_many_arguments)]
#[tracing::instrument(skip(tmp_dir, bytes, validations))]
2023-07-13 22:42:21 +00:00
async fn process_video(
tmp_dir: &TmpDir,
2023-07-13 22:42:21 +00:00
bytes: Bytes,
input: InputVideoFormat,
2023-07-13 22:42:21 +00:00
width: u16,
height: u16,
frames: u32,
validations: &crate::config::Video,
2023-08-05 17:41:06 +00:00
timeout: u64,
) -> Result<(InternalFormat, ProcessRead), Error> {
2023-07-13 22:42:21 +00:00
validate_video(bytes.len(), width, height, frames, validations)?;
let output = input.build_output(
validations.video_codec,
validations.audio_codec,
validations.allow_audio,
);
let crf = validations.crf_for(width, height);
let process_read = ffmpeg::transcode_bytes(tmp_dir, input, output, crf, timeout, bytes).await?;
2023-07-13 22:42:21 +00:00
Ok((
InternalFormat::Video(output.format.internal_format()),
process_read,
))
}