use crate::stream::Process; use actix_web::web::Bytes; use tokio::{io::AsyncRead, process::Command}; #[derive(Debug, thiserror::Error)] pub(crate) enum VideoError { #[error("Failed to interface with transcode process")] IO(#[from] std::io::Error), #[error("Failed to convert file")] Status, } pub(crate) enum InputFormat { Gif, Mp4, } pub(crate) enum ThumbnailFormat { Jpeg, // Webp, } impl InputFormat { fn as_format(&self) -> &'static str { match self { InputFormat::Gif => "gif_pipe", InputFormat::Mp4 => "mp4", } } } impl ThumbnailFormat { fn as_codec(&self) -> &'static str { match self { ThumbnailFormat::Jpeg => "mjpeg", // ThumbnailFormat::Webp => "webp", } } fn as_format(&self) -> &'static str { match self { ThumbnailFormat::Jpeg => "singlejpeg", // ThumbnailFormat::Webp => "webp", } } } pub(crate) fn to_mp4_bytes( input: Bytes, input_format: InputFormat, ) -> std::io::Result { let process = Process::spawn(Command::new("ffmpeg").args([ "-f", input_format.as_format(), "-i", "pipe:", "-movflags", "faststart+frag_keyframe+empty_moov", "-pix_fmt", "yuv420p", "-vf", "scale=trunc(iw/2)*2:trunc(ih/2)*2", "-an", "-codec", "h264", "-f", "mp4", "pipe:", ]))?; Ok(process.bytes_read(input).unwrap()) } pub(crate) async fn thumbnail( from: P1, to: P2, format: ThumbnailFormat, ) -> Result<(), VideoError> where P1: AsRef, P2: AsRef, { let mut child = Command::new("ffmpeg") .arg(&"-i") .arg(&from.as_ref()) .args([ "-vframes", "1", "-codec", format.as_codec(), "-f", format.as_format(), ]) .arg(&to.as_ref()) .spawn()?; let status = child.wait().await?; if !status.success() { return Err(VideoError::Status); } Ok(()) }