2018-12-11 20:52:48 +00:00
|
|
|
#!/usr/bin/env python3
|
2019-07-27 17:39:40 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
2018-12-11 20:52:48 +00:00
|
|
|
"""
|
2019-07-27 17:39:40 +00:00
|
|
|
Name: Video2X FFmpeg Controller
|
2018-12-11 20:52:48 +00:00
|
|
|
Author: K4YT3X
|
|
|
|
Date Created: Feb 24, 2018
|
2019-07-27 17:39:40 +00:00
|
|
|
Last Modified: July 27, 2019
|
2018-12-11 20:52:48 +00:00
|
|
|
|
2019-07-26 18:22:49 +00:00
|
|
|
Description: This class handles all FFmpeg related operations.
|
2018-12-11 20:52:48 +00:00
|
|
|
"""
|
2019-07-27 17:39:40 +00:00
|
|
|
|
|
|
|
# built-in imports
|
2019-02-21 17:26:05 +00:00
|
|
|
import json
|
2019-06-14 23:19:12 +00:00
|
|
|
import os
|
2019-07-27 21:29:33 +00:00
|
|
|
import pathlib
|
|
|
|
import subprocess
|
2018-12-11 20:52:48 +00:00
|
|
|
|
2019-07-27 17:39:40 +00:00
|
|
|
# third-party imports
|
|
|
|
from avalon_framework import Avalon
|
|
|
|
|
2018-12-11 20:52:48 +00:00
|
|
|
|
|
|
|
class Ffmpeg:
|
2019-07-26 18:22:49 +00:00
|
|
|
"""This class communicates with FFmpeg
|
2018-12-11 20:52:48 +00:00
|
|
|
|
2019-07-27 17:39:40 +00:00
|
|
|
This class deals with FFmpeg. It handles extracting
|
2018-12-11 20:52:48 +00:00
|
|
|
frames, stripping audio, converting images into videos
|
|
|
|
and inserting audio tracks to videos.
|
|
|
|
"""
|
|
|
|
|
2019-03-30 18:02:05 +00:00
|
|
|
def __init__(self, ffmpeg_settings, image_format):
|
2019-03-09 17:50:54 +00:00
|
|
|
self.ffmpeg_settings = ffmpeg_settings
|
2019-03-30 18:02:05 +00:00
|
|
|
|
2019-07-27 21:29:33 +00:00
|
|
|
self.ffmpeg_path = pathlib.Path(self.ffmpeg_settings['ffmpeg_path'])
|
|
|
|
self.ffmpeg_binary = self.ffmpeg_path / 'ffmpeg.exe'
|
|
|
|
self.ffmpeg_probe_binary = self.ffmpeg_path / 'ffprobe.exe'
|
2019-03-30 18:02:05 +00:00
|
|
|
self.image_format = image_format
|
2019-07-10 00:47:55 +00:00
|
|
|
self.pixel_format = None
|
2019-07-27 21:29:33 +00:00
|
|
|
|
2019-07-10 00:47:55 +00:00
|
|
|
def get_pixel_formats(self):
|
|
|
|
""" Get a dictionary of supported pixel formats
|
|
|
|
|
2019-07-27 17:39:40 +00:00
|
|
|
List all supported pixel formats and their
|
|
|
|
corresponding bit depth.
|
2019-07-10 00:47:55 +00:00
|
|
|
|
|
|
|
Returns:
|
|
|
|
dictionary -- JSON dict of all pixel formats to bit depth
|
|
|
|
"""
|
|
|
|
execute = [
|
|
|
|
self.ffmpeg_probe_binary,
|
|
|
|
'-v',
|
|
|
|
'quiet',
|
|
|
|
'-pix_fmts'
|
|
|
|
]
|
|
|
|
|
2019-07-27 21:29:33 +00:00
|
|
|
# turn elements into str
|
|
|
|
execute = [str(e) for e in execute]
|
|
|
|
|
2019-07-10 00:47:55 +00:00
|
|
|
Avalon.debug_info(f'Executing: {" ".join(execute)}')
|
|
|
|
|
|
|
|
# initialize dictionary to store pixel formats
|
|
|
|
pixel_formats = {}
|
|
|
|
|
|
|
|
# record all pixel formats into dictionary
|
|
|
|
for line in subprocess.run(execute, check=True, stdout=subprocess.PIPE).stdout.decode().split('\n'):
|
|
|
|
try:
|
|
|
|
pixel_formats[' '.join(line.split()).split()[1]] = int(' '.join(line.split()).split()[3])
|
|
|
|
except (IndexError, ValueError):
|
|
|
|
pass
|
|
|
|
|
|
|
|
# print pixel formats for debugging
|
|
|
|
Avalon.debug_info(pixel_formats)
|
|
|
|
|
|
|
|
return pixel_formats
|
2018-12-11 20:52:48 +00:00
|
|
|
|
2019-02-21 17:26:05 +00:00
|
|
|
def get_video_info(self, input_video):
|
|
|
|
""" Gets input video information
|
2018-12-11 20:52:48 +00:00
|
|
|
|
2019-02-21 17:26:05 +00:00
|
|
|
This method reads input video information
|
2019-07-27 17:39:40 +00:00
|
|
|
using ffprobe in dictionary
|
2018-12-11 20:52:48 +00:00
|
|
|
|
|
|
|
Arguments:
|
2019-02-21 17:26:05 +00:00
|
|
|
input_video {string} -- input video file path
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
dictionary -- JSON text of input video information
|
2018-12-11 20:52:48 +00:00
|
|
|
"""
|
2019-03-30 18:02:05 +00:00
|
|
|
|
|
|
|
# this execution command needs to be hard-coded
|
|
|
|
# since video2x only strictly recignizes this one format
|
2019-03-19 17:07:20 +00:00
|
|
|
execute = [
|
2019-03-30 18:02:05 +00:00
|
|
|
self.ffmpeg_probe_binary,
|
2019-03-19 17:07:20 +00:00
|
|
|
'-v',
|
|
|
|
'quiet',
|
|
|
|
'-print_format',
|
|
|
|
'json',
|
|
|
|
'-show_format',
|
|
|
|
'-show_streams',
|
|
|
|
'-i',
|
2019-04-22 05:26:36 +00:00
|
|
|
input_video
|
2019-03-19 17:07:20 +00:00
|
|
|
]
|
|
|
|
|
2019-07-27 21:29:33 +00:00
|
|
|
# turn elements into str
|
|
|
|
execute = [str(e) for e in execute]
|
|
|
|
|
2019-04-22 05:26:36 +00:00
|
|
|
Avalon.debug_info(f'Executing: {" ".join(execute)}')
|
2019-03-19 17:07:20 +00:00
|
|
|
json_str = subprocess.run(execute, check=True, stdout=subprocess.PIPE).stdout
|
2019-02-21 17:26:05 +00:00
|
|
|
return json.loads(json_str.decode('utf-8'))
|
2018-12-11 20:52:48 +00:00
|
|
|
|
2019-02-21 17:26:05 +00:00
|
|
|
def extract_frames(self, input_video, extracted_frames):
|
|
|
|
"""Extract every frame from original videos
|
2018-12-11 20:52:48 +00:00
|
|
|
|
2019-07-27 17:39:40 +00:00
|
|
|
This method extracts every frame from input video using FFmpeg
|
2018-12-11 20:52:48 +00:00
|
|
|
|
|
|
|
Arguments:
|
2019-02-21 17:26:05 +00:00
|
|
|
input_video {string} -- input video path
|
2019-04-29 04:06:54 +00:00
|
|
|
extracted_frames {string} -- video output directory
|
2018-12-11 20:52:48 +00:00
|
|
|
"""
|
2019-03-19 17:07:20 +00:00
|
|
|
execute = [
|
2019-05-06 20:25:10 +00:00
|
|
|
self.ffmpeg_binary
|
|
|
|
]
|
|
|
|
|
|
|
|
execute.extend([
|
2019-03-19 17:07:20 +00:00
|
|
|
'-i',
|
2019-06-10 01:31:13 +00:00
|
|
|
input_video
|
2019-05-06 20:25:10 +00:00
|
|
|
])
|
|
|
|
|
2019-06-05 16:18:51 +00:00
|
|
|
execute.extend(self._read_configuration(phase='video_to_frames', section='output_options'))
|
|
|
|
|
2019-06-10 01:31:13 +00:00
|
|
|
execute.extend([
|
2019-07-27 21:29:33 +00:00
|
|
|
extracted_frames / f'extracted_%0d.{self.image_format}'
|
2019-06-10 01:31:13 +00:00
|
|
|
])
|
|
|
|
|
2019-06-05 16:18:51 +00:00
|
|
|
execute.extend(self._read_configuration(phase='video_to_frames'))
|
|
|
|
|
2019-05-06 20:25:10 +00:00
|
|
|
self._execute(execute)
|
2018-12-11 20:52:48 +00:00
|
|
|
|
2019-02-21 17:26:05 +00:00
|
|
|
def convert_video(self, framerate, resolution, upscaled_frames):
|
2018-12-11 20:52:48 +00:00
|
|
|
"""Converts images into videos
|
|
|
|
|
2019-07-27 17:39:40 +00:00
|
|
|
This method converts a set of images into a video
|
2018-12-11 20:52:48 +00:00
|
|
|
|
|
|
|
Arguments:
|
|
|
|
framerate {float} -- target video framerate
|
|
|
|
resolution {string} -- target video resolution
|
2019-04-29 04:06:54 +00:00
|
|
|
upscaled_frames {string} -- source images directory
|
2018-12-11 20:52:48 +00:00
|
|
|
"""
|
2019-03-19 17:07:20 +00:00
|
|
|
execute = [
|
|
|
|
self.ffmpeg_binary,
|
|
|
|
'-r',
|
|
|
|
str(framerate),
|
|
|
|
'-s',
|
2019-05-06 20:25:10 +00:00
|
|
|
resolution
|
|
|
|
]
|
|
|
|
|
|
|
|
# read FFmpeg input options
|
|
|
|
execute.extend(self._read_configuration(phase='frames_to_video', section='input_options'))
|
|
|
|
|
2019-07-12 22:12:08 +00:00
|
|
|
# WORKAROUND FOR WAIFU2X-NCNN-VULKAN
|
2019-07-27 21:29:33 +00:00
|
|
|
# Dev: SAT3LL
|
|
|
|
# rename all .png.png suffixes to .png
|
2019-07-12 22:12:08 +00:00
|
|
|
import re
|
|
|
|
import shutil
|
|
|
|
regex = re.compile(r'\.png\.png$')
|
2019-07-27 21:29:33 +00:00
|
|
|
for frame_name in upscaled_frames.iterdir():
|
|
|
|
(upscaled_frames / frame_name).rename(upscaled_frames / regex.sub('.png', str(frame_name)))
|
2019-07-12 22:12:08 +00:00
|
|
|
# END WORKAROUND
|
|
|
|
|
2019-05-06 20:25:10 +00:00
|
|
|
# append input frames path into command
|
|
|
|
execute.extend([
|
2019-03-19 17:07:20 +00:00
|
|
|
'-i',
|
2019-07-27 21:29:33 +00:00
|
|
|
upscaled_frames / f'extracted_%d.{self.image_format}'
|
2019-05-06 20:25:10 +00:00
|
|
|
])
|
|
|
|
|
|
|
|
# read FFmpeg output options
|
|
|
|
execute.extend(self._read_configuration(phase='frames_to_video', section='output_options'))
|
|
|
|
|
|
|
|
# read other options
|
|
|
|
execute.extend(self._read_configuration(phase='frames_to_video'))
|
|
|
|
|
|
|
|
# specify output file location
|
|
|
|
execute.extend([
|
2019-07-27 21:29:33 +00:00
|
|
|
upscaled_frames / 'no_audio.mp4'
|
2019-05-06 20:25:10 +00:00
|
|
|
])
|
|
|
|
|
|
|
|
self._execute(execute)
|
2018-12-11 20:52:48 +00:00
|
|
|
|
2019-02-21 17:26:05 +00:00
|
|
|
def migrate_audio_tracks_subtitles(self, input_video, output_video, upscaled_frames):
|
|
|
|
""" Migrates audio tracks and subtitles from input video to output video
|
2018-12-11 20:52:48 +00:00
|
|
|
|
|
|
|
Arguments:
|
2019-02-21 17:26:05 +00:00
|
|
|
input_video {string} -- input video file path
|
|
|
|
output_video {string} -- output video file path
|
|
|
|
upscaled_frames {string} -- directory containing upscaled frames
|
2018-12-11 20:52:48 +00:00
|
|
|
"""
|
2019-03-19 17:07:20 +00:00
|
|
|
execute = [
|
|
|
|
self.ffmpeg_binary,
|
|
|
|
'-i',
|
2019-07-27 21:29:33 +00:00
|
|
|
upscaled_frames / 'no_audio.mp4',
|
2019-03-19 17:07:20 +00:00
|
|
|
'-i',
|
2019-05-06 20:25:10 +00:00
|
|
|
input_video
|
2019-03-19 17:07:20 +00:00
|
|
|
]
|
|
|
|
|
2019-06-14 05:15:13 +00:00
|
|
|
execute.extend(self._read_configuration(phase='migrating_tracks', section='output_options'))
|
2019-03-30 18:02:05 +00:00
|
|
|
|
2019-05-06 20:25:10 +00:00
|
|
|
execute.extend([
|
|
|
|
output_video
|
|
|
|
])
|
2019-03-30 18:02:05 +00:00
|
|
|
|
2019-05-06 20:25:10 +00:00
|
|
|
execute.extend(self._read_configuration(phase='migrating_tracks'))
|
|
|
|
|
|
|
|
self._execute(execute)
|
|
|
|
|
|
|
|
def _read_configuration(self, phase, section=None):
|
|
|
|
""" read configuration from JSON
|
|
|
|
|
|
|
|
Read the configurations (arguments) from the JSON
|
|
|
|
configuration file and append them to the end of the
|
|
|
|
FFmpeg command.
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
execute {list} -- list of arguments to be executed
|
|
|
|
phase {str} -- phase of operation
|
|
|
|
"""
|
|
|
|
|
|
|
|
configuration = []
|
|
|
|
|
|
|
|
# if section is specified, read configurations or keys
|
|
|
|
# from only that section
|
|
|
|
if section:
|
|
|
|
source = self.ffmpeg_settings[phase][section].keys()
|
2019-07-10 00:47:55 +00:00
|
|
|
|
|
|
|
# if pixel format is not specified, use the source pixel format
|
|
|
|
try:
|
|
|
|
if self.ffmpeg_settings[phase][section].get('-pix_fmt') is None:
|
|
|
|
self.ffmpeg_settings[phase][section]['-pix_fmt'] = self.pixel_format
|
|
|
|
except KeyError:
|
|
|
|
pass
|
2019-05-06 20:25:10 +00:00
|
|
|
else:
|
|
|
|
source = self.ffmpeg_settings[phase].keys()
|
|
|
|
|
|
|
|
for key in source:
|
|
|
|
|
|
|
|
if section:
|
|
|
|
value = self.ffmpeg_settings[phase][section][key]
|
|
|
|
else:
|
|
|
|
value = self.ffmpeg_settings[phase][key]
|
2019-03-30 18:02:05 +00:00
|
|
|
|
|
|
|
# null or None means that leave this option out (keep default)
|
2019-07-26 18:22:49 +00:00
|
|
|
if value is None or value is False or isinstance(value, dict):
|
2019-03-30 18:02:05 +00:00
|
|
|
continue
|
2019-07-26 18:22:49 +00:00
|
|
|
|
|
|
|
# if the value is a list, append the same argument and all values
|
|
|
|
elif isinstance(value, list):
|
|
|
|
|
|
|
|
for subvalue in value:
|
|
|
|
configuration.append(key)
|
|
|
|
if value is not True:
|
|
|
|
configuration.append(str(subvalue))
|
|
|
|
|
2019-07-27 21:29:33 +00:00
|
|
|
# otherwise the value is typical
|
2019-03-30 18:02:05 +00:00
|
|
|
else:
|
2019-05-06 20:25:10 +00:00
|
|
|
configuration.append(key)
|
2019-03-30 18:02:05 +00:00
|
|
|
|
|
|
|
# true means key is an option
|
|
|
|
if value is True:
|
|
|
|
continue
|
|
|
|
|
2019-05-06 20:25:10 +00:00
|
|
|
configuration.append(str(value))
|
|
|
|
|
|
|
|
return configuration
|
|
|
|
|
|
|
|
def _execute(self, execute):
|
|
|
|
""" execute command
|
2019-03-30 18:02:05 +00:00
|
|
|
|
2019-05-06 20:25:10 +00:00
|
|
|
Arguments:
|
|
|
|
execute {list} -- list of arguments to be executed
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
int -- execution return code
|
|
|
|
"""
|
2019-04-22 05:26:36 +00:00
|
|
|
Avalon.debug_info(f'Executing: {execute}')
|
2019-07-27 21:29:33 +00:00
|
|
|
|
|
|
|
# turn all list elements into string to avoid errors
|
|
|
|
execute = [str(e) for e in execute]
|
|
|
|
|
2019-03-30 18:02:05 +00:00
|
|
|
return subprocess.run(execute, shell=True, check=True).returncode
|