pyfedi/app/shared/reply.py

340 lines
13 KiB
Python
Raw Normal View History

from app import cache, db
from app.activitypub.signature import default_context, post_request_in_background, post_request
from app.community.util import send_to_remote_instance
from app.constants import *
2024-10-27 10:20:38 +00:00
from app.models import Instance, Notification, NotificationSubscription, Post, PostReply, PostReplyBookmark, Report, Site, User, utcnow
2024-11-02 23:56:56 +00:00
from app.shared.tasks import task_selector
from app.utils import gibberish, instance_banned, render_template, authorise_api_user, recently_upvoted_post_replies, recently_downvoted_post_replies, shorten_string, \
piefed_markdown_to_lemmy_markdown, markdown_to_html, ap_datetime
from flask import abort, current_app, flash, redirect, request, url_for
from flask_babel import _
from flask_login import current_user
# would be in app/constants.py
SRC_WEB = 1
SRC_PUB = 2
SRC_API = 3
# function can be shared between WEB and API (only API calls it for now)
# comment_vote in app/post/routes would just need to do 'return vote_for_reply(reply_id, vote_direction, SRC_WEB)'
def vote_for_reply(reply_id: int, vote_direction, src, auth=None):
if src == SRC_API:
reply = PostReply.query.filter_by(id=reply_id).one()
user = authorise_api_user(auth, return_type='model')
else:
reply = PostReply.query.get_or_404(reply_id)
user = current_user
undo = reply.vote(user, vote_direction)
2024-11-02 23:56:56 +00:00
task_selector('vote_for_reply', user_id=user.id, reply_id=reply_id, vote_to_undo=undo, vote_direction=vote_direction)
if src == SRC_API:
return user.id
else:
recently_upvoted = []
recently_downvoted = []
if vote_direction == 'upvote' and undo is None:
recently_upvoted = [reply_id]
elif vote_direction == 'downvote' and undo is None:
recently_downvoted = [reply_id]
return render_template('post/_reply_voting_buttons.html', comment=reply,
recently_upvoted_replies=recently_upvoted,
recently_downvoted_replies=recently_downvoted,
community=reply.community)
# function can be shared between WEB and API (only API calls it for now)
# post_reply_bookmark in app/post/routes would just need to do 'return bookmark_the_post_reply(comment_id, SRC_WEB)'
def bookmark_the_post_reply(comment_id: int, src, auth=None):
if src == SRC_API:
post_reply = PostReply.query.filter_by(id=comment_id, deleted=False).one()
user_id = authorise_api_user(auth)
else:
post_reply = PostReply.query.get_or_404(comment_id)
if post_reply.deleted:
abort(404)
user_id = current_user.id
existing_bookmark = PostReplyBookmark.query.filter(PostReplyBookmark.post_reply_id == comment_id,
PostReplyBookmark.user_id == user_id).first()
if not existing_bookmark:
db.session.add(PostReplyBookmark(post_reply_id=comment_id, user_id=user_id))
db.session.commit()
if src == SRC_WEB:
flash(_('Bookmark added.'))
else:
if src == SRC_WEB:
flash(_('This comment has already been bookmarked'))
if src == SRC_API:
return user_id
else:
return redirect(url_for('activitypub.post_ap', post_id=post_reply.post_id, _anchor=f'comment_{comment_id}'))
# function can be shared between WEB and API (only API calls it for now)
# post_reply_remove_bookmark in app/post/routes would just need to do 'return remove_the_bookmark_from_post_reply(comment_id, SRC_WEB)'
def remove_the_bookmark_from_post_reply(comment_id: int, src, auth=None):
if src == SRC_API:
post_reply = PostReply.query.filter_by(id=comment_id, deleted=False).one()
user_id = authorise_api_user(auth)
else:
post_reply = PostReply.query.get_or_404(comment_id)
if post_reply.deleted:
abort(404)
user_id = current_user.id
existing_bookmark = PostReplyBookmark.query.filter(PostReplyBookmark.post_reply_id == comment_id,
PostReplyBookmark.user_id == user_id).first()
if existing_bookmark:
db.session.delete(existing_bookmark)
db.session.commit()
if src == SRC_WEB:
flash(_('Bookmark has been removed.'))
if src == SRC_API:
return user_id
else:
return redirect(url_for('activitypub.post_ap', post_id=post_reply.post_id))
# function can be shared between WEB and API (only API calls it for now)
# post_reply_notification in app/post/routes would just need to do 'return toggle_post_reply_notification(post_reply_id, SRC_WEB)'
def toggle_post_reply_notification(post_reply_id: int, src, auth=None):
# Toggle whether the current user is subscribed to notifications about replies to this reply or not
if src == SRC_API:
post_reply = PostReply.query.filter_by(id=post_reply_id, deleted=False).one()
user_id = authorise_api_user(auth)
else:
post_reply = PostReply.query.get_or_404(post_reply_id)
if post_reply.deleted:
abort(404)
user_id = current_user.id
existing_notification = NotificationSubscription.query.filter(NotificationSubscription.entity_id == post_reply.id,
NotificationSubscription.user_id == user_id,
NotificationSubscription.type == NOTIF_REPLY).first()
if existing_notification:
db.session.delete(existing_notification)
db.session.commit()
else: # no subscription yet, so make one
new_notification = NotificationSubscription(name=shorten_string(_('Replies to my comment on %(post_title)s',
post_title=post_reply.post.title)), user_id=user_id, entity_id=post_reply.id,
type=NOTIF_REPLY)
db.session.add(new_notification)
db.session.commit()
if src == SRC_API:
return user_id
else:
return render_template('post/_reply_notification_toggle.html', comment={'comment': post_reply})
# there are undoubtedly better algos for this
def basic_rate_limit_check(user):
weeks_active = int((utcnow() - user.created).days / 7)
score = user.post_reply_count * weeks_active
if score > 100:
score = 10
else:
score = int(score/10)
# a user with a 10-week old account, who has made 10 replies, will score 10, so their rate limit will be 0
# a user with a new account, and/or has made zero replies, will score 0 (so will have to wait 10 minutes between each new comment)
# other users will score from 1-9, so their rate limits will be between 9 and 1 minutes.
rate_limit = (10-score)*60
recent_reply = cache.get(f'{user.id} has recently replied')
if not recent_reply:
cache.set(f'{user.id} has recently replied', True, timeout=rate_limit)
return True
else:
return False
def make_reply(input, post, parent_id, src, auth=None):
if src == SRC_API:
user = authorise_api_user(auth, return_type='model')
2024-11-02 23:56:56 +00:00
#if not basic_rate_limit_check(user):
# raise Exception('rate_limited')
content = input['body']
notify_author = input['notify_author']
language_id = input['language_id']
else:
user = current_user
content = input.body.data
notify_author = input.notify_author.data
language_id = input.language_id.data
if parent_id:
parent_reply = PostReply.query.filter_by(id=parent_id).one()
else:
parent_reply = None
# WEBFORM would call 'make_reply' in a try block, so any exception from 'new' would bubble-up for it to handle
reply = PostReply.new(user, post, in_reply_to=parent_reply, body=piefed_markdown_to_lemmy_markdown(content),
body_html=markdown_to_html(content), notify_author=notify_author,
language_id=language_id)
user.language_id = language_id
reply.ap_id = reply.profile_id()
db.session.commit()
if src == SRC_WEB:
input.body.data = ''
flash('Your comment has been added.')
2024-11-02 23:56:56 +00:00
task_selector('make_reply', user_id=user.id, reply_id=reply.id, parent_id=parent_id)
if src == SRC_API:
return user.id, reply
else:
return reply
def edit_reply(input, reply, post, src, auth=None):
if src == SRC_API:
user = authorise_api_user(auth, return_type='model', id_match=reply.user_id)
content = input['body']
notify_author = input['notify_author']
language_id = input['language_id']
else:
user = current_user
content = input.body.data
notify_author = input.notify_author.data
language_id = input.language_id.data
reply.body = piefed_markdown_to_lemmy_markdown(content)
reply.body_html = markdown_to_html(content)
reply.notify_author = notify_author
reply.community.last_active = utcnow()
reply.edited_at = utcnow()
reply.language_id = language_id
db.session.commit()
if src == SRC_WEB:
flash(_('Your changes have been saved.'), 'success')
2024-11-02 23:56:56 +00:00
task_selector('edit_reply', user_id=user.id, reply_id=reply.id, parent_id=reply.parent_id)
if src == SRC_API:
return user.id, reply
else:
return
# just for deletes by owner (mod deletes are classed as 'remove')
def delete_reply(reply_id, src, auth):
if src == SRC_API:
2025-01-18 14:52:09 +00:00
user_id = authorise_api_user(auth)
else:
2024-11-02 23:56:56 +00:00
user_id = current_user.id
2025-01-18 14:52:09 +00:00
reply = PostReply.query.filter_by(id=reply_id, user_id=user_id, deleted=False).one()
reply.deleted = True
2024-11-02 23:56:56 +00:00
reply.deleted_by = user_id
if not reply.author.bot:
2024-11-02 23:56:56 +00:00
reply.post.reply_count -= 1
reply.author.post_reply_count -= 1
db.session.commit()
if src == SRC_WEB:
flash(_('Comment deleted.'))
2024-11-02 23:56:56 +00:00
task_selector('delete_reply', user_id=user_id, reply_id=reply.id)
if src == SRC_API:
2024-11-02 23:56:56 +00:00
return user_id, reply
else:
return
def restore_reply(reply_id, src, auth):
if src == SRC_API:
2025-01-18 14:52:09 +00:00
user_id = authorise_api_user(auth)
else:
2024-11-02 23:56:56 +00:00
user_id = current_user.id
2025-01-18 14:52:09 +00:00
reply = PostReply.query.filter_by(id=reply_id, user_id=user_id, deleted=True).one()
reply.deleted = False
reply.deleted_by = None
if not reply.author.bot:
2024-11-02 23:56:56 +00:00
reply.post.reply_count += 1
reply.author.post_reply_count += 1
db.session.commit()
if src == SRC_WEB:
flash(_('Comment restored.'))
2024-11-02 23:56:56 +00:00
task_selector('restore_reply', user_id=user_id, reply_id=reply.id)
if src == SRC_API:
2024-11-02 23:56:56 +00:00
return user_id, reply
else:
return
2024-10-27 10:20:38 +00:00
def report_reply(reply_id, input, src, auth=None):
if src == SRC_API:
reply = PostReply.query.filter_by(id=reply_id).one()
2024-11-02 23:56:56 +00:00
user_id = authorise_api_user(auth)
2024-10-27 10:20:38 +00:00
reason = input['reason']
description = input['description']
report_remote = input['report_remote']
else:
reply = PostReply.query.get_or_404(reply_id)
2024-11-02 23:56:56 +00:00
user_id = current_user.id
2024-10-27 10:20:38 +00:00
reason = input.reasons_to_string(input.reasons.data)
description = input.description.data
report_remote = input.report_remote.data
if reply.reports == -1: # When a mod decides to ignore future reports, reply.reports is set to -1
if src == SRC_API:
raise Exception('already_reported')
else:
flash(_('Comment has already been reported, thank you!'))
return
2024-11-02 23:56:56 +00:00
report = Report(reasons=reason, description=description, type=2, reporter_id=user_id, suspect_post_id=reply.post.id, suspect_community_id=reply.community.id,
2024-10-27 10:20:38 +00:00
suspect_user_id=reply.author.id, suspect_post_reply_id=reply.id, in_community_id=reply.community.id, source_instance_id=1)
db.session.add(report)
# Notify moderators
already_notified = set()
for mod in reply.community.moderators():
moderator = User.query.get(mod.user_id)
if moderator and moderator.is_local():
notification = Notification(user_id=mod.user_id, title=_('A comment has been reported'),
url=f"https://{current_app.config['SERVER_NAME']}/comment/{reply.id}",
2024-11-02 23:56:56 +00:00
author_id=user_id)
2024-10-27 10:20:38 +00:00
db.session.add(notification)
already_notified.add(mod.user_id)
reply.reports += 1
# todo: only notify admins for certain types of report
for admin in Site.admins():
if admin.id not in already_notified:
2024-11-02 23:56:56 +00:00
notify = Notification(title='Suspicious content', url='/admin/reports', user_id=admin.id, author_id=user_id)
2024-10-27 10:20:38 +00:00
db.session.add(notify)
admin.unread_notifications += 1
db.session.commit()
# federate report to originating instance
if not reply.community.is_local() and report_remote:
summary = reason
if description:
summary += ' - ' + description
2024-11-02 23:56:56 +00:00
task_selector('report_reply', user_id=user_id, reply_id=reply_id, summary=summary)
2024-10-27 10:20:38 +00:00
if src == SRC_API:
2024-11-02 23:56:56 +00:00
return user_id, report
2024-10-27 10:20:38 +00:00
else:
return