#!/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 . ----------------------------------------------------------- Opus Audiobook profile """ import os from ..profile import * abdefines = dict( bitrate = "Use custom target bitrate", stereo = "Use 2 channels (Ignored for mono source streams)", fancy = "Use higher bitrates (48k mono/64k stereo)", ogg = "Use the .ogg file extension (Currently required on Android)" ) def apply_stream(stream, defines): """ Apply the audiobook profile to an output stream """ stream.codec = "libopus" stream.channels = 1 stream.bitrate = 40_000 stream.set(vbr="on", application="voip") # High Quality if "fancy" in defines: stream.bitrate = 48_000 stream.set(application="audio") # Stereo Options if stream.source.channels > 1 and "stereo" in defines: stream.channels = 2 stream.bitrate = 48_000 if "fancy" in defines: stream.bitrate = 64_000 # Custom bitrate if "bitrate" in defines: stream.bitrate = defines["bitrate"] # Limit to input bitrate stream.bitrate = min(stream.bitrate, stream.source.bitrate) @profile @description("Encode Opus Audiobook") @output(container="ogg", ext="opus") @defines(**abdefines) @singleaudio def audiobook(task, stream, defines): if "ogg" in defines: task.change_format(ext="ogg") apply_stream(task.map_stream(stream), defines) return True # Create multiple files from a single big one @profile @description("Split & Encode Opus Audiobook from M4B chapters") @output(container="ogg", ext="opus") @features(no_single_output=True) @defines(ignore_ends="Ignore chapter end marks and continue until next chapter starts", chapter_only_names="Don't include the input filename in the output filename", **abdefines) @singleaudio def from_chapters(task, stream, defines): # Read chapters from input if "ignore_ends" in defines: # Make sure nothing is cut out because of # broken chapter (end) markers it = iter(task.iter_chapters()) first_chapter = next(it) chapters = [{"title": first_chapter.title}] for chapter in it: chapters[-1]["to"] = chapter.start_time chapters.append({"ss": chapter.start_time, "title": chapter.title}) else: chapters = [{"ss": chapter.start_time, "to": chapter.end_time, "title": chapter.title} for chapter in task.iter_chapters()] # Output filenames ext = "ogg" if "ogg" in defines else "opus" if "chapter_only_names" in defines: fn_template = os.path.join(task.output_directory, "%%s.%s" % ext) else: fn_template = "%s - %%s.%s" % (task.output_prefix, ext) # Set up output files for chapter in chapters: out = task.add_output(fn_template % chapter.pop("title"), "ogg") out.set(**chapter) apply_stream(out.map_stream(stream), defines) return True @profile @description("Split into uniform pieces & Encode Opus Audiobook") @output(container="ogg", ext="opus") @features(no_single_output=True) @defines(interval="Length of a piece in minutes (default 30)", minimum="Don't open a new piece if it's shorter than this (default 5)", **abdefines) @singleaudio def explode(task, stream, defines): interval = float(defines.get("interval", 30)) * 60 min_intv = float(defines.get("minimum", 5)) * 60 last_duration = stream.duration % interval npieces = int(stream.duration // interval) + (1 if last_duration > min_intv else 0) ext = "ogg" if "ogg" in defines else "opus" if npieces > 1: fn_template = "%s - %%02i.%s" % (task.output_prefix, ext) out = task.add_output(fn_template % 1, "ogg") for i in range(1, npieces): ts = i * interval out.set(to=ts) out = task.add_output(fn_template % (i + 1), "ogg") out.set(ss=ts) else: task.add_output("%s.%s" % (task.output_prefix, ext), "ogg") for out in task.outputs: apply_stream(out.map_stream(stream), defines) return True