contentdb/app/blueprints/threads/__init__.py

383 lines
12 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-06-11 23:49:25 +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-06-11 23:49:25 +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-06-11 23:49:25 +02:00
#
2021-01-30 17:59:42 +01:00
# You should have received a copy of the GNU Affero General Public License
2018-06-11 23:49:25 +02:00
# along with this program. If not, see <https://www.gnu.org/licenses/>.
from flask import *
2022-01-07 22:55:33 +01:00
from flask_babel import gettext, lazy_gettext
2022-01-17 16:06:03 +01:00
from app.markdown import get_user_mentions, render_markdown
2021-08-17 22:16:43 +02:00
from app.tasks.webhooktasks import post_discord_webhook
bp = Blueprint("threads", __name__)
from flask_login import current_user, login_required
2018-06-11 23:49:25 +02:00
from app.models import *
from app.utils import addNotification, isYes, addAuditLog, get_system_user, rank_required
2018-06-11 23:49:25 +02:00
from flask_wtf import FlaskForm
from wtforms import *
from wtforms.validators import *
2020-01-21 23:40:51 +01:00
from app.utils import get_int_or_abort
2018-06-11 23:49:25 +02:00
2022-01-03 02:41:50 +01:00
@bp.route("/threads/")
def list_all():
2018-07-13 22:28:08 +02:00
query = Thread.query
if not Permission.SEE_THREAD.check(current_user):
query = query.filter_by(private=False)
2020-01-21 23:40:51 +01:00
2022-04-23 22:17:03 +02:00
package = None
2020-01-21 23:40:51 +01:00
pid = request.args.get("pid")
if pid:
pid = get_int_or_abort(pid)
2022-04-23 22:17:03 +02:00
package = Package.query.get(pid)
query = query.filter_by(package=package)
2020-01-21 23:40:51 +01:00
2020-12-22 13:22:52 +01:00
query = query.filter_by(review_id=None)
2020-07-10 20:46:14 +02:00
query = query.order_by(db.desc(Thread.created_at))
2020-12-22 13:22:52 +01:00
page = get_int_or_abort(request.args.get("page"), 1)
num = min(40, get_int_or_abort(request.args.get("n"), 100))
pagination = query.paginate(page, num, True)
2022-04-23 22:17:03 +02:00
return render_template("threads/list.html", pagination=pagination, threads=pagination.items, package=package)
2018-06-11 23:49:25 +02:00
2018-07-28 16:30:59 +02:00
@bp.route("/threads/<int:id>/subscribe/", methods=["POST"])
2018-07-28 16:30:59 +02:00
@login_required
def subscribe(id):
2018-07-28 16:30:59 +02:00
thread = Thread.query.get(id)
if thread is None or not thread.checkPerm(current_user, Permission.SEE_THREAD):
abort(404)
if current_user in thread.watchers:
2022-01-07 22:46:16 +01:00
flash(gettext("Already subscribed!"), "success")
2018-07-28 16:30:59 +02:00
else:
2022-01-07 22:46:16 +01:00
flash(gettext("Subscribed to thread"), "success")
2018-07-28 16:30:59 +02:00
thread.watchers.append(current_user)
db.session.commit()
2020-07-11 02:34:51 +02:00
return redirect(thread.getViewURL())
2018-07-28 16:30:59 +02:00
@bp.route("/threads/<int:id>/unsubscribe/", methods=["POST"])
2018-07-28 16:30:59 +02:00
@login_required
def unsubscribe(id):
2018-07-28 16:30:59 +02:00
thread = Thread.query.get(id)
if thread is None or not thread.checkPerm(current_user, Permission.SEE_THREAD):
abort(404)
if current_user in thread.watchers:
2022-01-07 22:46:16 +01:00
flash(gettext("Unsubscribed!"), "success")
2018-07-28 16:30:59 +02:00
thread.watchers.remove(current_user)
db.session.commit()
else:
2022-01-07 22:46:16 +01:00
flash(gettext("Already not subscribed!"), "success")
2018-07-28 16:30:59 +02:00
2020-07-11 02:34:51 +02:00
return redirect(thread.getViewURL())
@bp.route("/threads/<int:id>/set-lock/", methods=["POST"])
@login_required
def set_lock(id):
thread = Thread.query.get(id)
if thread is None or not thread.checkPerm(current_user, Permission.LOCK_THREAD):
abort(404)
thread.locked = isYes(request.args.get("lock"))
if thread.locked is None:
abort(400)
2020-07-11 03:32:17 +02:00
msg = None
2020-07-11 02:34:51 +02:00
if thread.locked:
2020-07-11 03:32:17 +02:00
msg = "Locked thread '{}'".format(thread.title)
2022-01-07 22:46:16 +01:00
flash(gettext("Locked thread"), "success")
2020-07-11 02:34:51 +02:00
else:
2020-07-11 03:32:17 +02:00
msg = "Unlocked thread '{}'".format(thread.title)
2022-01-07 22:46:16 +01:00
flash(gettext("Unlocked thread"), "success")
2020-07-11 02:34:51 +02:00
2020-12-05 04:44:34 +01:00
addNotification(thread.watchers, current_user, NotificationType.OTHER, msg, thread.getViewURL(), thread.package)
2021-06-14 18:40:38 +02:00
addAuditLog(AuditSeverity.MODERATION, current_user, msg, thread.getViewURL(), thread.package)
2020-07-11 03:32:17 +02:00
db.session.commit()
2020-07-11 02:34:51 +02:00
return redirect(thread.getViewURL())
2018-07-28 16:30:59 +02:00
@bp.route("/threads/<int:id>/delete/", methods=["GET", "POST"])
@login_required
def delete_thread(id):
thread = Thread.query.get(id)
if thread is None or not thread.checkPerm(current_user, Permission.DELETE_THREAD):
abort(404)
if request.method == "GET":
return render_template("threads/delete_thread.html", thread=thread)
summary = "\n\n".join([("<{}> {}".format(reply.author.display_name, reply.comment)) for reply in thread.replies])
msg = "Deleted thread {} by {}".format(thread.title, thread.author.display_name)
db.session.delete(thread)
addAuditLog(AuditSeverity.MODERATION, current_user, msg, None, thread.package, summary)
db.session.commit()
return redirect(url_for("homepage.home"))
@bp.route("/threads/<int:id>/delete-reply/", methods=["GET", "POST"])
@login_required
def delete_reply(id):
thread = Thread.query.get(id)
if thread is None:
abort(404)
reply_id = request.args.get("reply")
if reply_id is None:
abort(404)
reply = ThreadReply.query.get(reply_id)
if reply is None or reply.thread != thread:
abort(404)
if thread.first_reply == reply:
2022-01-07 22:46:16 +01:00
flash(gettext("Cannot delete thread opening post!"), "danger")
return redirect(thread.getViewURL())
if not reply.checkPerm(current_user, Permission.DELETE_REPLY):
abort(403)
if request.method == "GET":
return render_template("threads/delete_reply.html", thread=thread, reply=reply)
msg = "Deleted reply by {}".format(reply.author.display_name)
addAuditLog(AuditSeverity.MODERATION, current_user, msg, thread.getViewURL(), thread.package, reply.comment)
db.session.delete(reply)
db.session.commit()
return redirect(thread.getViewURL())
2020-07-11 04:52:56 +02:00
class CommentForm(FlaskForm):
2022-01-07 22:55:33 +01:00
comment = TextAreaField(lazy_gettext("Comment"), [InputRequired(), Length(10, 2000)])
submit = SubmitField(lazy_gettext("Comment"))
2020-07-11 04:52:56 +02:00
@bp.route("/threads/<int:id>/edit/", methods=["GET", "POST"])
@login_required
def edit_reply(id):
thread = Thread.query.get(id)
if thread is None:
abort(404)
reply_id = request.args.get("reply")
if reply_id is None:
abort(404)
reply = ThreadReply.query.get(reply_id)
if reply is None or reply.thread != thread:
abort(404)
if not reply.checkPerm(current_user, Permission.EDIT_REPLY):
abort(403)
form = CommentForm(formdata=request.form, obj=reply)
2020-12-05 00:07:19 +01:00
if form.validate_on_submit():
2020-07-11 04:52:56 +02:00
comment = form.comment.data
msg = "Edited reply by {}".format(reply.author.display_name)
severity = AuditSeverity.NORMAL if current_user == reply.author else AuditSeverity.MODERATION
2020-12-05 04:44:34 +01:00
addNotification(reply.author, current_user, NotificationType.OTHER, msg, thread.getViewURL(), thread.package)
2020-07-11 04:52:56 +02:00
addAuditLog(severity, current_user, msg, thread.getViewURL(), thread.package, reply.comment)
reply.comment = comment
db.session.commit()
return redirect(thread.getViewURL())
return render_template("threads/edit_reply.html", thread=thread, reply=reply, form=form)
@bp.route("/threads/<int:id>/", methods=["GET", "POST"])
def view(id):
thread: Thread = Thread.query.get(id)
2018-06-11 23:49:25 +02:00
if thread is None or not thread.checkPerm(current_user, Permission.SEE_THREAD):
abort(404)
2022-04-23 21:05:19 +02:00
form = CommentForm(formdata=request.form) if thread.checkPerm(current_user, Permission.COMMENT_THREAD) else None
2018-06-11 23:49:25 +02:00
# Check that title is none to load comments into textarea if redirected from new thread page
if form and form.validate_on_submit() and request.form.get("title") is None:
2022-04-23 21:05:19 +02:00
comment = form.comment.data
2020-07-11 02:34:51 +02:00
if not current_user.canCommentRL():
2022-01-07 22:46:16 +01:00
flash(gettext("Please wait before commenting again"), "danger")
2020-07-11 02:34:51 +02:00
return redirect(thread.getViewURL())
2022-04-23 21:05:19 +02:00
reply = ThreadReply()
reply.author = current_user
reply.comment = comment
db.session.add(reply)
2018-06-12 00:38:03 +02:00
2022-04-23 21:05:19 +02:00
thread.replies.append(reply)
if current_user not in thread.watchers:
thread.watchers.append(current_user)
2022-01-17 16:06:03 +01:00
2022-04-23 21:05:19 +02:00
for mentioned_username in get_user_mentions(render_markdown(comment)):
2022-04-23 22:31:59 +02:00
mentioned = User.query.filter_by(username=mentioned_username).first()
2022-04-23 21:05:19 +02:00
if mentioned is None:
continue
2022-01-17 16:06:03 +01:00
2022-04-23 21:05:19 +02:00
msg = "Mentioned by {} in '{}'".format(current_user.display_name, thread.title)
addNotification(mentioned, current_user, NotificationType.THREAD_REPLY,
msg, thread.getViewURL(), thread.package)
thread.watchers.append(mentioned)
2022-04-23 21:05:19 +02:00
msg = "New comment on '{}'".format(thread.title)
addNotification(thread.watchers, current_user, NotificationType.THREAD_REPLY, msg, thread.getViewURL(), thread.package)
2022-04-23 21:05:19 +02:00
if thread.author == get_system_user():
approvers = User.query.filter(User.rank >= UserRank.APPROVER).all()
addNotification(approvers, current_user, NotificationType.EDITOR_MISC, msg,
thread.getViewURL(), thread.package)
post_discord_webhook.delay(current_user.username,
"Replied to bot messages: {}".format(thread.getViewURL(absolute=True)), True)
2018-06-11 23:49:25 +02:00
2022-04-23 21:05:19 +02:00
db.session.commit()
2018-06-11 23:49:25 +02:00
2022-04-23 21:05:19 +02:00
return redirect(thread.getViewURL())
2018-06-11 23:49:25 +02:00
2022-04-23 21:05:19 +02:00
return render_template("threads/view.html", thread=thread, form=form)
2018-06-11 23:49:25 +02:00
class ThreadForm(FlaskForm):
2022-01-07 22:55:33 +01:00
title = StringField(lazy_gettext("Title"), [InputRequired(), Length(3,100)])
comment = TextAreaField(lazy_gettext("Comment"), [InputRequired(), Length(10, 2000)])
private = BooleanField(lazy_gettext("Private"))
submit = SubmitField(lazy_gettext("Open Thread"))
2018-06-11 23:49:25 +02:00
@bp.route("/threads/new/", methods=["GET", "POST"])
2018-06-11 23:49:25 +02:00
@login_required
def new():
2018-06-11 23:49:25 +02:00
form = ThreadForm(formdata=request.form)
package = None
if "pid" in request.args:
package = Package.query.get(int(request.args.get("pid")))
if package is None:
2022-04-23 22:17:03 +02:00
abort(404)
2018-06-11 23:49:25 +02:00
2022-04-23 22:17:03 +02:00
def_is_private = request.args.get("private") or False
if package is None and not current_user.rank.atLeast(UserRank.APPROVER):
abort(404)
2018-06-11 23:49:25 +02:00
2022-04-23 22:44:27 +02:00
allow_private_change = not package or package.approved
is_review_thread = package and not package.approved
2018-06-11 23:49:25 +02:00
# Check that user can make the thread
2022-04-23 22:17:03 +02:00
if package and not package.checkPerm(current_user, Permission.CREATE_THREAD):
2022-01-07 22:46:16 +01:00
flash(gettext("Unable to create thread!"), "danger")
2019-11-21 20:38:26 +01:00
return redirect(url_for("homepage.home"))
2018-06-11 23:49:25 +02:00
# Only allow creating one thread when not approved
elif is_review_thread and package.review_thread is not None:
# Redirect submit to `view` page, which checks for `title` in the form data and so won't commit the reply
flash(gettext("An approval thread already exists! Consider replying there instead"), "danger")
return redirect(package.review_thread.getViewURL(), code=307)
elif not current_user.canOpenThreadRL():
2022-01-07 22:46:16 +01:00
flash(gettext("Please wait before opening another thread"), "danger")
if package:
2021-07-24 05:30:14 +02:00
return redirect(package.getURL("packages.view"))
else:
2019-11-21 20:38:26 +01:00
return redirect(url_for("homepage.home"))
2018-06-11 23:49:25 +02:00
# Set default values
elif request.method == "GET":
form.private.data = def_is_private
form.title.data = request.args.get("title") or ""
# Validate and submit
2020-12-05 00:07:19 +01:00
elif form.validate_on_submit():
2018-06-11 23:49:25 +02:00
thread = Thread()
thread.author = current_user
thread.title = form.title.data
2022-04-23 22:44:27 +02:00
thread.private = form.private.data if allow_private_change else def_is_private
2018-06-11 23:49:25 +02:00
thread.package = package
db.session.add(thread)
2018-06-12 00:38:03 +02:00
thread.watchers.append(current_user)
2022-04-23 22:17:03 +02:00
if package and package.author != current_user:
2018-06-12 00:38:03 +02:00
thread.watchers.append(package.author)
2018-06-11 23:49:25 +02:00
reply = ThreadReply()
reply.thread = thread
reply.author = current_user
reply.comment = form.comment.data
db.session.add(reply)
thread.replies.append(reply)
db.session.commit()
if is_review_thread:
package.review_thread = thread
2022-01-17 16:06:03 +01:00
for mentioned_username in get_user_mentions(render_markdown(form.comment.data)):
2022-04-23 22:31:59 +02:00
mentioned = User.query.filter_by(username=mentioned_username).first()
2022-01-17 16:06:03 +01:00
if mentioned is None:
continue
msg = "Mentioned by {} in new thread '{}'".format(current_user.display_name, thread.title)
addNotification(mentioned, current_user, NotificationType.NEW_THREAD,
msg, thread.getViewURL(), thread.package)
2022-04-23 22:17:03 +02:00
thread.watchers.append(mentioned)
notif_msg = "New thread '{}'".format(thread.title)
2018-06-11 23:49:25 +02:00
if package is not None:
2020-12-05 04:44:34 +01:00
addNotification(package.maintainers, current_user, NotificationType.NEW_THREAD, notif_msg, thread.getViewURL(), package)
2021-08-16 19:57:05 +02:00
approvers = User.query.filter(User.rank >= UserRank.APPROVER).all()
addNotification(approvers, current_user, NotificationType.EDITOR_MISC, notif_msg, thread.getViewURL(), package)
2018-06-11 23:49:25 +02:00
2021-08-17 22:16:43 +02:00
if is_review_thread:
post_discord_webhook.delay(current_user.username,
2022-01-05 01:27:48 +01:00
"Opened approval thread: {}".format(thread.getViewURL(absolute=True)), True)
2021-08-17 22:16:43 +02:00
2018-06-11 23:49:25 +02:00
db.session.commit()
2020-07-11 03:32:17 +02:00
return redirect(thread.getViewURL())
2018-06-11 23:49:25 +02:00
2022-04-23 22:44:27 +02:00
return render_template("threads/new.html", form=form, allow_private_change=allow_private_change, package=package)
2022-01-03 02:41:50 +01:00
2022-01-03 02:44:12 +01:00
@bp.route("/users/<username>/comments/")
@rank_required(UserRank.EDITOR)
2022-01-03 02:44:12 +01:00
def user_comments(username):
2022-01-03 02:41:50 +01:00
user = User.query.filter_by(username=username).first()
if user is None:
abort(404)
2022-04-23 21:05:19 +02:00
return render_template("threads/user_comments.html", user=user, replies=user.replies)