contentdb/app/blueprints/packages/reviews.py

246 lines
7.9 KiB
Python
Raw Permalink Normal View History

2020-07-12 17:34:25 +02:00
# ContentDB
2020-07-09 05:10:09 +02:00
# Copyright (C) 2020 rubenwardy
#
# 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
2020-07-09 05:10:09 +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.
2020-07-09 05:10:09 +02:00
#
2021-01-30 17:59:42 +01:00
# You should have received a copy of the GNU Affero General Public License
2020-07-09 05:10:09 +02:00
# along with this program. If not, see <https://www.gnu.org/licenses/>.
2022-01-01 23:17:39 +01:00
from collections import namedtuple
2020-07-09 05:10:09 +02:00
2022-01-07 22:55:33 +01:00
from flask_babel import gettext, lazy_gettext
2022-01-07 22:46:16 +01:00
2020-07-09 05:10:09 +02:00
from . import bp
from flask import *
from flask_login import current_user, login_required
2020-07-09 05:10:09 +02:00
from flask_wtf import FlaskForm
from wtforms import *
from wtforms.validators import *
from app.models import db, PackageReview, Thread, ThreadReply, NotificationType, PackageReviewVote, Package, UserRank, \
Permission, AuditSeverity
from app.utils import is_package_page, addNotification, get_int_or_abort, isYes, is_safe_url, rank_required, addAuditLog
2021-08-17 22:16:43 +02:00
from app.tasks.webhooktasks import post_discord_webhook
2020-07-09 05:10:09 +02:00
2020-07-10 21:30:31 +02:00
@bp.route("/reviews/")
def list_reviews():
2020-12-04 00:41:11 +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 = PackageReview.query.order_by(db.desc(PackageReview.created_at)).paginate(page, num, True)
return render_template("packages/reviews_list.html", pagination=pagination, reviews=pagination.items)
2020-07-10 21:30:31 +02:00
2020-07-09 05:10:09 +02:00
class ReviewForm(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)])
recommends = RadioField(lazy_gettext("Private"), [InputRequired()],
choices=[("yes", lazy_gettext("Yes")), ("no", lazy_gettext("No"))])
submit = SubmitField(lazy_gettext("Save"))
2020-07-09 05:10:09 +02:00
@bp.route("/packages/<author>/<name>/review/", methods=["GET", "POST"])
@login_required
@is_package_page
def review(package):
if current_user in package.maintainers:
2022-01-07 22:46:16 +01:00
flash(gettext("You can't review your own package!"), "danger")
2021-07-24 05:30:14 +02:00
return redirect(package.getURL("packages.view"))
2020-07-09 05:10:09 +02:00
review = PackageReview.query.filter_by(package=package, author=current_user).first()
2022-05-09 13:40:01 +02:00
can_review = review is not None or current_user.canReviewRL()
if not can_review:
flash(gettext("You've reviewed too many packages recently. Please wait before trying again, and consider making your reviews more detailed"), "danger")
2020-07-09 05:10:09 +02:00
form = ReviewForm(formdata=request.form, obj=review)
# Set default values
if request.method == "GET" and review:
form.title.data = review.thread.title
form.recommends.data = "yes" if review.recommends else "no"
form.comment.data = review.thread.replies[0].comment
# Validate and submit
2022-05-09 13:40:01 +02:00
elif can_review and form.validate_on_submit():
2020-07-09 05:10:09 +02:00
was_new = False
if not review:
was_new = True
review = PackageReview()
review.package = package
review.author = current_user
db.session.add(review)
review.recommends = form.recommends.data == "yes"
thread = review.thread
if not thread:
thread = Thread()
thread.author = current_user
thread.private = False
thread.package = package
thread.review = review
db.session.add(thread)
thread.watchers.append(current_user)
reply = ThreadReply()
reply.thread = thread
reply.author = current_user
reply.comment = form.comment.data
db.session.add(reply)
thread.replies.append(reply)
else:
reply = thread.replies[0]
reply.comment = form.comment.data
thread.title = form.title.data
db.session.commit()
2020-07-09 05:32:13 +02:00
package.recalcScore()
2020-07-09 05:10:09 +02:00
if was_new:
notif_msg = "New review '{}'".format(form.title.data)
2020-12-05 04:44:34 +01:00
type = NotificationType.NEW_REVIEW
2020-07-09 05:10:09 +02:00
else:
notif_msg = "Updated review '{}'".format(form.title.data)
2020-12-05 04:44:34 +01:00
type = NotificationType.OTHER
2020-07-09 05:10:09 +02:00
2020-12-05 04:44:34 +01:00
addNotification(package.maintainers, current_user, type, notif_msg,
url_for("threads.view", id=thread.id), package)
2020-07-09 05:10:09 +02:00
2021-08-17 22:16:43 +02:00
if was_new:
post_discord_webhook.delay(thread.author.username,
"Reviewed {}: {}".format(package.title, thread.getViewURL(absolute=True)), False)
2020-07-09 05:10:09 +02:00
db.session.commit()
2021-07-24 05:30:14 +02:00
return redirect(package.getURL("packages.view"))
2020-07-09 05:10:09 +02:00
2020-12-04 03:23:04 +01:00
return render_template("packages/review_create_edit.html",
2020-07-10 20:26:37 +02:00
form=form, package=package, review=review)
@bp.route("/packages/<author>/<name>/reviews/<reviewer>/delete/", methods=["POST"])
2020-07-10 20:26:37 +02:00
@login_required
@is_package_page
def delete_review(package, reviewer):
review = PackageReview.query \
.filter(PackageReview.package == package, PackageReview.author.has(username=reviewer)) \
.first()
2020-07-10 20:26:37 +02:00
if review is None or review.package != package:
abort(404)
if not review.checkPerm(current_user, Permission.DELETE_REVIEW):
abort(403)
2020-07-10 20:26:37 +02:00
thread = review.thread
reply = ThreadReply()
reply.thread = thread
reply.author = current_user
reply.comment = "_converted review into a thread_"
2022-04-23 21:53:38 +02:00
reply.is_status_update = True
2020-07-10 20:26:37 +02:00
db.session.add(reply)
thread.review = None
msg = "Converted review by {} to thread".format(review.author.display_name)
addAuditLog(AuditSeverity.MODERATION if current_user.username != reviewer else AuditSeverity.NORMAL,
current_user, msg, thread.getViewURL(), thread.package)
notif_msg = "Deleted review '{}', comments were kept as a thread".format(thread.title)
2020-12-05 04:44:34 +01:00
addNotification(package.maintainers, current_user, NotificationType.OTHER, notif_msg, url_for("threads.view", id=thread.id), package)
2020-07-10 20:26:37 +02:00
db.session.delete(review)
2022-02-09 20:11:31 +01:00
package.recalcScore()
2020-07-10 20:26:37 +02:00
db.session.commit()
return redirect(thread.getViewURL())
2021-08-18 23:09:41 +02:00
def handle_review_vote(package: Package, review_id: int):
if current_user in package.maintainers:
2022-01-07 22:46:16 +01:00
flash(gettext("You can't vote on the reviews on your own package!"), "danger")
2021-08-18 23:09:41 +02:00
return
review: PackageReview = PackageReview.query.get(review_id)
if review is None or review.package != package:
abort(404)
if review.author == current_user:
2022-01-07 22:46:16 +01:00
flash(gettext("You can't vote on your own reviews!"), "danger")
2021-08-18 23:09:41 +02:00
return
2021-08-19 14:45:32 +02:00
is_positive = isYes(request.form["is_positive"])
2021-08-18 23:09:41 +02:00
vote = PackageReviewVote.query.filter_by(review=review, user=current_user).first()
if vote is None:
vote = PackageReviewVote()
vote.review = review
vote.user = current_user
2021-08-19 14:45:32 +02:00
vote.is_positive = is_positive
2021-08-18 23:09:41 +02:00
db.session.add(vote)
2021-08-19 14:45:32 +02:00
elif vote.is_positive == is_positive:
db.session.delete(vote)
2021-08-18 23:09:41 +02:00
else:
2021-08-19 14:45:32 +02:00
vote.is_positive = is_positive
2021-08-18 23:09:41 +02:00
review.update_score()
db.session.commit()
@bp.route("/packages/<author>/<name>/review/<int:review_id>/", methods=["POST"])
@login_required
@is_package_page
def review_vote(package, review_id):
handle_review_vote(package, review_id)
next_url = request.args.get("r")
if next_url and is_safe_url(next_url):
return redirect(next_url)
else:
return redirect(review.thread.getViewURL())
2022-01-01 23:17:39 +01:00
@bp.route("/packages/<author>/<name>/review-votes/")
@rank_required(UserRank.ADMIN)
@is_package_page
def review_votes(package):
user_biases = {}
for review in package.reviews:
review_sign = 1 if review.recommends else -1
for vote in review.votes:
user_biases[vote.user.username] = user_biases.get(vote.user.username, [0, 0])
vote_sign = 1 if vote.is_positive else -1
vote_bias = review_sign * vote_sign
if vote_bias == 1:
user_biases[vote.user.username][0] += 1
else:
user_biases[vote.user.username][1] += 1
BiasInfo = namedtuple("BiasInfo", "username balance with_ against no_vote perc_with")
user_biases_info = []
for username, bias in user_biases.items():
total_votes = bias[0] + bias[1]
balance = bias[0] - bias[1]
perc_with = round((100 * bias[0]) / total_votes)
user_biases_info.append(BiasInfo(username, balance, bias[0], bias[1], len(package.reviews) - total_votes, perc_with))
user_biases_info.sort(key=lambda x: -abs(x.balance))
return render_template("packages/review_votes.html", form=form, package=package, reviews=package.reviews,
user_biases=user_biases_info)