yet another simple static gallery generator
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.
 
 
 
 

264 lines
9.5 KiB

# Copyright (c) 2013 - Christophe-Marie Duquesne
# Copyright (c) 2013-2023 - Simon Conseil
# Copyright (c) 2021 - Keith Feldman
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to
# deal in the Software without restriction, including without limitation the
# rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
# sell copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:
# The above copyright notice and this permission notice shall be included in
# all copies or substantial portions of the Software.
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
# FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
# IN THE SOFTWARE.
import logging
import os
import re
import shutil
import subprocess
from os.path import splitext
from PIL import Image as PILImage
from . import image, utils
from .utils import is_valid_html5_video
class SubprocessException(Exception):
pass
def check_subprocess(cmd, source, outname=None):
"""Run the command to resize the video and remove the output file if the
processing fails.
"""
logger = logging.getLogger(__name__)
try:
res = subprocess.run(cmd, capture_output=True)
except KeyboardInterrupt:
logger.debug("Process terminated, removing file %s", outname)
if outname and os.path.isfile(outname):
os.remove(outname)
raise
if res.returncode:
logger.debug("STDOUT:\n %s", res.stdout.decode("utf8"))
logger.debug("STDERR:\n %s", res.stderr.decode("utf8"))
if outname and os.path.isfile(outname):
logger.debug("Removing file %s", outname)
os.remove(outname)
raise SubprocessException("Failed to process " + source)
def video_size(source, converter="ffmpeg"):
"""Return the dimensions of the video."""
res = subprocess.run([converter, "-i", source], stderr=subprocess.PIPE)
stderr = res.stderr.decode("utf8", errors="ignore")
pattern = re.compile(r"Stream.*Video.* ([0-9]+)x([0-9]+)")
match = pattern.search(stderr)
rot_pattern = re.compile(r"rotate\s*:\s*-?(90|270)")
rot_match = rot_pattern.search(stderr)
if match:
x, y = int(match.groups()[0]), int(match.groups()[1])
else:
x = y = 0
if rot_match:
x, y = y, x
return x, y
def get_resize_options(source, converter, output_size):
"""Figure out resize options for video from src and dst sizes.
:param source: path to a video
:param outname: path to the generated video
:param settings: settings dict
"""
logger = logging.getLogger(__name__)
w_src, h_src = video_size(source, converter=converter)
w_dst, h_dst = output_size
logger.debug("Video size: %i, %i -> %i, %i", w_src, h_src, w_dst, h_dst)
# do not resize if input dimensions are smaller than output dimensions
if w_src <= w_dst and h_src <= h_dst:
return []
# http://stackoverflow.com/questions/8218363/maintaining-ffmpeg-aspect-ratio
# + I made a drawing on paper to figure this out
if h_dst * w_src < h_src * w_dst:
# biggest fitting dimension is height
resize_opt = ["-vf", "scale=trunc(oh*a/2)*2:%i" % h_dst]
else:
# biggest fitting dimension is width
resize_opt = ["-vf", "scale=%i:trunc(ow/a/2)*2" % w_dst]
return resize_opt
def _get_empty_if_none_else_variable(variable):
return [] if not variable else variable
def generate_video_pass(converter, source, options, outname=None):
"""Run a single pass of encoding.
:param source: source video
:param options: options to pass to encoder
:param outname: if multi-pass, this is None on the first pass
"""
logger = logging.getLogger(__name__)
outname_opt = [] if not outname else [outname]
# Encoding options improved, thanks to
# http://ffmpeg.org/trac/ffmpeg/wiki/vpxEncodingGuide
cmd = [converter, "-i", source, "-y"] # -y to overwrite output files
cmd += options + outname_opt
logger.debug("Processing video: %s", " ".join(cmd))
check_subprocess(cmd, source, outname=outname)
def generate_video(source, outname, settings):
"""Video processor.
:param source: path to a video
:param outname: path to the generated video
:param settings: settings dict
:param options: array of options passed to ffmpeg
"""
logger = logging.getLogger(__name__)
video_format = settings.get("video_format")
options = settings.get(video_format + "_options")
second_pass_options = settings.get(video_format + "_options_second_pass")
video_always_convert = settings.get("video_always_convert")
converter = settings["video_converter"]
resize_opt = []
if settings.get("video_size"):
resize_opt = get_resize_options(source, converter, settings["video_size"])
base, src_ext = splitext(source)
base, dst_ext = splitext(outname)
if dst_ext == src_ext and not resize_opt and not video_always_convert:
logger.debug(
"For %s, the source and destination extension are the "
"same, there is no resizing to be done, and "
"video_always_convert is False, so the output is "
" being copied",
outname,
)
shutil.copy(source, outname)
return
final_pass_options = _get_empty_if_none_else_variable(options) + resize_opt
if second_pass_options:
generate_video_pass(converter, source, final_pass_options)
final_second_pass_options = (
_get_empty_if_none_else_variable(second_pass_options) + resize_opt
)
generate_video_pass(converter, source, final_second_pass_options, outname)
else:
generate_video_pass(converter, source, final_pass_options, outname)
def generate_thumbnail(
source,
outname,
box,
delay,
fit=True,
options=None,
converter="ffmpeg",
black_retries=0,
black_offset=1,
black_max_colors=4,
):
"""Create a thumbnail image for the video source, based on ffmpeg."""
logger = logging.getLogger(__name__)
tmpfile = outname + ".tmp.jpg"
currentTry = 0
iDelay = int(delay)
while currentTry <= abs(black_retries):
# dump an image of the video
cmd = [converter, "-i", source, "-an", "-r", "1"]
cmd += ["-ss", str(iDelay), "-vframes", "1", "-y", tmpfile]
logger.debug("Create thumbnail for video: %s", " ".join(cmd))
check_subprocess(cmd, source, outname)
if os.path.isfile(tmpfile) and black_retries > 0:
with PILImage.open(tmpfile) as img:
colors = img.getcolors(maxcolors=black_max_colors)
if colors is None:
# There were more colors than maxcolors in the image, it
# looks suitable for a valid thumbnail
break
else:
# Only found 'maxcolors' unique colors, looks like a solid
# color, try again with another seek delay
currentTry += 1
iDelay += abs(black_offset)
else:
break
# Sometimes ffmpeg fails with returncode zero but without producing an
# output file Thus, we need to check if an output file was created. If
# not, assume ffmpeg failed
if not os.path.isfile(tmpfile):
logger.debug("Thumbnail generation failed. Likely due to short video length.")
cmd = [converter, "-i", source, "-an", "-r", "1"]
cmd += ["-ss", "0", "-vframes", "1", "-y", tmpfile]
logger.debug("Retry to create thumbnail for video: %s", " ".join(cmd))
check_subprocess(cmd, source, outname)
# use the generate_thumbnail function from sigal.image
image.generate_thumbnail(tmpfile, outname, box, fit=fit, options=options)
# remove the image
os.unlink(tmpfile)
def process_video(media):
"""Process a video: resize, create thumbnail."""
logger = logging.getLogger(__name__)
settings = media.settings
with utils.raise_if_debug() as status:
if settings["use_orig"] and is_valid_html5_video(media.src_ext):
utils.copy(media.src_path, media.dst_path, symlink=settings["orig_link"])
else:
valid_formats = ["mp4", "webm"]
video_format = settings["video_format"]
if video_format not in valid_formats:
logger.error(
"Invalid video_format. Please choose one of: %s", valid_formats
)
raise ValueError
generate_video(media.src_path, media.dst_path, settings)
if settings["make_thumbs"]:
generate_thumbnail(
media.dst_path,
media.thumb_path,
settings["thumb_size"],
settings["thumb_video_delay"],
fit=settings["thumb_fit"],
options=settings["jpg_options"],
converter=settings["video_converter"],
black_retries=settings["thumb_video_black_retries"],
black_offset=settings["thumb_video_black_retry_offset"],
black_max_colors=settings["thumb_video_black_max_colors"],
)
return status.value