2023-04-06 16:22:56 +00:00
|
|
|
"""
|
|
|
|
Onlylegs - Image View
|
|
|
|
"""
|
2023-09-24 14:42:26 +00:00
|
|
|
import os
|
|
|
|
import logging
|
|
|
|
import pathlib
|
2023-04-06 18:42:04 +00:00
|
|
|
from math import ceil
|
2023-09-24 14:42:26 +00:00
|
|
|
from flask import (
|
|
|
|
Blueprint,
|
|
|
|
render_template,
|
|
|
|
url_for,
|
|
|
|
current_app,
|
|
|
|
request,
|
|
|
|
flash,
|
|
|
|
jsonify,
|
|
|
|
)
|
|
|
|
from flask_login import current_user
|
2023-08-04 17:34:08 +00:00
|
|
|
from onlylegs.models import Pictures, AlbumJunction, Albums
|
2023-04-12 16:58:13 +00:00
|
|
|
from onlylegs.extensions import db
|
2023-04-06 16:22:56 +00:00
|
|
|
|
|
|
|
|
2023-04-07 12:35:30 +00:00
|
|
|
blueprint = Blueprint("image", __name__, url_prefix="/image")
|
2023-04-06 16:22:56 +00:00
|
|
|
|
|
|
|
|
2023-09-24 14:42:26 +00:00
|
|
|
@blueprint.route("/<int:image_id>", methods=["GET"])
|
2023-04-06 16:22:56 +00:00
|
|
|
def image(image_id):
|
|
|
|
"""
|
|
|
|
Image view, shows the image and its metadata
|
|
|
|
"""
|
|
|
|
# Get the image, if it doesn't exist, 404
|
2023-08-04 17:34:08 +00:00
|
|
|
image = db.get_or_404(Pictures, image_id, description="Image not found :<")
|
2023-04-06 16:22:56 +00:00
|
|
|
|
2023-04-12 15:16:43 +00:00
|
|
|
# Get all groups the image is in
|
2023-04-07 12:35:30 +00:00
|
|
|
groups = (
|
2023-08-04 17:34:08 +00:00
|
|
|
AlbumJunction.query.with_entities(AlbumJunction.album_id)
|
|
|
|
.filter(AlbumJunction.picture_id == image_id)
|
2023-04-07 12:35:30 +00:00
|
|
|
.all()
|
|
|
|
)
|
2023-04-06 16:22:56 +00:00
|
|
|
|
2023-04-12 15:16:43 +00:00
|
|
|
# Get the group data for each group the image is in
|
2023-04-06 16:22:56 +00:00
|
|
|
image.groups = []
|
|
|
|
for group in groups:
|
2023-04-07 12:35:30 +00:00
|
|
|
image.groups.append(
|
2023-08-04 17:34:08 +00:00
|
|
|
Albums.query.with_entities(Albums.id, Albums.name)
|
|
|
|
.filter(Albums.id == group[0])
|
2023-04-07 12:35:30 +00:00
|
|
|
.first()
|
|
|
|
)
|
2023-04-06 16:22:56 +00:00
|
|
|
|
|
|
|
# Get the next and previous images
|
|
|
|
# Check if there is a group ID set
|
2023-04-07 12:35:30 +00:00
|
|
|
next_url = (
|
2023-08-04 17:34:08 +00:00
|
|
|
Pictures.query.with_entities(Pictures.id)
|
|
|
|
.filter(Pictures.id > image_id)
|
|
|
|
.order_by(Pictures.id.asc())
|
2023-04-07 12:35:30 +00:00
|
|
|
.first()
|
|
|
|
)
|
|
|
|
prev_url = (
|
2023-08-04 17:34:08 +00:00
|
|
|
Pictures.query.with_entities(Pictures.id)
|
|
|
|
.filter(Pictures.id < image_id)
|
|
|
|
.order_by(Pictures.id.desc())
|
2023-04-07 12:35:30 +00:00
|
|
|
.first()
|
|
|
|
)
|
2023-04-06 16:22:56 +00:00
|
|
|
|
|
|
|
# If there is a next or previous image, get the url
|
2023-09-24 14:42:26 +00:00
|
|
|
next_url = url_for("image.image", image_id=next_url[0]) if next_url else None
|
|
|
|
prev_url = url_for("image.image", image_id=prev_url[0]) if prev_url else None
|
2023-04-07 12:35:30 +00:00
|
|
|
|
2023-04-06 18:42:04 +00:00
|
|
|
# Yoink all the images in the database
|
2023-08-04 17:34:08 +00:00
|
|
|
total_images = (
|
|
|
|
Pictures.query.with_entities(Pictures.id).order_by(Pictures.id.desc()).all()
|
|
|
|
)
|
2023-04-07 12:35:30 +00:00
|
|
|
limit = current_app.config["UPLOAD_CONF"]["max-load"]
|
2023-04-06 18:42:04 +00:00
|
|
|
|
|
|
|
# If the number of items is less than the limit, no point of calculating the page
|
|
|
|
if len(total_images) <= limit:
|
2023-04-07 09:18:03 +00:00
|
|
|
return_page = None
|
2023-04-06 18:42:04 +00:00
|
|
|
else:
|
|
|
|
# How many pages should there be
|
|
|
|
for i in range(ceil(len(total_images) / limit)):
|
|
|
|
# Slice the list of IDs into chunks of the limit
|
2023-04-07 12:35:30 +00:00
|
|
|
for j in total_images[i * limit : (i + 1) * limit]:
|
2023-04-06 18:42:04 +00:00
|
|
|
# Is our image in this chunk?
|
2023-04-10 23:12:37 +00:00
|
|
|
if not image_id > j[-1]:
|
2023-04-06 18:42:04 +00:00
|
|
|
return_page = i + 1
|
|
|
|
break
|
2023-04-06 16:22:56 +00:00
|
|
|
|
2023-05-29 10:40:29 +00:00
|
|
|
close_tab = True
|
|
|
|
if request.cookies.get("image-info") == "0":
|
|
|
|
close_tab = False
|
|
|
|
|
2023-04-07 12:35:30 +00:00
|
|
|
return render_template(
|
|
|
|
"image.html",
|
|
|
|
image=image,
|
|
|
|
next_url=next_url,
|
|
|
|
prev_url=prev_url,
|
|
|
|
return_page=return_page,
|
2023-05-29 10:40:29 +00:00
|
|
|
close_tab=close_tab,
|
2023-04-07 12:35:30 +00:00
|
|
|
)
|
2023-09-24 14:42:26 +00:00
|
|
|
|
|
|
|
|
|
|
|
@blueprint.route("/<int:image_id>", methods=["PUT"])
|
|
|
|
def image_put(image_id):
|
|
|
|
"""
|
|
|
|
Update the image metadata
|
|
|
|
"""
|
|
|
|
image_record = db.get_or_404(Pictures, image_id, description="Image not found :<")
|
|
|
|
|
|
|
|
image_record.description = request.form.get("description", image_record.description)
|
|
|
|
image_record.alt = request.form.get("alt", image_record.alt)
|
|
|
|
|
|
|
|
print(request.form.get("description"))
|
|
|
|
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
flash(["Image updated!", "1"])
|
|
|
|
return "OK", 200
|
|
|
|
|
|
|
|
|
|
|
|
@blueprint.route("/<int:image_id>", methods=["DELETE"])
|
|
|
|
def image_delete(image_id):
|
|
|
|
image_record = db.get_or_404(Pictures, image_id)
|
|
|
|
|
|
|
|
# Check if image exists and if user is allowed to delete it (author)
|
|
|
|
if image_record.author_id != current_user.id:
|
|
|
|
logging.info("User %s tried to delete image %s", current_user.id, image_id)
|
|
|
|
return (
|
|
|
|
jsonify({"message": "You are not allowed to delete this image, heck off"}),
|
|
|
|
403,
|
|
|
|
)
|
|
|
|
|
|
|
|
# Delete file
|
|
|
|
try:
|
|
|
|
os.remove(
|
|
|
|
os.path.join(current_app.config["UPLOAD_FOLDER"], image_record.filename)
|
|
|
|
)
|
|
|
|
except FileNotFoundError:
|
|
|
|
logging.warning(
|
|
|
|
"File not found: %s, already deleted or never existed",
|
|
|
|
image_record.filename,
|
|
|
|
)
|
|
|
|
|
|
|
|
# Delete cached files
|
|
|
|
cache_name = image_record.filename.rsplit(".")[0]
|
|
|
|
for cache_file in pathlib.Path(current_app.config["CACHE_FOLDER"]).glob(
|
|
|
|
cache_name + "*"
|
|
|
|
):
|
|
|
|
os.remove(cache_file)
|
|
|
|
|
|
|
|
AlbumJunction.query.filter_by(picture_id=image_id).delete()
|
|
|
|
db.session.delete(image_record)
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
logging.info("Removed image (%s) %s", image_id, image_record.filename)
|
|
|
|
flash(["Image was all in Le Head!", "1"])
|
|
|
|
return jsonify({"message": "Image deleted"}), 200
|