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/>.
|
|
|
|
|
2024-06-07 07:25:32 +02:00
|
|
|
import re
|
|
|
|
import requests
|
|
|
|
from flask import abort, send_file, Blueprint, current_app, request
|
2023-11-10 19:57:49 +01:00
|
|
|
import os
|
|
|
|
from PIL import Image
|
|
|
|
|
2019-11-16 00:51:42 +01:00
|
|
|
|
|
|
|
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)]
|
2024-07-27 23:17:04 +02:00
|
|
|
ALLOWED_MIMETYPES = {
|
|
|
|
"png": "image/png",
|
|
|
|
"webp": "image/webp",
|
|
|
|
"jpg": "image/jpeg",
|
|
|
|
}
|
2023-11-10 19:57:49 +01:00
|
|
|
|
2018-05-29 17:19:17 +02:00
|
|
|
|
|
|
|
def mkdir(path):
|
2020-01-18 02:38:00 +01:00
|
|
|
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:]
|
2024-07-27 23:17:04 +02:00
|
|
|
if ext not in ALLOWED_MIMETYPES:
|
2023-11-10 19:57:49 +01:00
|
|
|
abort(404)
|
|
|
|
|
2024-07-27 23:17:04 +02:00
|
|
|
for other_ext in ALLOWED_MIMETYPES.keys():
|
2023-11-10 19:57:49 +01:00
|
|
|
other_path = f"{start}.{other_ext}"
|
|
|
|
if ext != other_ext and os.path.isfile(other_path):
|
|
|
|
return other_path
|
|
|
|
|
|
|
|
abort(404)
|
|
|
|
|
|
|
|
|
2024-07-27 23:17:04 +02:00
|
|
|
def get_mimetype(cache_filepath: str) -> str:
|
|
|
|
period = cache_filepath.rfind(".")
|
|
|
|
ext = cache_filepath[period + 1:]
|
|
|
|
mimetype = ALLOWED_MIMETYPES.get(ext)
|
|
|
|
if mimetype is None:
|
|
|
|
abort(404)
|
|
|
|
return mimetype
|
|
|
|
|
|
|
|
|
2019-11-16 00:51:42 +01:00
|
|
|
@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))
|
|
|
|
|
2024-07-27 23:17:04 +02:00
|
|
|
res = send_file(cache_filepath, mimetype=get_mimetype(cache_filepath))
|
2023-12-16 17:16:41 +01:00
|
|
|
res.headers["Cache-Control"] = "max-age=604800" # 1 week
|
2023-11-12 17:18:20 +01:00
|
|
|
return res
|
2024-06-07 07:25:32 +02:00
|
|
|
|
|
|
|
|
|
|
|
@bp.route("/thumbnails/youtube/<id_>.jpg")
|
|
|
|
def youtube(id_: str):
|
|
|
|
if not re.match(r"^[A-Za-z0-9\-_]+$", id_):
|
|
|
|
abort(400)
|
|
|
|
|
|
|
|
cache_dir = os.path.join(current_app.config["THUMBNAIL_DIR"], "youtube")
|
|
|
|
os.makedirs(cache_dir, exist_ok=True)
|
|
|
|
cache_filepath = os.path.join(cache_dir, id_ + ".jpg")
|
|
|
|
|
|
|
|
url = f"https://img.youtube.com/vi/{id_}/default.jpg"
|
|
|
|
|
|
|
|
response = requests.get(url, stream=True)
|
|
|
|
if response.status_code != 200:
|
|
|
|
abort(response.status_code)
|
|
|
|
|
|
|
|
with open(cache_filepath, "wb") as file:
|
|
|
|
file.write(response.content)
|
|
|
|
|
|
|
|
res = send_file(cache_filepath)
|
|
|
|
res.headers["Cache-Control"] = "max-age=604800" # 1 week
|
|
|
|
return res
|