use crate::{ config::ImageFormat, either::Either, error::{Error, UploadError}, ffmpeg::InputFormat, magick::ValidInputType, }; use actix_web::web::Bytes; use tokio::io::AsyncRead; use tracing::instrument; struct UnvalidatedBytes { bytes: Bytes, written: usize, } impl UnvalidatedBytes { fn new(bytes: Bytes) -> Self { UnvalidatedBytes { bytes, written: 0 } } } impl AsyncRead for UnvalidatedBytes { fn poll_read( mut self: std::pin::Pin<&mut Self>, _cx: &mut std::task::Context<'_>, buf: &mut tokio::io::ReadBuf<'_>, ) -> std::task::Poll> { let bytes_to_write = (self.bytes.len() - self.written).min(buf.remaining()); if bytes_to_write > 0 { let end = self.written + bytes_to_write; buf.put_slice(&self.bytes[self.written..end]); self.written = end; } std::task::Poll::Ready(Ok(())) } } #[instrument(name = "Validate image", skip(bytes))] pub(crate) async fn validate_image_bytes( bytes: Bytes, prescribed_format: Option, enable_silent_video: bool, validate: bool, ) -> Result<(ValidInputType, impl AsyncRead + Unpin), Error> { let input_type = crate::magick::input_type_bytes(bytes.clone()).await?; if !validate { return Ok((input_type, Either::left(UnvalidatedBytes::new(bytes)))); } match (prescribed_format, input_type) { (_, ValidInputType::Gif) => { if !enable_silent_video { return Err(UploadError::SilentVideoDisabled.into()); } Ok(( ValidInputType::Mp4, Either::right(Either::left( crate::ffmpeg::to_mp4_bytes(bytes, InputFormat::Gif).await?, )), )) } (_, ValidInputType::Mp4) => { if !enable_silent_video { return Err(UploadError::SilentVideoDisabled.into()); } Ok(( ValidInputType::Mp4, Either::right(Either::left( crate::ffmpeg::to_mp4_bytes(bytes, InputFormat::Mp4).await?, )), )) } (Some(ImageFormat::Jpeg) | None, ValidInputType::Jpeg) => Ok(( ValidInputType::Jpeg, Either::right(Either::right(Either::left( crate::exiftool::clear_metadata_bytes_read(bytes)?, ))), )), (Some(ImageFormat::Png) | None, ValidInputType::Png) => Ok(( ValidInputType::Png, Either::right(Either::right(Either::left( crate::exiftool::clear_metadata_bytes_read(bytes)?, ))), )), (Some(ImageFormat::Webp) | None, ValidInputType::Webp) => Ok(( ValidInputType::Webp, Either::right(Either::right(Either::right(Either::left( crate::magick::clear_metadata_bytes_read(bytes)?, )))), )), (Some(format), _) => Ok(( ValidInputType::from_format(format), Either::right(Either::right(Either::right(Either::right( crate::magick::convert_bytes_read(bytes, format)?, )))), )), } }