Detect input transparency for gif/webm

This commit is contained in:
asonix 2023-02-25 12:59:47 -06:00
parent 134c8d2423
commit 9a2f846cd1

View file

@ -6,6 +6,8 @@ use crate::{
store::Store, store::Store,
}; };
use actix_web::web::Bytes; use actix_web::web::Bytes;
use once_cell::sync::OnceCell;
use std::collections::HashSet;
use tokio::io::{AsyncRead, AsyncReadExt}; use tokio::io::{AsyncRead, AsyncReadExt};
#[derive(Debug)] #[derive(Debug)]
@ -87,7 +89,23 @@ impl TranscodeOptions {
} }
} }
fn execute(&self, input_path: &str, output_path: &str) -> Result<Process, std::io::Error> { const fn supports_alpha(&self) -> bool {
match self.output {
TranscodeOutputOptions::Gif
| TranscodeOutputOptions::Video {
video_codec: VideoCodec::Vp8 | VideoCodec::Vp9,
..
} => true,
_ => false,
}
}
fn execute(
&self,
input_path: &str,
output_path: &str,
alpha: bool,
) -> Result<Process, std::io::Error> {
if let Some(audio_codec) = self.output_ffmpeg_audio_codec() { if let Some(audio_codec) = self.output_ffmpeg_audio_codec() {
Process::run( Process::run(
"ffmpeg", "ffmpeg",
@ -95,7 +113,7 @@ impl TranscodeOptions {
"-i", "-i",
input_path, input_path,
"-pix_fmt", "-pix_fmt",
"yuv420p", if alpha { "yuva420p" } else { "yuv420p" },
"-vf", "-vf",
"scale=trunc(iw/2)*2:trunc(ih/2)*2", "scale=trunc(iw/2)*2:trunc(ih/2)*2",
"-c:a", "-c:a",
@ -114,7 +132,7 @@ impl TranscodeOptions {
"-i", "-i",
input_path, input_path,
"-pix_fmt", "-pix_fmt",
"yuv420p", if alpha { "yuva420p" } else { "yuv420p" },
"-vf", "-vf",
"scale=trunc(iw/2)*2:trunc(ih/2)*2", "scale=trunc(iw/2)*2:trunc(ih/2)*2",
"-an", "-an",
@ -327,6 +345,45 @@ pub(crate) async fn details_bytes(input: Bytes) -> Result<Option<Details>, Error
.await .await
} }
async fn alpha_pixel_formats() -> Result<HashSet<String>, Error> {
let process = Process::run(
"ffprobe",
&[
"-v",
"0",
"-show_entries",
"pixel_format=name:flags=alpha",
"-of",
"compact=p=0",
],
)?;
let mut output = Vec::new();
process.read().read_to_end(&mut output).await?;
let output = String::from_utf8_lossy(&output);
let formats = output
.split('\n')
.filter_map(|format| {
if format.is_empty() {
return None;
}
if !format.ends_with("1") {
return None;
}
Some(
format
.trim_start_matches("name=")
.trim_end_matches("|flags:alpha=1")
.to_string(),
)
})
.collect();
Ok(formats)
}
#[tracing::instrument(skip(f))] #[tracing::instrument(skip(f))]
async fn details_file<F, Fut>(f: F) -> Result<Option<Details>, Error> async fn details_file<F, Fut>(f: F) -> Result<Option<Details>, Error>
where where
@ -417,6 +474,27 @@ fn parse_details_inner(
}) })
} }
async fn pixel_format(input_file: &str) -> Result<String, Error> {
let process = Process::run(
"ffprobe",
&[
"-v",
"0",
"-select_streams",
"v:0",
"-show_entries",
"stream=pix_fmt",
"-of",
"compact=p=0:nk=1",
input_file,
],
)?;
let mut output = Vec::new();
process.read().read_to_end(&mut output).await?;
Ok(String::from_utf8_lossy(&output).trim().to_string())
}
#[tracing::instrument(skip(input))] #[tracing::instrument(skip(input))]
pub(crate) async fn transcode_bytes( pub(crate) async fn transcode_bytes(
input: Bytes, input: Bytes,
@ -434,7 +512,25 @@ pub(crate) async fn transcode_bytes(
tmp_one.write_from_bytes(input).await?; tmp_one.write_from_bytes(input).await?;
tmp_one.close().await?; tmp_one.close().await?;
let process = transcode_options.execute(input_file_str, output_file_str)?; let alpha = if transcode_options.supports_alpha() {
static ALPHA_PIXEL_FORMATS: OnceCell<HashSet<String>> = OnceCell::new();
let format = pixel_format(input_file_str).await?;
match ALPHA_PIXEL_FORMATS.get() {
Some(alpha_pixel_formats) => alpha_pixel_formats.contains(&format),
None => {
let pixel_formats = alpha_pixel_formats().await?;
let alpha = pixel_formats.contains(&format);
let _ = ALPHA_PIXEL_FORMATS.set(pixel_formats);
alpha
}
}
} else {
false
};
let process = transcode_options.execute(input_file_str, output_file_str, alpha)?;
process.wait().await?; process.wait().await?;
tokio::fs::remove_file(input_file).await?; tokio::fs::remove_file(input_file).await?;