mirror of
https://github.com/aljazceru/CTFd.git
synced 2025-12-17 05:54:19 +01:00
Upgrading manage.py to work with Flask 2.0 (#2361)
* Fix manage.py to work with Flask CLI * Support both the legacy `manage.py` CLI as well as the new `flask` CLI * Closes #2354 * Closes #2352
This commit is contained in:
@@ -19,6 +19,7 @@ from CTFd.constants.themes import ADMIN_THEME, DEFAULT_THEME
|
|||||||
from CTFd.plugins import init_plugins
|
from CTFd.plugins import init_plugins
|
||||||
from CTFd.utils.crypto import sha256
|
from CTFd.utils.crypto import sha256
|
||||||
from CTFd.utils.initialization import (
|
from CTFd.utils.initialization import (
|
||||||
|
init_cli,
|
||||||
init_events,
|
init_events,
|
||||||
init_logs,
|
init_logs,
|
||||||
init_request_processors,
|
init_request_processors,
|
||||||
@@ -312,5 +313,6 @@ def create_app(config="CTFd.config.Config"):
|
|||||||
init_logs(app)
|
init_logs(app)
|
||||||
init_events(app)
|
init_events(app)
|
||||||
init_plugins(app)
|
init_plugins(app)
|
||||||
|
init_cli(app)
|
||||||
|
|
||||||
return app
|
return app
|
||||||
|
|||||||
92
CTFd/cli/__init__.py
Normal file
92
CTFd/cli/__init__.py
Normal file
@@ -0,0 +1,92 @@
|
|||||||
|
import datetime
|
||||||
|
import shutil
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
import click
|
||||||
|
from flask import Blueprint, current_app
|
||||||
|
|
||||||
|
from CTFd.utils import get_config as get_config_util
|
||||||
|
from CTFd.utils import set_config as set_config_util
|
||||||
|
from CTFd.utils.config import ctf_name
|
||||||
|
from CTFd.utils.exports import export_ctf as export_ctf_util
|
||||||
|
from CTFd.utils.exports import import_ctf as import_ctf_util
|
||||||
|
from CTFd.utils.exports import set_import_end_time, set_import_error
|
||||||
|
|
||||||
|
_cli = Blueprint("cli", __name__)
|
||||||
|
|
||||||
|
|
||||||
|
def jsenums():
|
||||||
|
import json
|
||||||
|
import os
|
||||||
|
|
||||||
|
from CTFd.constants import JS_ENUMS
|
||||||
|
|
||||||
|
path = os.path.join(current_app.root_path, "themes/core/assets/js/constants.js")
|
||||||
|
|
||||||
|
with open(path, "w+") as f:
|
||||||
|
for k, v in JS_ENUMS.items():
|
||||||
|
f.write("const {} = Object.freeze({});".format(k, json.dumps(v)))
|
||||||
|
|
||||||
|
|
||||||
|
BUILD_COMMANDS = {"jsenums": jsenums}
|
||||||
|
|
||||||
|
|
||||||
|
@_cli.cli.command("get_config")
|
||||||
|
@click.argument("key")
|
||||||
|
def get_config(key):
|
||||||
|
print(get_config_util(key))
|
||||||
|
|
||||||
|
|
||||||
|
@_cli.cli.command("set_config")
|
||||||
|
@click.argument("key")
|
||||||
|
@click.argument("value")
|
||||||
|
def set_config(key, value):
|
||||||
|
print(set_config_util(key, value).value)
|
||||||
|
|
||||||
|
|
||||||
|
@_cli.cli.command("build")
|
||||||
|
@click.argument("cmd")
|
||||||
|
def build(cmd):
|
||||||
|
cmd = BUILD_COMMANDS.get(cmd)
|
||||||
|
cmd()
|
||||||
|
|
||||||
|
|
||||||
|
@_cli.cli.command("export_ctf")
|
||||||
|
@click.argument("path", default="")
|
||||||
|
def export_ctf(path):
|
||||||
|
backup = export_ctf_util()
|
||||||
|
|
||||||
|
if path:
|
||||||
|
with open(path, "wb") as target:
|
||||||
|
shutil.copyfileobj(backup, target)
|
||||||
|
else:
|
||||||
|
name = ctf_name()
|
||||||
|
day = datetime.datetime.now().strftime("%Y-%m-%d_%T")
|
||||||
|
full_name = f"{name}.{day}.zip"
|
||||||
|
|
||||||
|
with open(full_name, "wb") as target:
|
||||||
|
shutil.copyfileobj(backup, target)
|
||||||
|
|
||||||
|
print(f"Exported {full_name}")
|
||||||
|
|
||||||
|
|
||||||
|
@_cli.cli.command("import_ctf")
|
||||||
|
@click.argument("path", type=click.Path(exists=True))
|
||||||
|
@click.option(
|
||||||
|
"--delete_import_on_finish",
|
||||||
|
default=False,
|
||||||
|
is_flag=True,
|
||||||
|
help="Delete import file when import is finished",
|
||||||
|
)
|
||||||
|
def import_ctf(path, delete_import_on_finish=False):
|
||||||
|
try:
|
||||||
|
import_ctf_util(path)
|
||||||
|
except Exception as e:
|
||||||
|
from CTFd.utils.dates import unix_time
|
||||||
|
|
||||||
|
set_import_error("Import Failure: " + str(e))
|
||||||
|
set_import_end_time(value=unix_time(datetime.datetime.utcnow()))
|
||||||
|
|
||||||
|
if delete_import_on_finish:
|
||||||
|
print(f"Deleting {path}")
|
||||||
|
Path(path).unlink()
|
||||||
@@ -43,6 +43,12 @@ from CTFd.utils.user import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def init_cli(app):
|
||||||
|
from CTFd.cli import _cli
|
||||||
|
|
||||||
|
app.register_blueprint(_cli, cli_group=None)
|
||||||
|
|
||||||
|
|
||||||
def init_template_filters(app):
|
def init_template_filters(app):
|
||||||
app.jinja_env.filters["markdown"] = markdown
|
app.jinja_env.filters["markdown"] = markdown
|
||||||
app.jinja_env.filters["unix_time"] = unix_time
|
app.jinja_env.filters["unix_time"] = unix_time
|
||||||
|
|||||||
90
manage.py
90
manage.py
@@ -1,95 +1,11 @@
|
|||||||
import datetime
|
from flask.cli import FlaskGroup
|
||||||
import shutil
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
from flask_migrate import MigrateCommand
|
|
||||||
from flask_script import Manager
|
|
||||||
|
|
||||||
from CTFd import create_app
|
from CTFd import create_app
|
||||||
from CTFd.utils import get_config as get_config_util
|
|
||||||
from CTFd.utils import set_config as set_config_util
|
|
||||||
from CTFd.utils.config import ctf_name
|
|
||||||
from CTFd.utils.exports import export_ctf as export_ctf_util
|
|
||||||
from CTFd.utils.exports import import_ctf as import_ctf_util
|
|
||||||
from CTFd.utils.exports import (
|
|
||||||
set_import_end_time,
|
|
||||||
set_import_error,
|
|
||||||
)
|
|
||||||
|
|
||||||
app = create_app()
|
app = create_app()
|
||||||
|
|
||||||
manager = Manager(app)
|
cli = FlaskGroup(app)
|
||||||
manager.add_command("db", MigrateCommand)
|
|
||||||
|
|
||||||
|
|
||||||
def jsenums():
|
|
||||||
from CTFd.constants import JS_ENUMS
|
|
||||||
import json
|
|
||||||
import os
|
|
||||||
|
|
||||||
path = os.path.join(app.root_path, "themes/core/assets/js/constants.js")
|
|
||||||
|
|
||||||
with open(path, "w+") as f:
|
|
||||||
for k, v in JS_ENUMS.items():
|
|
||||||
f.write("const {} = Object.freeze({});".format(k, json.dumps(v)))
|
|
||||||
|
|
||||||
|
|
||||||
BUILD_COMMANDS = {"jsenums": jsenums}
|
|
||||||
|
|
||||||
|
|
||||||
@manager.command
|
|
||||||
def get_config(key):
|
|
||||||
with app.app_context():
|
|
||||||
print(get_config_util(key))
|
|
||||||
|
|
||||||
|
|
||||||
@manager.command
|
|
||||||
def set_config(key, value):
|
|
||||||
with app.app_context():
|
|
||||||
print(set_config_util(key, value).value)
|
|
||||||
|
|
||||||
|
|
||||||
@manager.command
|
|
||||||
def build(cmd):
|
|
||||||
with app.app_context():
|
|
||||||
cmd = BUILD_COMMANDS.get(cmd)
|
|
||||||
cmd()
|
|
||||||
|
|
||||||
|
|
||||||
@manager.command
|
|
||||||
def export_ctf(path=None):
|
|
||||||
with app.app_context():
|
|
||||||
backup = export_ctf_util()
|
|
||||||
|
|
||||||
if path:
|
|
||||||
with open(path, "wb") as target:
|
|
||||||
shutil.copyfileobj(backup, target)
|
|
||||||
else:
|
|
||||||
name = ctf_name()
|
|
||||||
day = datetime.datetime.now().strftime("%Y-%m-%d_%T")
|
|
||||||
full_name = f"{name}.{day}.zip"
|
|
||||||
|
|
||||||
with open(full_name, "wb") as target:
|
|
||||||
shutil.copyfileobj(backup, target)
|
|
||||||
|
|
||||||
print(f"Exported {full_name}")
|
|
||||||
|
|
||||||
|
|
||||||
@manager.command
|
|
||||||
def import_ctf(path, delete_import_on_finish=False):
|
|
||||||
with app.app_context():
|
|
||||||
try:
|
|
||||||
import_ctf_util(path)
|
|
||||||
except Exception as e:
|
|
||||||
from CTFd.utils.dates import unix_time
|
|
||||||
|
|
||||||
set_import_error(f"Import Failure: " + str(e))
|
|
||||||
set_import_end_time(value=unix_time(datetime.datetime.utcnow()))
|
|
||||||
|
|
||||||
if delete_import_on_finish:
|
|
||||||
print(f"Deleting {path}")
|
|
||||||
Path(path).unlink()
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
manager.run()
|
cli()
|
||||||
|
|||||||
Reference in New Issue
Block a user