2023-11-30 23:21:37 +13:00
|
|
|
from datetime import datetime, timedelta
|
|
|
|
|
2023-10-07 21:32:19 +13:00
|
|
|
from flask import redirect, url_for, flash, request, make_response, session, Markup, current_app, abort
|
|
|
|
from flask_login import login_user, logout_user, current_user, login_required
|
|
|
|
from flask_babel import _
|
|
|
|
|
|
|
|
from app import db
|
2023-11-30 23:21:37 +13:00
|
|
|
from app.models import Post, Community, CommunityMember, User, PostReply, PostVote, Notification
|
2023-10-07 21:32:19 +13:00
|
|
|
from app.user import bp
|
|
|
|
from app.user.forms import ProfileForm, SettingsForm
|
2023-10-23 20:18:46 +13:00
|
|
|
from app.utils import get_setting, render_template, markdown_to_html, user_access, markdown_to_text, shorten_string
|
2023-11-30 23:21:37 +13:00
|
|
|
from sqlalchemy import desc, or_, text
|
2023-10-07 21:32:19 +13:00
|
|
|
|
|
|
|
|
|
|
|
def show_profile(user):
|
2023-10-21 15:49:01 +13:00
|
|
|
if user.deleted or user.banned and current_user.is_anonymous():
|
|
|
|
abort(404)
|
2023-11-12 20:54:22 +13:00
|
|
|
posts = Post.query.filter_by(user_id=user.id).order_by(desc(Post.posted_at)).limit(20).all()
|
|
|
|
moderates = Community.query.filter_by(banned=False).join(CommunityMember).filter(CommunityMember.user_id == user.id)\
|
|
|
|
.filter(or_(CommunityMember.is_moderator, CommunityMember.is_owner))
|
|
|
|
upvoted = Post.query.join(PostVote).filter(Post.id == PostVote.post_id, PostVote.effect > 0).order_by(desc(Post.posted_at)).limit(10).all()
|
|
|
|
subscribed = Community.query.filter_by(banned=False).join(CommunityMember).filter(CommunityMember.user_id == user.id).all()
|
2023-10-21 15:49:01 +13:00
|
|
|
if current_user.is_anonymous or user.id != current_user.id:
|
2023-10-07 21:32:19 +13:00
|
|
|
moderates = moderates.filter(Community.private_mods == False)
|
2023-11-12 20:54:22 +13:00
|
|
|
post_replies = PostReply.query.filter_by(user_id=user.id).order_by(desc(PostReply.posted_at)).limit(20).all()
|
2023-10-07 21:32:19 +13:00
|
|
|
canonical = user.ap_public_url if user.ap_public_url else None
|
2023-10-10 22:25:37 +13:00
|
|
|
user.about_html = markdown_to_html(user.about)
|
2023-10-23 20:18:46 +13:00
|
|
|
description = shorten_string(markdown_to_text(user.about), 150) if user.about else None
|
2023-10-07 21:32:19 +13:00
|
|
|
return render_template('user/show_profile.html', user=user, posts=posts, post_replies=post_replies,
|
|
|
|
moderates=moderates.all(), canonical=canonical, title=_('Posts by %(user_name)s',
|
2023-10-23 20:18:46 +13:00
|
|
|
user_name=user.user_name),
|
2023-11-12 20:54:22 +13:00
|
|
|
description=description, subscribed=subscribed, upvoted=upvoted)
|
2023-10-07 21:32:19 +13:00
|
|
|
|
|
|
|
|
|
|
|
@bp.route('/u/<actor>/profile', methods=['GET', 'POST'])
|
2023-10-23 13:03:35 +13:00
|
|
|
@login_required
|
2023-10-07 21:32:19 +13:00
|
|
|
def edit_profile(actor):
|
|
|
|
actor = actor.strip()
|
|
|
|
user = User.query.filter_by(user_name=actor, deleted=False, banned=False, ap_id=None).first()
|
|
|
|
if user is None:
|
|
|
|
abort(404)
|
2023-10-23 20:18:46 +13:00
|
|
|
if current_user.id != user.id:
|
|
|
|
abort(401)
|
2023-10-07 21:32:19 +13:00
|
|
|
form = ProfileForm()
|
|
|
|
if form.validate_on_submit():
|
|
|
|
current_user.email = form.email.data
|
|
|
|
if form.password_field.data.strip() != '':
|
|
|
|
current_user.set_password(form.password_field.data)
|
|
|
|
current_user.about = form.about.data
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
flash(_('Your changes have been saved.'), 'success')
|
|
|
|
return redirect(url_for('user.edit_profile', actor=actor))
|
|
|
|
elif request.method == 'GET':
|
|
|
|
form.email.data = current_user.email
|
|
|
|
form.about.data = current_user.about
|
|
|
|
form.password_field.data = ''
|
|
|
|
|
|
|
|
return render_template('user/edit_profile.html', title=_('Edit profile'), form=form, user=current_user)
|
|
|
|
|
|
|
|
|
|
|
|
@bp.route('/u/<actor>/settings', methods=['GET', 'POST'])
|
2023-10-23 13:03:35 +13:00
|
|
|
@login_required
|
2023-10-07 21:32:19 +13:00
|
|
|
def change_settings(actor):
|
|
|
|
actor = actor.strip()
|
|
|
|
user = User.query.filter_by(user_name=actor, deleted=False, banned=False, ap_id=None).first()
|
|
|
|
if user is None:
|
|
|
|
abort(404)
|
2023-10-23 20:18:46 +13:00
|
|
|
if current_user.id != user.id:
|
|
|
|
abort(401)
|
2023-10-07 21:32:19 +13:00
|
|
|
form = SettingsForm()
|
|
|
|
if form.validate_on_submit():
|
|
|
|
current_user.newsletter = form.newsletter.data
|
|
|
|
current_user.bot = form.bot.data
|
|
|
|
current_user.ignore_bots = form.ignore_bots.data
|
|
|
|
current_user.show_nsfw = form.nsfw.data
|
|
|
|
current_user.show_nsfl = form.nsfl.data
|
|
|
|
current_user.searchable = form.searchable.data
|
|
|
|
current_user.indexable = form.indexable.data
|
|
|
|
current_user.ap_manually_approves_followers = form.manually_approves_followers.data
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
flash(_('Your changes have been saved.'), 'success')
|
|
|
|
return redirect(url_for('user.change_settings', actor=actor))
|
|
|
|
elif request.method == 'GET':
|
|
|
|
form.newsletter.data = current_user.newsletter
|
|
|
|
form.bot.data = current_user.bot
|
|
|
|
form.ignore_bots.data = current_user.ignore_bots
|
|
|
|
form.nsfw.data = current_user.show_nsfw
|
|
|
|
form.nsfl.data = current_user.show_nsfl
|
|
|
|
form.searchable.data = current_user.searchable
|
|
|
|
form.indexable.data = current_user.indexable
|
|
|
|
form.manually_approves_followers.data = current_user.ap_manually_approves_followers
|
|
|
|
|
|
|
|
return render_template('user/edit_settings.html', title=_('Edit profile'), form=form, user=current_user)
|
2023-10-21 15:49:01 +13:00
|
|
|
|
|
|
|
|
|
|
|
@bp.route('/u/<actor>/ban', methods=['GET'])
|
2023-10-23 13:03:35 +13:00
|
|
|
@login_required
|
2023-10-21 15:49:01 +13:00
|
|
|
def ban_profile(actor):
|
|
|
|
if user_access('ban users', current_user.id):
|
|
|
|
actor = actor.strip()
|
|
|
|
user = User.query.filter_by(user_name=actor, deleted=False).first()
|
|
|
|
if user is None:
|
|
|
|
user = User.query.filter_by(ap_id=actor, deleted=False).first()
|
|
|
|
if user is None:
|
|
|
|
abort(404)
|
|
|
|
|
|
|
|
if user.id == current_user.id:
|
2023-11-30 23:21:37 +13:00
|
|
|
flash(_('You cannot ban yourself.'), 'error')
|
2023-10-21 15:49:01 +13:00
|
|
|
else:
|
|
|
|
user.banned = True
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
flash(f'{actor} has been banned.')
|
|
|
|
else:
|
|
|
|
abort(401)
|
|
|
|
|
2023-10-22 18:26:57 +13:00
|
|
|
goto = request.args.get('redirect') if 'redirect' in request.args else f'/u/{actor}'
|
|
|
|
return redirect(goto)
|
|
|
|
|
|
|
|
|
|
|
|
@bp.route('/u/<actor>/unban', methods=['GET'])
|
2023-10-23 13:03:35 +13:00
|
|
|
@login_required
|
2023-10-22 18:26:57 +13:00
|
|
|
def unban_profile(actor):
|
|
|
|
if user_access('ban users', current_user.id):
|
|
|
|
actor = actor.strip()
|
|
|
|
user = User.query.filter_by(user_name=actor, deleted=False).first()
|
|
|
|
if user is None:
|
|
|
|
user = User.query.filter_by(ap_id=actor, deleted=False).first()
|
|
|
|
if user is None:
|
|
|
|
abort(404)
|
|
|
|
|
|
|
|
if user.id == current_user.id:
|
2023-11-30 23:21:37 +13:00
|
|
|
flash(_('You cannot unban yourself.'), 'error')
|
2023-10-22 18:26:57 +13:00
|
|
|
else:
|
|
|
|
user.banned = False
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
flash(f'{actor} has been unbanned.')
|
|
|
|
else:
|
|
|
|
abort(401)
|
|
|
|
|
|
|
|
goto = request.args.get('redirect') if 'redirect' in request.args else f'/u/{actor}'
|
|
|
|
return redirect(goto)
|
2023-10-21 15:49:01 +13:00
|
|
|
|
|
|
|
|
|
|
|
@bp.route('/u/<actor>/delete', methods=['GET'])
|
2023-10-23 13:03:35 +13:00
|
|
|
@login_required
|
2023-10-21 15:49:01 +13:00
|
|
|
def delete_profile(actor):
|
|
|
|
if user_access('manage users', current_user.id):
|
|
|
|
actor = actor.strip()
|
|
|
|
user = User.query.filter_by(user_name=actor, deleted=False).first()
|
|
|
|
if user is None:
|
|
|
|
user = User.query.filter_by(ap_id=actor, deleted=False).first()
|
|
|
|
if user is None:
|
|
|
|
abort(404)
|
|
|
|
if user.id == current_user.id:
|
2023-11-30 23:21:37 +13:00
|
|
|
flash(_('You cannot delete yourself.'), 'error')
|
2023-10-21 15:49:01 +13:00
|
|
|
else:
|
|
|
|
user.banned = True
|
|
|
|
user.deleted = True
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
flash(f'{actor} has been deleted.')
|
|
|
|
else:
|
|
|
|
abort(401)
|
|
|
|
|
2023-10-22 18:26:57 +13:00
|
|
|
goto = request.args.get('redirect') if 'redirect' in request.args else f'/u/{actor}'
|
|
|
|
return redirect(goto)
|
2023-10-21 15:49:01 +13:00
|
|
|
|
|
|
|
|
|
|
|
@bp.route('/u/<actor>/ban_purge', methods=['GET'])
|
2023-10-23 13:03:35 +13:00
|
|
|
@login_required
|
2023-10-21 15:49:01 +13:00
|
|
|
def ban_purge_profile(actor):
|
|
|
|
if user_access('manage users', current_user.id):
|
|
|
|
actor = actor.strip()
|
|
|
|
user = User.query.filter_by(user_name=actor, deleted=False).first()
|
|
|
|
if user is None:
|
|
|
|
user = User.query.filter_by(ap_id=actor, deleted=False).first()
|
|
|
|
if user is None:
|
|
|
|
abort(404)
|
|
|
|
|
|
|
|
if user.id == current_user.id:
|
2023-11-30 23:21:37 +13:00
|
|
|
flash(_('You cannot purge yourself.'), 'error')
|
2023-10-21 15:49:01 +13:00
|
|
|
else:
|
|
|
|
user.banned = True
|
|
|
|
user.deleted = True
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
user.purge_content()
|
|
|
|
db.session.delete(user)
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
# todo: empty relevant caches
|
|
|
|
# todo: federate deletion
|
|
|
|
|
|
|
|
flash(f'{actor} has been banned, deleted and all their content deleted.')
|
|
|
|
else:
|
|
|
|
abort(401)
|
|
|
|
|
2023-10-22 18:26:57 +13:00
|
|
|
goto = request.args.get('redirect') if 'redirect' in request.args else f'/u/{actor}'
|
|
|
|
return redirect(goto)
|
2023-11-30 23:21:37 +13:00
|
|
|
|
|
|
|
|
|
|
|
@bp.route('/notifications', methods=['GET', 'POST'])
|
|
|
|
@login_required
|
|
|
|
def notifications():
|
|
|
|
"""Remove notifications older than 30 days"""
|
|
|
|
db.session.query(Notification).filter(
|
|
|
|
Notification.created_at < datetime.utcnow() - timedelta(days=30)).delete()
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
# Update unread notifications count
|
|
|
|
current_user.unread_notifications = Notification.query.filter_by(user_id=current_user.id, read=False).count()
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
notification_list = Notification.query.filter_by(user_id=current_user.id).order_by(desc(Notification.created_at)).all()
|
|
|
|
|
|
|
|
return render_template('user/notifications.html', title=_('Notifications'), notifications=notification_list, user=current_user)
|
|
|
|
|
|
|
|
|
|
|
|
@bp.route('/notification/<int:notification_id>/goto', methods=['GET', 'POST'])
|
|
|
|
@login_required
|
|
|
|
def notification_goto(notification_id):
|
|
|
|
notification = Notification.query.get_or_404(notification_id)
|
|
|
|
if notification.user_id == current_user.id:
|
|
|
|
if not notification.read:
|
|
|
|
current_user.unread_notifications -= 1
|
|
|
|
notification.read = True
|
|
|
|
db.session.commit()
|
|
|
|
return redirect(notification.url)
|
|
|
|
else:
|
|
|
|
abort(403)
|
|
|
|
|
|
|
|
|
|
|
|
@bp.route('/notification/<int:notification_id>/delete', methods=['GET', 'POST'])
|
|
|
|
@login_required
|
|
|
|
def notification_delete(notification_id):
|
|
|
|
notification = Notification.query.get_or_404(notification_id)
|
|
|
|
if notification.user_id == current_user.id:
|
|
|
|
if not notification.read:
|
|
|
|
current_user.unread_notifications -= 1
|
|
|
|
db.session.delete(notification)
|
|
|
|
db.session.commit()
|
|
|
|
return redirect(url_for('user.notifications'))
|
|
|
|
|
|
|
|
|
|
|
|
@bp.route('/notifications/all_read', methods=['GET', 'POST'])
|
|
|
|
@login_required
|
|
|
|
def notifications_all_read():
|
|
|
|
db.session.execute(text('UPDATE notification SET read=true WHERE user_id = :user_id'), {'user_id': current_user.id})
|
|
|
|
current_user.unread_notifications = 0
|
|
|
|
db.session.commit()
|
|
|
|
flash(_('All notifications marked as read.'))
|
|
|
|
return redirect(url_for('user.notifications'))
|