2023-04-24 16:10:16 +03:00
|
|
|
from flask import Blueprint, render_template, request, flash, redirect, url_for, jsonify
|
2023-05-11 09:51:50 +03:00
|
|
|
from flask_login import login_required, current_user, logout_user
|
2023-04-20 16:10:16 +03:00
|
|
|
from app.controllers import create_pagination
|
2023-06-06 16:36:48 +03:00
|
|
|
from sqlalchemy import func, not_, or_
|
2023-04-20 16:10:16 +03:00
|
|
|
|
|
|
|
from app import models as m, db
|
|
|
|
from app import forms as f
|
|
|
|
from app.logger import log
|
2023-04-24 16:10:16 +03:00
|
|
|
from config import config
|
2023-04-20 16:10:16 +03:00
|
|
|
|
2023-04-24 16:10:16 +03:00
|
|
|
configuration = config()
|
2023-04-20 16:10:16 +03:00
|
|
|
bp = Blueprint("user", __name__, url_prefix="/user")
|
|
|
|
|
|
|
|
|
|
|
|
@bp.route("/", methods=["GET"])
|
|
|
|
@login_required
|
|
|
|
def get_all():
|
|
|
|
q = request.args.get("q", type=str, default=None)
|
|
|
|
users = m.User.query.order_by(m.User.id)
|
|
|
|
if q:
|
2023-04-20 17:37:38 +03:00
|
|
|
users = users.filter(m.User.username.like(f"{q}%"))
|
2023-04-20 16:10:16 +03:00
|
|
|
|
|
|
|
pagination = create_pagination(total=users.count())
|
|
|
|
|
|
|
|
return render_template(
|
|
|
|
"user/users.html",
|
|
|
|
users=users.paginate(page=pagination.page, per_page=pagination.per_page),
|
|
|
|
page=pagination,
|
|
|
|
search_query=q,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2023-05-12 17:05:30 +03:00
|
|
|
@bp.route("/edit_profile", methods=["GET", "POST"])
|
2023-05-04 14:18:49 +03:00
|
|
|
@login_required
|
2023-05-12 17:05:30 +03:00
|
|
|
def edit_profile():
|
2023-05-04 14:18:49 +03:00
|
|
|
form = f.EditUserForm()
|
|
|
|
if form.validate_on_submit():
|
|
|
|
user: m.User = current_user
|
2023-06-07 16:10:43 +03:00
|
|
|
user.username = form.username.data
|
2023-05-04 14:18:49 +03:00
|
|
|
if form.avatar_img.data:
|
2023-05-26 12:40:23 +03:00
|
|
|
current_user.avatar_img = (
|
|
|
|
form.avatar_img.data
|
|
|
|
) # form.avatar_img.data is changed in form validator
|
2023-05-04 14:18:49 +03:00
|
|
|
user.is_activated = True
|
|
|
|
user.save()
|
|
|
|
return redirect(url_for("main.index"))
|
2023-05-15 12:14:49 +03:00
|
|
|
elif form.is_submitted():
|
2023-05-04 14:18:49 +03:00
|
|
|
log(log.ERROR, "Update user errors: [%s]", form.errors)
|
|
|
|
for field, errors in form.errors.items():
|
|
|
|
field_label = form._fields[field].label.text
|
|
|
|
for error in errors:
|
|
|
|
flash(error.replace("Field", field_label), "danger")
|
|
|
|
|
|
|
|
if current_user.is_activated:
|
2023-06-07 16:10:43 +03:00
|
|
|
form.username.data = current_user.username
|
2023-05-12 17:05:30 +03:00
|
|
|
return render_template("user/edit_profile.html", form=form)
|
2023-05-04 14:18:49 +03:00
|
|
|
|
|
|
|
|
2023-05-26 12:40:23 +03:00
|
|
|
@bp.route("/delete_avatar", methods=["POST"])
|
|
|
|
@login_required
|
|
|
|
def delete_avatar():
|
|
|
|
user: m.User = current_user
|
|
|
|
current_user.avatar_img = None
|
|
|
|
log(log.ERROR, "Delete user [%s] avatar", user)
|
|
|
|
current_user.save()
|
|
|
|
|
|
|
|
return redirect(url_for("user.edit_profile"))
|
|
|
|
|
|
|
|
|
2023-05-12 17:05:30 +03:00
|
|
|
@bp.route("/<int:user_id>/profile")
|
|
|
|
def profile(user_id: int):
|
2023-05-15 11:47:37 +03:00
|
|
|
user: m.User = db.session.get(m.User, user_id)
|
2023-05-12 17:05:30 +03:00
|
|
|
interpretations: m.Interpretation = m.Interpretation.query.filter_by(
|
|
|
|
user_id=user_id
|
|
|
|
)
|
|
|
|
if not user:
|
|
|
|
log(log.ERROR, "Not found user by id : [%s]", user_id)
|
|
|
|
flash("Cannot find user data", "danger")
|
|
|
|
return render_template(
|
|
|
|
"user/profile.html", user=user, interpretations=interpretations
|
|
|
|
)
|
2023-04-20 16:10:16 +03:00
|
|
|
|
|
|
|
|
2023-05-11 09:51:50 +03:00
|
|
|
@bp.route("/profile_delete", methods=["POST"])
|
2023-04-20 16:10:16 +03:00
|
|
|
@login_required
|
2023-05-11 09:51:50 +03:00
|
|
|
def profile_delete():
|
|
|
|
user: m.User = db.session.get(m.User, current_user.id)
|
|
|
|
user.is_deleted = True
|
|
|
|
log(log.INFO, "User deleted. User: [%s]", user)
|
2023-05-12 12:00:54 +03:00
|
|
|
user.save()
|
|
|
|
logout_user()
|
2023-04-20 16:10:16 +03:00
|
|
|
flash("User deleted!", "success")
|
2023-05-11 09:51:50 +03:00
|
|
|
return redirect(url_for("home.get_all"))
|
2023-04-24 16:10:16 +03:00
|
|
|
|
|
|
|
|
2023-05-15 12:14:49 +03:00
|
|
|
@bp.route("/profile_reactivate", methods=["GET", "POST"])
|
|
|
|
def profile_reactivate():
|
|
|
|
user: m.User = db.session.get(m.User, current_user.id)
|
|
|
|
if not user:
|
|
|
|
log(log.CRITICAL, "No such user. User: [%s]", user)
|
|
|
|
return redirect(url_for("home.get_all"))
|
|
|
|
form = f.ReactivateUserForm()
|
|
|
|
if form.validate_on_submit():
|
|
|
|
user.is_deleted = False
|
|
|
|
log(log.INFO, "Form submitted. User reactivated: [%s]", user)
|
|
|
|
flash("User reactivated!", "success")
|
|
|
|
user.save()
|
|
|
|
return redirect(url_for("home.get_all"))
|
|
|
|
return render_template("user/reactivate.html", form=form)
|
|
|
|
|
|
|
|
|
2023-04-24 16:10:16 +03:00
|
|
|
@bp.route("/search", methods=["GET"])
|
|
|
|
@login_required
|
|
|
|
def search():
|
2023-06-06 16:36:48 +03:00
|
|
|
q = request.args.get("q", type=str, default="").lower()
|
2023-04-24 16:10:16 +03:00
|
|
|
if not q:
|
|
|
|
return jsonify({"message": "q parameter is required"}), 422
|
|
|
|
|
2023-04-25 11:02:22 +03:00
|
|
|
book_id = request.args.get("book_id", type=str, default=None)
|
|
|
|
|
|
|
|
query_user = m.User.query
|
|
|
|
query_user = query_user.order_by(m.User.username)
|
|
|
|
|
2023-06-06 16:36:48 +03:00
|
|
|
query_user = query_user.filter(
|
|
|
|
or_(
|
|
|
|
func.lower(m.User.username).like(f"%{q}%"),
|
|
|
|
func.lower(m.User.wallet_id).like(f"%{q}%"),
|
|
|
|
)
|
|
|
|
)
|
2023-04-25 11:02:22 +03:00
|
|
|
if book_id:
|
|
|
|
book_contributors = m.BookContributor.query.filter_by(book_id=book_id).all()
|
|
|
|
user_ids = [contributor.user_id for contributor in book_contributors]
|
2023-04-26 13:05:23 +03:00
|
|
|
user_ids.append(current_user.id)
|
2023-04-25 11:02:22 +03:00
|
|
|
query_user = query_user.filter(not_(m.User.id.in_(user_ids)))
|
|
|
|
query_user = query_user.limit(configuration.MAX_SEARCH_RESULTS)
|
|
|
|
|
|
|
|
users = [{"username": user.username, "id": user.id} for user in query_user.all()]
|
2023-04-24 16:10:16 +03:00
|
|
|
|
|
|
|
return jsonify({"users": users})
|