contentdb/app/blueprints/thumbnails/__init__.py

108 lines
3.2 KiB
Python
Raw Normal View History

2020-07-12 17:34:25 +02:00
# ContentDB
2021-01-30 17:59:42 +01:00
# Copyright (C) 2018-21 rubenwardy
2018-05-29 17:19:17 +02:00
#
# This program is free software: you can redistribute it and/or modify
2021-01-30 17:59:42 +01:00
# it under the terms of the GNU Affero General Public License as published by
2018-05-29 17:19:17 +02:00
# 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
2021-01-30 17:59:42 +01:00
# GNU Affero General Public License for more details.
2018-05-29 17:19:17 +02:00
#
2021-01-30 17:59:42 +01:00
# You should have received a copy of the GNU Affero General Public License
2018-05-29 17:19:17 +02:00
# along with this program. If not, see <https://www.gnu.org/licenses/>.
from flask import abort, send_file, Blueprint, current_app
2023-11-10 19:57:49 +01:00
import os
from PIL import Image
bp = Blueprint("thumbnails", __name__)
2018-05-29 17:19:17 +02:00
2023-11-10 19:57:49 +01:00
ALLOWED_RESOLUTIONS = [(100, 67), (270, 180), (350, 233), (1100, 520)]
ALLOWED_EXTENSIONS = {"png", "webp", "jpg"}
2023-11-10 19:57:49 +01:00
2018-05-29 17:19:17 +02:00
def mkdir(path):
assert path != "" and path is not None
2020-07-10 21:50:25 +02:00
try:
if not os.path.isdir(path):
os.mkdir(path)
except FileExistsError:
pass
2018-05-29 17:19:17 +02:00
2018-05-29 17:56:35 +02:00
2018-07-28 17:03:48 +02:00
def resize_and_crop(img_path, modified_path, size):
2023-11-12 17:11:38 +01:00
with Image.open(img_path) as img:
# Get current and desired ratio for the images
img_ratio = img.size[0] / float(img.size[1])
desired_ratio = size[0] / float(size[1])
# Is more portrait than target, scale and crop
if desired_ratio > img_ratio:
img = img.resize((int(size[0]), int(size[0] * img.size[1] / img.size[0])),
Image.BICUBIC)
box = (0, (img.size[1] - size[1]) / 2, img.size[0], (img.size[1] + size[1]) / 2)
img = img.crop(box)
# Is more landscape than target, scale and crop
elif desired_ratio < img_ratio:
img = img.resize((int(size[1] * img.size[0] / img.size[1]), int(size[1])),
Image.BICUBIC)
box = ((img.size[0] - size[0]) / 2, 0, (img.size[0] + size[0]) / 2, img.size[1])
img = img.crop(box)
# Is exactly the same ratio as target
else:
img = img.resize(size, Image.BICUBIC)
if modified_path.endswith(".jpg") and img.mode != "RGB":
img = img.convert("RGB")
img.save(modified_path, lossless=True)
2018-07-28 17:03:48 +02:00
2018-12-21 22:22:15 +01:00
2023-11-10 19:57:49 +01:00
def find_source_file(img):
upload_dir = current_app.config["UPLOAD_DIR"]
source_filepath = os.path.join(upload_dir, img)
if os.path.isfile(source_filepath):
return source_filepath
period = source_filepath.rfind(".")
start = source_filepath[:period]
ext = source_filepath[period + 1:]
if ext not in ALLOWED_EXTENSIONS:
abort(404)
for other_ext in ALLOWED_EXTENSIONS:
other_path = f"{start}.{other_ext}"
if ext != other_ext and os.path.isfile(other_path):
return other_path
abort(404)
@bp.route("/thumbnails/<int:level>/<img>")
2018-12-21 22:22:15 +01:00
def make_thumbnail(img, level):
if level > len(ALLOWED_RESOLUTIONS) or level <= 0:
2018-05-29 17:19:17 +02:00
abort(403)
2018-12-21 22:22:15 +01:00
w, h = ALLOWED_RESOLUTIONS[level - 1]
2020-01-18 02:20:32 +01:00
thumbnail_dir = current_app.config["THUMBNAIL_DIR"]
mkdir(thumbnail_dir)
2018-05-29 17:19:17 +02:00
2020-01-18 02:20:32 +01:00
output_dir = os.path.join(thumbnail_dir, str(level))
mkdir(output_dir)
2018-05-29 17:19:17 +02:00
2023-11-10 19:57:49 +01:00
cache_filepath = os.path.join(output_dir, img)
2023-11-12 17:18:20 +01:00
if not os.path.isfile(cache_filepath):
2023-12-15 16:57:54 +01:00
source_filepath = find_source_file(img)
2023-11-12 17:18:20 +01:00
resize_and_crop(source_filepath, cache_filepath, (w, h))
res = send_file(cache_filepath)
res.cache_control.max_age = 7*24*60*60
return res