#[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, #[error("Transcode semaphore is closed")] Closed, } pub(crate) enum ThumbnailFormat { Jpeg, Webp, } 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", } } } static MAX_TRANSCODES: once_cell::sync::OnceCell = once_cell::sync::OnceCell::new(); fn semaphore() -> &'static tokio::sync::Semaphore { MAX_TRANSCODES .get_or_init(|| tokio::sync::Semaphore::new(num_cpus::get().saturating_sub(1).max(1))) } pub(crate) async fn to_mp4(from: P1, to: P2) -> Result<(), VideoError> where P1: AsRef, P2: AsRef, { let permit = semaphore().acquire().await?; let mut child = tokio::process::Command::new("ffmpeg") .arg(&"-i") .arg(&from.as_ref()) .args([ &"-movflags", &"faststart", &"-pix_fmt", &"yuv420p", &"-vf", &"scale=trunc(iw/2)*2:trunc(ih/2)*2", &"-an", &"-codec", &"h264", &"-f", &"mp4", ]) .arg(&to.as_ref()) .spawn()?; let status = child.wait().await?; drop(permit); if !status.success() { return Err(VideoError::Status); } Ok(()) } pub(crate) async fn thumbnail( from: P1, to: P2, format: ThumbnailFormat, ) -> Result<(), VideoError> where P1: AsRef, P2: AsRef, { let permit = semaphore().acquire().await?; let mut child = tokio::process::Command::new("ffmpeg") .arg(&"-i") .arg(&from.as_ref()) .args([ &"-ss", &"00:00:01.000", &"-vframes", &"1", &"-codec", &format.as_codec(), &"-f", &format.as_format(), ]) .arg(&to.as_ref()) .spawn()?; let status = child.wait().await?; drop(permit); if !status.success() { return Err(VideoError::Status); } Ok(()) } impl From for VideoError { fn from(_: tokio::sync::AcquireError) -> VideoError { VideoError::Closed } }