Dotfiles
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

201 lines
10 KiB

#!/usr/bin/env python3
"""
xconv ffmpeg wrapper based on AdvancedAV
-----------------------------------------------------------
AdvancedAV helps with constructing FFmpeg commandline arguments.
It can automatically parse input files with the help of FFmpeg's ffprobe tool (WiP)
and allows programatically mapping streams to output files and setting metadata on them.
-----------------------------------------------------------
Copyright (c) 2015-2017 Taeyeon Mori
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
-----------------------------------------------------------
Commandline Parsing
"""
from .profileman import load_all_profiles, load_profile
from . import version_info
from advancedav import version_info as aav_version_info
from argparse import ArgumentParser, Action
from pathlib import Path
from os.path import basename
from multiprocessing import cpu_count
version = "%s (AdvancedAV %s)" % (".".join(map(str, version_info)), ".".join(map(str, aav_version_info)))
# == Support code ==
class TerminalAction(Action):
def __init__(self, option_strings, dest, nargs=0, default=None, **kwargs):
super().__init__(option_strings, dest, nargs=nargs, default=default or {}, **kwargs)
def __call__(self, parser, namespace, values, option_string=None):
self.run(parser, *values)
parser.exit()
class ProfilesAction(TerminalAction):
def run(self, parser):
print("Available Profiles:")
for name, profile in sorted(load_all_profiles().items()):
print(" %-25s %s" % (name, profile.description if profile.description else ""))
class ProfileInfoAction(TerminalAction):
def __init__(self, option_strings, dest, nargs=1, default=None, **kwargs):
super().__init__(option_strings, dest, nargs=nargs, default=default or {}, **kwargs)
def run(self, parser, profile_name):
profile = load_profile(profile_name)
print("Profile '%s':" % profile_name)
if profile.description:
print(" Description: %s" % profile.description)
if "output" in profile.features:
output = profile.features["output"]
output_info = []
if output[0]:
output_info.append("Format: %s" % output[0])
if output[1]:
output_info.append("File extension: %s" % output[0])
if output_info:
print(" Output: %s" % "; ".join(output_info))
if profile.features:
print(" Flags: %s" % ", ".join("%s(%r)" % (k, v) if v is not None else k for k, v in profile.features.items()))
if profile.defines:
print(" Supported defines:")
for define in sorted(profile.defines.items()):
print(" %s: %s" % define)
class DefineAction(Action):
def __init__(self, option_strings, dest, nargs=1, default=None, **kwargs):
super().__init__(option_strings, dest, nargs=nargs, default=default or {}, **kwargs)
def __call__(self, parser, namespace, values, option_string=None):
value = values[0]
dest = getattr(namespace, self.dest)
if "=" in value:
k, v = value.split("=")
dest[k] = v
else:
dest[value] = True
class ExtendAction(Action):
def __init__(self, option_strings, dest, nargs="+", default=None, **kwargs):
super().__init__(option_strings, dest, nargs=nargs, default=default, **kwargs)
def __call__(self, parser, namespace, values, option_string=None):
items = getattr(namespace, self.dest) or []
items.extend(values)
setattr(namespace, self.dest, items)
def parse_args(argv):
prog = basename(argv[0])
if prog == "__main__.py":
prog = "python -m xconv"
parser = ArgumentParser(prog=prog,
usage="""%(prog)s [-h | -l | -i PROFILE]
%(prog)s [option]... -p PROFILE [-DNAME[=VALUE]]... [-B] [-T] input output
%(prog)s [option]... -p PROFILE [-DNAME[=VALUE]]... -M [-T] inputs... output
%(prog)s [option]... -p PROFILE [-DNAME[=VALUE]]... -C [-T] inputs... output
%(prog)s [option]... -p PROFILE [-DNAME[=VALUE]]... [-B] inputs... directory
%(prog)s [option]... -p PROFILE [-DNAME[=VALUE]]... [-B] -t directory inputs...""",
description="""FFmpeg wrapper based on AdvancedAV""")
parser.add_argument("-V", "--version", help="Show version and quit", action="version",
version="""XConv %s""" % version)
# Available Options
parser.add_argument("-v", "--verbose", help="Enable verbose output", action="store_true")
parser.add_argument("-q", "--quiet", help="Be less verbose", action="store_true")
6 years ago
parser.add_argument("-j", "--concurrent", help="Run ffmpeg concurrently using at most N instances [%(default)s]", metavar="N", type=int, default=cpu_count())
profile = parser.add_argument_group("Profile")
profile.add_argument("-l", "--list-profiles", help="List profiles and quit", action=ProfilesAction)
profile.add_argument("-i", "--profile-info", help="Give info about a profile and quit", metavar="PROFILE", action=ProfileInfoAction)
profile.add_argument("-p", "--profile", help="Specify the profile", metavar="PROFILE", required=True)
profile.add_argument("-D", "--define", help="Define an option to be used by the profile", metavar="NAME[=VALUE]", action=DefineAction)
mode = parser.add_argument_group("Mode").add_mutually_exclusive_group()
mode.add_argument("-B", "--batch", help="Batch process every input file into an output file (default)", action="store_true")
mode.add_argument("-M", "--merge", help="Merge streams from all inputs", action="store_true")
mode.add_argument("-C", "--concat", help="Concatenate streams from inputs", action="store_true")
files = parser.add_argument_group("Files")
files.add_argument("inputs", help="The input file(s)", nargs="+")
files.add_argument("output", help="The output filename or directory (unless -t is given)", nargs="?") # always empty
files.add_argument("-u", "--update", help="Only work on files that don't already exist", action="store_true")
files.add_argument("-c", "--create-directory", help="Create directories if they don't exist", action="store_true")
target = files.add_mutually_exclusive_group()
target.add_argument("-t", "--target-directory", help="Output into a directory", metavar="DIRECTORY", type=Path)
target.add_argument("-T", "--no-target-directory", help="Treat output as a normal file", action="store_true")
files.add_argument("-S", "--subdirectory", help="Work in a subdirectory of here and -t (use glob patterns for inputs)")
files.add_argument("-K", "--copy-files", help="Copy all following files unmodified", metavar="FILE", action=ExtendAction)
progs = parser.add_argument_group("Programs")
progs.add_argument("--ffmpeg", help="Path to the ffmpeg executable", default="ffmpeg")
progs.add_argument("--ffprobe", help="Path to the ffprobe executable", default="ffprobe")
# Parse arguments
args = parser.parse_args(argv[1:])
# Figure out output path
# ----------------------
# Fill in args.output
# args.output will never be filled in by argparse, since inputs consumes everything
if args.target_directory:
args.output = args.target_directory
elif len(args.inputs) < 2:
parser.error("Neither --target-directory nor output is given")
else:
args.output = Path(args.inputs.pop(-1))
if args.subdirectory:
subdir = Path(args.subdirectory)#.resolve()
outdir = Path(args.output, args.subdirectory)#.resolve()
if outdir.exists() and not outdir.is_dir():
parser.error("--subdirectory only works with output directories. '%s' exists and isn't a directory")
inputs = args.inputs
args.inputs = []
for pattern in inputs:
args.inputs.extend(subdir.glob(pattern))
files = args.copy_files
args.copy_files = []
for pattern in files:
args.copy_files.extend(subdir.glob(pattern))
args.output_directory = args.output = outdir
args.output_filename = None
else:
# Check if we're outputting to a directory
multiple_outputs = args.copy_files or not (args.merge or args.concat) and len(args.inputs) > 1
if args.target_directory or args.output.is_dir() or multiple_outputs:
if args.no_target_directory:
if multiple_outputs:
parser.error("Passed --no-target-directory, but operation would have multiple outputs. (See --merge or --concat)")
else:
parser.error("Passed --no-target-directory, but '%s' is an existing directory." % args.output)
args.output_filename = None
args.output_directory = args.output
else:
args.output_filename = args.output.name
args.output_directory = args.output.parent
return args