149 lines
4.3 KiB
Python
149 lines
4.3 KiB
Python
"""视频切片工具:将长视频按固定时长切分为多个片段。"""
|
||
|
||
from __future__ import annotations
|
||
|
||
import subprocess
|
||
import tempfile
|
||
from pathlib import Path
|
||
from typing import List, Tuple
|
||
|
||
from loguru import logger
|
||
|
||
|
||
def _get_ffmpeg_path() -> str:
|
||
"""获取可用的 ffmpeg 路径。"""
|
||
project_ffmpeg = Path("/home/ubuntu/projects/FishServer/tools/ffmpeg/bin/ffmpeg")
|
||
if project_ffmpeg.is_file():
|
||
return str(project_ffmpeg)
|
||
system_paths = ["/usr/bin/ffmpeg", "/usr/local/bin/ffmpeg"]
|
||
for path in system_paths:
|
||
if Path(path).is_file():
|
||
return path
|
||
return "ffmpeg"
|
||
|
||
|
||
def _get_ffprobe_path() -> str:
|
||
"""获取可用的 ffprobe 路径。"""
|
||
ffmpeg_path = Path(_get_ffmpeg_path())
|
||
ffprobe = ffmpeg_path.parent / "ffprobe"
|
||
if ffprobe.is_file():
|
||
return str(ffprobe)
|
||
return "ffprobe"
|
||
|
||
|
||
def get_video_duration(video_path: Path) -> float:
|
||
"""获取视频时长(秒)。"""
|
||
ffprobe_path = _get_ffprobe_path()
|
||
try:
|
||
result = subprocess.run(
|
||
[
|
||
ffprobe_path,
|
||
"-v", "error",
|
||
"-show_entries", "format=duration",
|
||
"-of", "default=noprint_wrappers=1:nokey=1",
|
||
str(video_path),
|
||
],
|
||
capture_output=True,
|
||
text=True,
|
||
timeout=30,
|
||
)
|
||
if result.returncode == 0:
|
||
duration = float(result.stdout.strip())
|
||
return duration
|
||
except Exception as e:
|
||
logger.warning("[video_slice] failed to get duration: {}", e)
|
||
return 0.0
|
||
|
||
|
||
def slice_video(
|
||
video_path: Path,
|
||
slice_duration: float = 10.0,
|
||
output_dir: Path | None = None,
|
||
) -> Tuple[List[Path], Path]:
|
||
"""将视频按固定时长切分为多个片段。
|
||
|
||
Args:
|
||
video_path: 源视频路径
|
||
slice_duration: 每个切片的时长(秒),默认10秒
|
||
output_dir: 输出目录,默认使用临时目录
|
||
|
||
Returns:
|
||
(切片文件列表, 输出目录路径)
|
||
"""
|
||
duration = get_video_duration(video_path)
|
||
if duration <= slice_duration:
|
||
# 视频太短,无需切片
|
||
return [video_path], video_path.parent
|
||
|
||
if output_dir is None:
|
||
output_dir = Path(tempfile.mkdtemp(prefix="video_slices_"))
|
||
output_dir.mkdir(parents=True, exist_ok=True)
|
||
|
||
ffmpeg_path = _get_ffmpeg_path()
|
||
base_name = video_path.stem
|
||
|
||
# 计算需要切多少片
|
||
num_slices = int(duration / slice_duration)
|
||
if duration % slice_duration > 1.0: # 剩余超过1秒才多切一片
|
||
num_slices += 1
|
||
|
||
logger.info(
|
||
"[video_slice] slicing {} ({}s) into {} slices of {}s each",
|
||
video_path.name,
|
||
duration,
|
||
num_slices,
|
||
slice_duration,
|
||
)
|
||
|
||
slice_files: List[Path] = []
|
||
for i in range(num_slices):
|
||
start_time = i * slice_duration
|
||
slice_file = output_dir / f"{base_name}_slice_{i:03d}.mp4"
|
||
|
||
# 使用 ffmpeg 切片,-c copy 快速复制,避免重新编码
|
||
cmd = [
|
||
ffmpeg_path,
|
||
"-y",
|
||
"-ss", str(start_time),
|
||
"-t", str(slice_duration),
|
||
"-i", str(video_path),
|
||
"-c", "copy",
|
||
"-avoid_negative_ts", "make_zero",
|
||
str(slice_file),
|
||
]
|
||
|
||
try:
|
||
result = subprocess.run(
|
||
cmd,
|
||
capture_output=True,
|
||
text=True,
|
||
timeout=60,
|
||
)
|
||
if result.returncode == 0 and slice_file.is_file():
|
||
slice_files.append(slice_file)
|
||
logger.debug(
|
||
"[video_slice] created slice {}: {} (start={}s)",
|
||
i,
|
||
slice_file.name,
|
||
start_time,
|
||
)
|
||
else:
|
||
logger.warning(
|
||
"[video_slice] failed to create slice {}: {}",
|
||
i,
|
||
result.stderr[-200:] if result.stderr else "unknown",
|
||
)
|
||
except Exception as e:
|
||
logger.warning("[video_slice] exception creating slice {}: {}", i, e)
|
||
|
||
if not slice_files:
|
||
# 切片失败,返回原视频
|
||
return [video_path], video_path.parent
|
||
|
||
logger.info(
|
||
"[video_slice] created {} slices for {}",
|
||
len(slice_files),
|
||
video_path.name,
|
||
)
|
||
return slice_files, output_dir
|