2018-06-15 20:26:15 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2020-05-18 01:35:53 +02:00
|
|
|
# Copyright 2018-2020 Mike Fährmann
|
2018-06-15 20:26:15 +02:00
|
|
|
#
|
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License version 2 as
|
|
|
|
# published by the Free Software Foundation.
|
|
|
|
|
2020-05-18 01:35:53 +02:00
|
|
|
"""Convert Pixiv Ugoira to WebM"""
|
2018-06-15 20:26:15 +02:00
|
|
|
|
|
|
|
from .common import PostProcessor
|
2018-06-18 17:25:52 +02:00
|
|
|
from .. import util
|
2018-07-20 22:06:48 +02:00
|
|
|
import collections
|
2018-06-15 20:26:15 +02:00
|
|
|
import subprocess
|
|
|
|
import tempfile
|
|
|
|
import zipfile
|
2018-06-20 18:48:10 +02:00
|
|
|
import os
|
2018-06-15 20:26:15 +02:00
|
|
|
|
|
|
|
|
|
|
|
class UgoiraPP(PostProcessor):
|
|
|
|
|
2020-05-18 01:35:53 +02:00
|
|
|
def __init__(self, job, options):
|
|
|
|
PostProcessor.__init__(self, job)
|
2018-06-16 12:03:53 +02:00
|
|
|
self.extension = options.get("extension") or "webm"
|
2018-09-23 22:26:35 +02:00
|
|
|
self.args = options.get("ffmpeg-args") or ()
|
2018-08-29 15:58:01 +02:00
|
|
|
self.twopass = options.get("ffmpeg-twopass", False)
|
|
|
|
self.output = options.get("ffmpeg-output", True)
|
2018-06-18 17:25:52 +02:00
|
|
|
self.delete = not options.get("keep-files", False)
|
2018-06-15 20:26:15 +02:00
|
|
|
|
2018-06-20 18:48:10 +02:00
|
|
|
ffmpeg = options.get("ffmpeg-location")
|
|
|
|
self.ffmpeg = util.expand_path(ffmpeg) if ffmpeg else "ffmpeg"
|
|
|
|
|
2018-07-20 22:06:48 +02:00
|
|
|
rate = options.get("framerate", "auto")
|
|
|
|
if rate != "auto":
|
|
|
|
self.calculate_framerate = lambda _: (None, rate)
|
|
|
|
|
2018-09-23 22:26:35 +02:00
|
|
|
if options.get("libx264-prevent-odd", True):
|
|
|
|
# get last video-codec argument
|
|
|
|
vcodec = None
|
|
|
|
for index, arg in enumerate(self.args):
|
|
|
|
arg, _, stream = arg.partition(":")
|
|
|
|
if arg == "-vcodec" or arg in ("-c", "-codec") and (
|
|
|
|
not stream or stream.partition(":")[0] in ("v", "V")):
|
|
|
|
vcodec = self.args[index + 1]
|
2020-02-07 17:56:26 +01:00
|
|
|
# use filter when using libx264/5
|
2018-09-23 22:26:35 +02:00
|
|
|
self.prevent_odd = (
|
|
|
|
vcodec in ("libx264", "libx265") or
|
|
|
|
not vcodec and self.extension.lower() in ("mp4", "mkv"))
|
|
|
|
else:
|
|
|
|
self.prevent_odd = False
|
2018-09-21 19:52:45 +02:00
|
|
|
|
2018-10-18 22:32:03 +02:00
|
|
|
def prepare(self, pathfmt):
|
|
|
|
self._frames = None
|
|
|
|
|
2019-08-12 21:40:37 +02:00
|
|
|
if pathfmt.extension != "zip":
|
2018-08-27 20:58:45 +02:00
|
|
|
return
|
|
|
|
|
2019-08-12 21:40:37 +02:00
|
|
|
if "frames" in pathfmt.kwdict:
|
|
|
|
self._frames = pathfmt.kwdict["frames"]
|
|
|
|
elif "pixiv_ugoira_frame_data" in pathfmt.kwdict:
|
|
|
|
self._frames = pathfmt.kwdict["pixiv_ugoira_frame_data"]["data"]
|
2018-08-27 20:58:45 +02:00
|
|
|
else:
|
2018-06-15 20:26:15 +02:00
|
|
|
return
|
|
|
|
|
2018-10-18 22:32:03 +02:00
|
|
|
if self.delete:
|
|
|
|
pathfmt.set_extension(self.extension)
|
|
|
|
|
|
|
|
def run(self, pathfmt):
|
|
|
|
if not self._frames:
|
|
|
|
return
|
|
|
|
|
|
|
|
rate_in, rate_out = self.calculate_framerate(self._frames)
|
2018-06-15 20:26:15 +02:00
|
|
|
|
|
|
|
with tempfile.TemporaryDirectory() as tempdir:
|
|
|
|
# extract frames
|
2018-06-18 17:25:52 +02:00
|
|
|
with zipfile.ZipFile(pathfmt.temppath) as zfile:
|
2018-06-15 20:26:15 +02:00
|
|
|
zfile.extractall(tempdir)
|
|
|
|
|
|
|
|
# write ffconcat file
|
|
|
|
ffconcat = tempdir + "/ffconcat.txt"
|
|
|
|
with open(ffconcat, "w") as file:
|
|
|
|
file.write("ffconcat version 1.0\n")
|
2018-10-18 22:32:03 +02:00
|
|
|
for frame in self._frames:
|
2018-07-20 22:06:48 +02:00
|
|
|
file.write("file '{}'\n".format(frame["file"]))
|
|
|
|
file.write("duration {}\n".format(frame["delay"] / 1000))
|
|
|
|
if self.extension != "gif":
|
|
|
|
# repeat the last frame to prevent it from only being
|
|
|
|
# displayed for a very short amount of time
|
2018-10-18 22:32:03 +02:00
|
|
|
file.write("file '{}'\n".format(self._frames[-1]["file"]))
|
2018-06-15 20:26:15 +02:00
|
|
|
|
2018-07-20 22:06:48 +02:00
|
|
|
# collect command-line arguments
|
|
|
|
args = [self.ffmpeg]
|
|
|
|
if rate_in:
|
2020-02-07 17:56:26 +01:00
|
|
|
args += ("-r", str(rate_in))
|
|
|
|
args += ("-i", ffconcat)
|
2018-07-20 22:06:48 +02:00
|
|
|
if rate_out:
|
2020-02-07 17:56:26 +01:00
|
|
|
args += ("-r", str(rate_out))
|
2018-09-21 19:52:45 +02:00
|
|
|
if self.prevent_odd:
|
2020-02-07 17:56:26 +01:00
|
|
|
args += ("-vf", "crop=iw-mod(iw\\,2):ih-mod(ih\\,2)")
|
2018-06-16 12:03:53 +02:00
|
|
|
if self.args:
|
|
|
|
args += self.args
|
2018-07-20 22:06:48 +02:00
|
|
|
self.log.debug("ffmpeg args: %s", args)
|
|
|
|
|
|
|
|
# invoke ffmpeg
|
|
|
|
pathfmt.set_extension(self.extension)
|
2019-08-31 21:55:42 +02:00
|
|
|
try:
|
|
|
|
if self.twopass:
|
|
|
|
if "-f" not in args:
|
2020-02-07 17:56:26 +01:00
|
|
|
args += ("-f", self.extension)
|
|
|
|
args += ("-passlogfile", tempdir + "/ffmpeg2pass", "-pass")
|
2019-08-31 21:55:42 +02:00
|
|
|
self._exec(args + ["1", "-y", os.devnull])
|
|
|
|
self._exec(args + ["2", pathfmt.realpath])
|
|
|
|
else:
|
|
|
|
args.append(pathfmt.realpath)
|
|
|
|
self._exec(args)
|
|
|
|
except OSError as exc:
|
|
|
|
print()
|
|
|
|
self.log.error("Unable to invoke FFmpeg (%s: %s)",
|
|
|
|
exc.__class__.__name__, exc)
|
|
|
|
pathfmt.realpath = pathfmt.temppath
|
2018-06-20 18:48:10 +02:00
|
|
|
else:
|
2019-08-31 21:55:42 +02:00
|
|
|
if self.delete:
|
|
|
|
pathfmt.delete = True
|
|
|
|
else:
|
|
|
|
pathfmt.set_extension("zip")
|
2018-06-15 20:26:15 +02:00
|
|
|
|
2018-08-29 15:58:01 +02:00
|
|
|
def _exec(self, args):
|
|
|
|
out = None if self.output else subprocess.DEVNULL
|
|
|
|
return subprocess.Popen(args, stdout=out, stderr=out).wait()
|
|
|
|
|
2018-07-20 22:06:48 +02:00
|
|
|
@staticmethod
|
|
|
|
def calculate_framerate(framelist):
|
|
|
|
counter = collections.Counter(frame["delay"] for frame in framelist)
|
|
|
|
fps = "1000/{}".format(min(counter))
|
|
|
|
return (fps, None) if len(counter) == 1 else (None, fps)
|
|
|
|
|
2018-06-15 20:26:15 +02:00
|
|
|
|
|
|
|
__postprocessor__ = UgoiraPP
|