use std::env;
use std::path::PathBuf;
use std::process::Command;
pub fn ffmpeg_path() -> String {
if let Ok(p) = env::var("VISER_FFMPEG") {
if !p.is_empty() {
return p;
}
}
if let Some(p) = local_binary("ffmpeg") {
return p;
}
"ffmpeg".into()
}
pub fn ffprobe_path() -> String {
if let Ok(p) = env::var("VISER_FFPROBE") {
if !p.is_empty() {
return p;
}
}
if let Some(p) = local_binary("ffprobe") {
return p;
}
"ffprobe".into()
}
const MIN_FFMPEG_MAJOR: u32 = 6;
#[derive(Debug, Clone)]
pub struct FfmpegVersion {
pub binary: String,
pub major: u32,
pub minor: u32,
pub raw: String,
}
pub fn check_ffmpeg() -> anyhow::Result<FfmpegVersion> {
let path = ffmpeg_path();
let output = Command::new(&path)
.arg("-version")
.output()
.map_err(|e| anyhow::anyhow!("ffmpeg not found at '{path}': {e}"))?;
if !output.status.success() {
anyhow::bail!("ffmpeg at '{path}' exited with error");
}
let stdout = String::from_utf8_lossy(&output.stdout);
let first_line = stdout.lines().next().unwrap_or("").trim().to_string();
let version = parse_ffmpeg_version(&first_line, path)?;
if version.major < MIN_FFMPEG_MAJOR {
anyhow::bail!(
"ffmpeg {}.{} is too old — viser requires FFmpeg >= {MIN_FFMPEG_MAJOR}.0 (found {})",
version.major,
version.minor,
version.raw,
);
}
Ok(version)
}
pub fn check_ffprobe() -> anyhow::Result<FfmpegVersion> {
let path = ffprobe_path();
let output = Command::new(&path)
.arg("-version")
.output()
.map_err(|e| anyhow::anyhow!("ffprobe not found at '{path}': {e}"))?;
if !output.status.success() {
anyhow::bail!("ffprobe at '{path}' exited with error");
}
let stdout = String::from_utf8_lossy(&output.stdout);
let first_line = stdout.lines().next().unwrap_or("").trim().to_string();
parse_ffmpeg_version(&first_line, path)
}
fn parse_ffmpeg_version(line: &str, path: String) -> anyhow::Result<FfmpegVersion> {
let version_str = line
.strip_prefix("ffmpeg version ")
.or_else(|| line.strip_prefix("ffprobe version "))
.and_then(|s| s.split_whitespace().next())
.map(|s| s.trim_start_matches('n'))
.ok_or_else(|| anyhow::anyhow!("could not parse version from: {line}"))?;
let parts: Vec<&str> = version_str.split('.').collect();
let major: u32 = parts
.first()
.and_then(|s| s.parse().ok())
.ok_or_else(|| anyhow::anyhow!("could not parse major version from: {version_str}"))?;
let minor: u32 = parts.get(1).and_then(|s| s.parse().ok()).unwrap_or(0);
Ok(FfmpegVersion { binary: path, major, minor, raw: version_str.to_string() })
}
const KNOWN_VMAF_MODELS: &[&str] =
&["vmaf_v0.6.1", "vmaf_v0.6.1neg", "vmaf_4k_v0.6.1", "vmaf_b_v0.6.3", "vmaf_4k_v0.6.1neg"];
pub fn validate_vmaf_model(model: &str) -> anyhow::Result<()> {
if KNOWN_VMAF_MODELS.contains(&model) {
return Ok(());
}
anyhow::bail!("unknown VMAF model '{model}'. Known models: {}", KNOWN_VMAF_MODELS.join(", "));
}
fn local_binary(name: &str) -> Option<String> {
let mut path = PathBuf::from("bin").join("ffmpeg");
if cfg!(windows) {
path = path.join(format!("{name}.exe"));
} else {
path = path.join(name);
}
if path.exists() { Some(path.to_string_lossy().into_owned()) } else { None }
}