From 22e2eacc8c7388a0553252f2bb1728eb5349f412 Mon Sep 17 00:00:00 2001 From: mtyton Date: Sat, 21 Dec 2024 13:05:14 +0100 Subject: [PATCH] Fixed issues with Image post editing. Added utility to check if image is local or remote. --- app/community/forms.py | 1 + app/models.py | 1 + app/post/routes.py | 27 ++++++++++++++++++--------- app/templates/post/post_edit.html | 2 -- app/utils.py | 8 ++++++++ requirements.txt | 1 + 6 files changed, 29 insertions(+), 11 deletions(-) diff --git a/app/community/forms.py b/app/community/forms.py index 6eb5c362..970dacda 100644 --- a/app/community/forms.py +++ b/app/community/forms.py @@ -180,6 +180,7 @@ class CreateImageForm(CreatePostForm): class EditImageForm(CreateImageForm): image_file = FileField(_l('Replace Image'), validators=[DataRequired()], render_kw={'accept': 'image/*'}) + image_file = FileField(_l('Image'), validators=[Optional()], render_kw={'accept': 'image/*'}) def validate(self, extra_validators=None) -> bool: if self.communities: diff --git a/app/models.py b/app/models.py index 7c35cee9..7457f7c6 100644 --- a/app/models.py +++ b/app/models.py @@ -1229,6 +1229,7 @@ class Post(db.Model): if blocked_phrase in post.body: return None + file_path = None if ('attachment' in request_json['object'] and isinstance(request_json['object']['attachment'], list) and len(request_json['object']['attachment']) > 0 and diff --git a/app/post/routes.py b/app/post/routes.py index 4e2f6ffa..61f43881 100644 --- a/app/post/routes.py +++ b/app/post/routes.py @@ -32,7 +32,7 @@ from app.utils import get_setting, render_template, allowlist_html, markdown_to_ blocked_instances, blocked_domains, community_moderators, blocked_phrases, show_ban_message, recently_upvoted_posts, \ recently_downvoted_posts, recently_upvoted_post_replies, recently_downvoted_post_replies, reply_is_stupid, \ languages_for_form, menu_topics, add_to_modlog, blocked_communities, piefed_markdown_to_lemmy_markdown, \ - permission_required, blocked_users, get_request + permission_required, blocked_users, get_request, is_local_image_url, is_video_url def show_post(post_id: int): @@ -730,14 +730,23 @@ def post_reply_options(post_id: int, comment_id: int): @login_required def post_edit(post_id: int): post = Post.query.get_or_404(post_id) + post_type = post.type if post.type == POST_TYPE_ARTICLE: form = CreateDiscussionForm() elif post.type == POST_TYPE_LINK: form = CreateLinkForm() elif post.type == POST_TYPE_IMAGE: - form = EditImageForm() + if post.image and post.image.source_url and is_local_image_url(post.image.source_url): + form = EditImageForm() + else: + form = CreateLinkForm() + post_type = POST_TYPE_LINK elif post.type == POST_TYPE_VIDEO: - form = CreateVideoForm() + if is_video_url(post.url): + form = CreateVideoForm() + else: + form = CreateLinkForm() + post_type = POST_TYPE_LINK elif post.type == POST_TYPE_POLL: form = CreatePollForm() poll = Poll.query.filter_by(post_id=post_id).first() @@ -769,7 +778,7 @@ def post_edit(post_id: int): form.language_id.choices = languages_for_form() if form.validate_on_submit(): - save_post(form, post, post.type) + save_post(form, post, post_type) post.community.last_active = utcnow() post.edited_at = utcnow() @@ -812,9 +821,9 @@ def post_edit(post_id: int): form.sticky.data = post.sticky form.language_id.data = post.language_id form.tags.data = tags_to_string(post) - if post.type == POST_TYPE_LINK: + if post_type == POST_TYPE_LINK: form.link_url.data = post.url - elif post.type == POST_TYPE_IMAGE: + elif post_type == POST_TYPE_IMAGE: # existing_image = True form.image_alt_text.data = post.image.alt_text path = post.image.file_path @@ -826,9 +835,9 @@ def post_edit(post_id: int): with open(path, "rb")as file: form.image_file.data = file.read() - elif post.type == POST_TYPE_VIDEO: + elif post_type == POST_TYPE_VIDEO: form.video_url.data = post.url - elif post.type == POST_TYPE_POLL: + elif post_type == POST_TYPE_POLL: poll = Poll.query.filter_by(post_id=post.id).first() form.mode.data = poll.mode form.local_only.data = poll.local_only @@ -841,7 +850,7 @@ def post_edit(post_id: int): if not (post.community.is_moderator() or post.community.is_owner() or current_user.is_admin()): form.sticky.render_kw = {'disabled': True} return render_template('post/post_edit.html', title=_('Edit post'), form=form, - post_type=post.type, community=post.community, post=post, + post_type=post_type, community=post.community, post=post, markdown_editor=current_user.markdown_editor, mods=mod_list, moderating_communities=moderating_communities(current_user.get_id()), joined_communities=joined_communities(current_user.get_id()), diff --git a/app/templates/post/post_edit.html b/app/templates/post/post_edit.html index 390ad57a..ec0fdc0d 100644 --- a/app/templates/post/post_edit.html +++ b/app/templates/post/post_edit.html @@ -32,8 +32,6 @@ {% endif -%} - {% else %} - {{ render_field(form.image_file) }} {% endif %} {{ render_field(form.image_file) }} {{ render_field(form.image_alt_text) }} diff --git a/app/utils.py b/app/utils.py index 14bfdc06..999c4115 100644 --- a/app/utils.py +++ b/app/utils.py @@ -25,6 +25,7 @@ import jwt warnings.filterwarnings("ignore", category=MarkupResemblesLocatorWarning) import os +from furl import furl from flask import current_app, json, redirect, url_for, request, make_response, Response, g, flash from flask_babel import _ from flask_login import current_user, logout_user @@ -199,6 +200,13 @@ def is_image_url(url): return any(path.endswith(extension) for extension in common_image_extensions) +def is_local_image_url(url): + if not is_image_url(url): + return False + f = furl(url) + return f.host in ["127.0.0.1", current_app.config["SERVER_NAME"]] + + def is_video_url(url: str) -> bool: common_video_extensions = ['.mp4', '.webm'] mime_type = mime_type_using_head(url) diff --git a/requirements.txt b/requirements.txt index 3f0cbefd..5a88b76b 100644 --- a/requirements.txt +++ b/requirements.txt @@ -33,3 +33,4 @@ pytesseract==0.3.10 sentry-sdk==1.40.6 python-slugify==8.0.4 moviepy==1.0.3 +furl==2.1.3