2022-12-23 18:08:33 +01:00
|
|
|
from flask import (
|
|
|
|
abort,
|
2024-12-16 10:07:21 +01:00
|
|
|
current_app,
|
|
|
|
Flask,
|
|
|
|
jsonify,
|
|
|
|
redirect,
|
|
|
|
render_template,
|
|
|
|
request,
|
2022-12-23 18:08:33 +01:00
|
|
|
send_from_directory,
|
|
|
|
url_for
|
|
|
|
)
|
2024-12-16 10:07:21 +01:00
|
|
|
from flask_login import current_user, login_required, logout_user
|
|
|
|
from threading import Thread
|
|
|
|
from app import db
|
|
|
|
from app.models import Avatar, User
|
2022-09-02 13:07:30 +02:00
|
|
|
from . import bp
|
2023-03-14 11:44:15 +01:00
|
|
|
|
2022-09-02 13:07:30 +02:00
|
|
|
|
2023-10-24 16:11:08 +02:00
|
|
|
@bp.route('')
|
2024-12-16 10:07:21 +01:00
|
|
|
@login_required
|
|
|
|
def index():
|
2023-10-24 16:11:08 +02:00
|
|
|
return redirect(url_for('main.social_area', _anchor='users'))
|
2023-03-14 11:44:15 +01:00
|
|
|
|
2022-09-02 13:07:30 +02:00
|
|
|
|
2023-10-24 16:11:08 +02:00
|
|
|
@bp.route('/<hashid:user_id>')
|
2024-12-16 10:07:21 +01:00
|
|
|
@login_required
|
|
|
|
def user(user_id: int):
|
2023-10-24 16:11:08 +02:00
|
|
|
user = User.query.get_or_404(user_id)
|
2024-12-16 10:07:21 +01:00
|
|
|
|
|
|
|
if not (
|
|
|
|
user.is_public
|
|
|
|
or user == current_user
|
|
|
|
or current_user.is_administrator
|
|
|
|
):
|
2023-10-24 16:11:08 +02:00
|
|
|
abort(403)
|
2024-12-16 10:07:21 +01:00
|
|
|
|
|
|
|
accept_json = request.accept_mimetypes.accept_json
|
|
|
|
accept_html = request.accept_mimetypes.accept_html
|
|
|
|
|
|
|
|
if accept_json and not accept_html:
|
|
|
|
return user.to_json_serializeable(
|
|
|
|
backrefs=True,
|
|
|
|
relationships=True
|
|
|
|
)
|
|
|
|
|
2023-10-24 16:11:08 +02:00
|
|
|
return render_template(
|
|
|
|
'users/user.html.j2',
|
|
|
|
title=user.username,
|
|
|
|
user=user
|
|
|
|
)
|
2022-09-02 13:07:30 +02:00
|
|
|
|
|
|
|
|
2024-12-16 10:07:21 +01:00
|
|
|
def _delete_user(app: Flask, user_id: int):
|
|
|
|
with app.app_context():
|
|
|
|
user = User.query.get(user_id)
|
|
|
|
user.delete()
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
|
|
|
|
@bp.route('/<hashid:user_id>', methods=['DELETE'])
|
|
|
|
@login_required
|
|
|
|
def delete_user(user_id: int):
|
|
|
|
user = User.query.get_or_404(user_id)
|
|
|
|
|
|
|
|
if not (
|
|
|
|
user == current_user
|
|
|
|
or current_user.is_administrator
|
|
|
|
):
|
|
|
|
abort(403)
|
|
|
|
|
|
|
|
if user == current_user:
|
|
|
|
logout_user()
|
|
|
|
|
|
|
|
thread = Thread(
|
|
|
|
target=_delete_user,
|
|
|
|
args=(current_app._get_current_object(), user.id)
|
|
|
|
)
|
|
|
|
thread.start()
|
|
|
|
|
|
|
|
return jsonify(f'User "{user.username}" marked for deletion'), 202
|
|
|
|
|
|
|
|
|
2023-10-24 16:11:08 +02:00
|
|
|
@bp.route('/<hashid:user_id>/avatar')
|
2024-12-16 10:07:21 +01:00
|
|
|
@login_required
|
|
|
|
def user_avatar(user_id: int):
|
2023-10-24 16:11:08 +02:00
|
|
|
user = User.query.get_or_404(user_id)
|
2024-12-16 10:07:21 +01:00
|
|
|
|
|
|
|
if not (
|
|
|
|
user.is_public
|
|
|
|
or user == current_user
|
|
|
|
or current_user.is_administrator
|
|
|
|
):
|
2023-10-24 16:11:08 +02:00
|
|
|
abort(403)
|
2024-12-16 10:07:21 +01:00
|
|
|
|
2023-10-24 16:11:08 +02:00
|
|
|
if user.avatar is None:
|
|
|
|
return redirect(url_for('static', filename='images/user_avatar.png'))
|
2024-12-16 10:07:21 +01:00
|
|
|
|
2023-10-24 16:11:08 +02:00
|
|
|
return send_from_directory(
|
2024-03-07 15:49:04 +01:00
|
|
|
user.avatar.path.parent,
|
|
|
|
user.avatar.path.name,
|
2023-10-24 16:11:08 +02:00
|
|
|
as_attachment=True,
|
2024-04-30 08:41:29 +02:00
|
|
|
download_name=user.avatar.filename,
|
2023-10-24 16:11:08 +02:00
|
|
|
mimetype=user.avatar.mimetype
|
|
|
|
)
|
2024-12-16 10:07:21 +01:00
|
|
|
|
|
|
|
|
|
|
|
def _delete_avatar(app: Flask, avatar_id: int):
|
|
|
|
with app.app_context():
|
|
|
|
avatar = Avatar.query.get(avatar_id)
|
|
|
|
avatar.delete()
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
|
|
|
|
@bp.route('/<hashid:user_id>/avatar', methods=['DELETE'])
|
|
|
|
@login_required
|
|
|
|
def delete_user_avatar(user_id: int):
|
|
|
|
user = User.query.get_or_404(user_id)
|
|
|
|
|
|
|
|
if user.avatar is None:
|
|
|
|
abort(409)
|
|
|
|
|
|
|
|
if not (
|
|
|
|
user == current_user
|
|
|
|
or current_user.is_administrator
|
|
|
|
):
|
|
|
|
abort(403)
|
|
|
|
|
|
|
|
thread = Thread(
|
|
|
|
target=_delete_avatar,
|
|
|
|
args=(current_app._get_current_object(), user.avatar.id)
|
|
|
|
)
|
|
|
|
thread.start()
|
|
|
|
|
|
|
|
return jsonify('Avatar marked for deletion'), 202
|
|
|
|
|
|
|
|
|
|
|
|
# TODO: Move this to main blueprint(?)
|
|
|
|
@bp.route('/accept-terms-of-use', methods=['POST'])
|
|
|
|
@login_required
|
|
|
|
def accept_terms_of_use():
|
|
|
|
if not (
|
|
|
|
current_user.is_authenticated
|
|
|
|
or current_user.confirmed
|
|
|
|
):
|
|
|
|
abort(403)
|
|
|
|
|
|
|
|
current_user.terms_of_use_accepted = True
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
return jsonify('You accepted the terms of use'), 202
|