mirror of
https://git.asonix.dog/asonix/pict-rs.git
synced 2025-01-07 10:05:28 +00:00
Detect input transparency for gif/webm
This commit is contained in:
parent
134c8d2423
commit
9a2f846cd1
1 changed files with 100 additions and 4 deletions
104
src/ffmpeg.rs
104
src/ffmpeg.rs
|
@ -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?;
|
||||||
|
|
Loading…
Reference in a new issue