mirror of
https://github.com/master-of-zen/Av1an.git
synced 2024-11-25 02:29:40 +00:00
ce03d7a6e2
Supports: * Pyscenedetect chunking * aom_keyframes chunking * Automatic detection of input as Vapoursynth script. * Cleaned up fast frame counting into unified method "frame_probe_fast" Areas for improvement: * Windows support * vspipe arg passing Fixes #173
69 lines
2.5 KiB
Python
Executable file
69 lines
2.5 KiB
Python
Executable file
#!/bin/env python
|
|
|
|
from os import mkfifo
|
|
from subprocess import Popen
|
|
|
|
from scenedetect.detectors import ContentDetector
|
|
from scenedetect.scene_manager import SceneManager
|
|
from scenedetect.video_manager import VideoManager
|
|
from scenedetect.frame_timecode import FrameTimecode
|
|
|
|
from Av1an.logger import log
|
|
from Av1an.utils import frame_probe
|
|
from Av1an.vapoursynth import compose_vapoursynth_pipe
|
|
|
|
|
|
def pyscene(video, threshold, min_scene_len, is_vs, temp):
|
|
"""
|
|
Running PySceneDetect detection on source video for segmenting.
|
|
Optimal threshold settings 15-50
|
|
"""
|
|
if not min_scene_len:
|
|
min_scene_len = 15
|
|
|
|
log(f'Starting PySceneDetect:\nThreshold: {threshold}, Min scene lenght: {min_scene_len}, Is Vapoursynth input: {is_vs}\n')
|
|
|
|
if is_vs:
|
|
# Handling vapoursynth, so we need to create a named pipe to feed to VideoManager.
|
|
# TODO: Do we clean this up after pyscenedetect has run, or leave it as part of the temp dir, where it will be cleaned up later?
|
|
vspipe_fifo = temp / 'vspipe.y4m'
|
|
mkfifo(vspipe_fifo)
|
|
vspipe_cmd = compose_vapoursynth_pipe(video, vspipe_fifo)
|
|
vspipe_process = Popen(vspipe_cmd)
|
|
|
|
# Get number of frames from Vapoursynth script to pass as duration to VideoManager.
|
|
# We need to pass the number of frames to the manager, otherwise it won't close the
|
|
# receiving end of the pipe, and will simply sit waiting after vspipe has finished sending
|
|
# the last frame.
|
|
frames = frame_probe(video)
|
|
|
|
video_manager = VideoManager([str(vspipe_fifo if is_vs else video)])
|
|
scene_manager = SceneManager()
|
|
scene_manager.add_detector(ContentDetector(threshold=threshold, min_scene_len=min_scene_len))
|
|
base_timecode = video_manager.get_base_timecode()
|
|
|
|
video_manager.set_duration(duration=FrameTimecode(frames, video_manager.get_framerate()) if is_vs else None)
|
|
|
|
# Set downscale factor to improve processing speed.
|
|
video_manager.set_downscale_factor()
|
|
|
|
# Start video_manager.
|
|
video_manager.start()
|
|
|
|
scene_manager.detect_scenes(frame_source=video_manager, show_progress=True)
|
|
|
|
# If fed using a vspipe process, ensure that vspipe has finished.
|
|
if is_vs:
|
|
vspipe_process.wait()
|
|
|
|
# Obtain list of detected scenes.
|
|
scene_list = scene_manager.get_scene_list(base_timecode)
|
|
|
|
scenes = [int(scene[0].get_frames()) for scene in scene_list]
|
|
|
|
# Remove 0 from list
|
|
if scenes[0] == 0:
|
|
scenes.remove(0)
|
|
log(f'Found scenes: {len(scenes)}\n')
|
|
|
|
return scenes
|