mirror of
https://github.com/aljazceru/CTFd.git
synced 2025-12-17 14:04:20 +01:00
2.3.0 / 2020-02-17
==================
**General**
* During setup, admins can register their email address with the CTFd LLC newsletter for news and updates
* Fix editting hints from the admin panel
* Allow admins to insert HTML code directly into the header and footer (end of body tag) of pages. This replaces and supercedes the custom CSS feature.
* The `views.custom_css` route has been removed.
* Admins can now customize the content of outgoing emails and inject certain variables into email content.
* The `manage.py` script can now manipulate the CTFd Configs table via the `get_config` and `set_config` commands. (e.g. `python manage.py get_config ctf_theme` and `python manage.py set_config ctf_theme core`)
**Themes**
* Themes should now reference the `theme_header` and `theme_footer` configs instead of the `views.custom_css` endpoint to allow for user customizations. See the `base.html` file of the core theme.
**Plugins**
* Make `ezq` functions available to `CTFd.js` under `CTFd.ui.ezq`
**Miscellaneous**
* Python imports sorted with `isort` and import order enforced
* Black formatter running on a majority of Python code
63 lines
1.6 KiB
Python
63 lines
1.6 KiB
Python
from flask import Blueprint, render_template, request
|
|
|
|
from CTFd.models import Users
|
|
from CTFd.utils import config
|
|
from CTFd.utils.decorators import authed_only
|
|
from CTFd.utils.decorators.visibility import (
|
|
check_account_visibility,
|
|
check_score_visibility,
|
|
)
|
|
from CTFd.utils.user import get_current_user
|
|
|
|
users = Blueprint("users", __name__)
|
|
|
|
|
|
@users.route("/users")
|
|
@check_account_visibility
|
|
def listing():
|
|
page = abs(request.args.get("page", 1, type=int))
|
|
results_per_page = 50
|
|
page_start = results_per_page * (page - 1)
|
|
page_end = results_per_page * (page - 1) + results_per_page
|
|
|
|
count = Users.query.filter_by(banned=False, hidden=False).count()
|
|
users = (
|
|
Users.query.filter_by(banned=False, hidden=False)
|
|
.slice(page_start, page_end)
|
|
.all()
|
|
)
|
|
|
|
pages = int(count / results_per_page) + (count % results_per_page > 0)
|
|
return render_template("users/users.html", users=users, pages=pages, curr_page=page)
|
|
|
|
|
|
@users.route("/profile")
|
|
@users.route("/user")
|
|
@authed_only
|
|
def private():
|
|
user = get_current_user()
|
|
|
|
solves = user.get_solves()
|
|
awards = user.get_awards()
|
|
|
|
place = user.place
|
|
score = user.score
|
|
|
|
return render_template(
|
|
"users/private.html",
|
|
solves=solves,
|
|
awards=awards,
|
|
user=user,
|
|
score=score,
|
|
place=place,
|
|
score_frozen=config.is_scoreboard_frozen(),
|
|
)
|
|
|
|
|
|
@users.route("/users/<int:user_id>")
|
|
@check_account_visibility
|
|
@check_score_visibility
|
|
def public(user_id):
|
|
user = Users.query.filter_by(id=user_id, banned=False, hidden=False).first_or_404()
|
|
return render_template("users/public.html", user=user)
|