2022-01-16 18:22:00 +00:00
|
|
|
import os
|
2022-04-02 21:33:23 +00:00
|
|
|
from flask import Flask, url_for, g, redirect
|
2022-01-16 18:22:00 +00:00
|
|
|
from functools import wraps
|
|
|
|
from flask_assets import Environment
|
|
|
|
from webassets import Bundle
|
|
|
|
import time
|
|
|
|
from app.models import db, migrate, PlayKey
|
|
|
|
from app.schemas import ma
|
|
|
|
from app.forms import CustomUserManager
|
2022-02-12 02:31:57 +00:00
|
|
|
from flask_user import user_registered, current_user, user_logged_in
|
2022-01-16 18:22:00 +00:00
|
|
|
from flask_wtf.csrf import CSRFProtect
|
|
|
|
from flask_apscheduler import APScheduler
|
2022-03-13 02:09:35 +00:00
|
|
|
from app.luclient import register_luclient_jinja_helpers
|
2022-12-17 07:47:10 +00:00
|
|
|
import pathlib
|
2022-01-16 18:22:00 +00:00
|
|
|
|
2022-04-02 21:33:23 +00:00
|
|
|
from app.commands import (
|
|
|
|
init_db,
|
|
|
|
init_accounts,
|
|
|
|
load_property,
|
|
|
|
gen_image_cache,
|
|
|
|
gen_model_cache,
|
|
|
|
fix_clone_ids
|
|
|
|
)
|
2022-02-12 05:05:00 +00:00
|
|
|
from app.models import Account, AccountInvitation, AuditLog
|
2022-01-16 18:22:00 +00:00
|
|
|
|
2022-02-12 02:31:57 +00:00
|
|
|
import logging
|
|
|
|
from logging.handlers import RotatingFileHandler
|
|
|
|
|
|
|
|
from werkzeug.exceptions import HTTPException
|
|
|
|
|
2022-01-16 18:22:00 +00:00
|
|
|
# Instantiate Flask extensions
|
|
|
|
csrf_protect = CSRFProtect()
|
|
|
|
scheduler = APScheduler()
|
|
|
|
# db and migrate is instantiated in models.py
|
|
|
|
|
|
|
|
|
|
|
|
def create_app():
|
|
|
|
app = Flask(__name__, instance_relative_config=True)
|
|
|
|
|
|
|
|
# decrement uses on a play key after a successful registration
|
|
|
|
# and increment the times it has been used
|
|
|
|
@user_registered.connect_via(app)
|
|
|
|
def after_register_hook(sender, user, **extra):
|
|
|
|
if app.config["REQUIRE_PLAY_KEY"]:
|
|
|
|
play_key_used = PlayKey.query.filter(PlayKey.id == user.play_key_id).first()
|
|
|
|
play_key_used.key_uses = play_key_used.key_uses - 1
|
|
|
|
play_key_used.times_used = play_key_used.times_used + 1
|
2022-02-12 02:31:57 +00:00
|
|
|
app.logger.info(
|
|
|
|
f"USERS::REGISTRATION User with ID {user.id} and name {user.username} Registered \
|
|
|
|
using Play Key ID {play_key_used.id} : {play_key_used.key_string}"
|
2022-03-13 02:09:35 +00:00
|
|
|
)
|
2022-01-16 18:22:00 +00:00
|
|
|
db.session.add(play_key_used)
|
|
|
|
db.session.commit()
|
2022-02-12 02:31:57 +00:00
|
|
|
else:
|
|
|
|
app.logger.info(f"USERS::REGISTRATION User with ID {user.id} and name {user.username} Registered")
|
|
|
|
|
|
|
|
@user_logged_in.connect_via(app)
|
|
|
|
def _after_login_hook(sender, user, **extra):
|
|
|
|
app.logger.info(f"{user.username} Logged in")
|
2022-01-16 18:22:00 +00:00
|
|
|
|
|
|
|
# A bunch of jinja filters to make things easiers
|
|
|
|
@app.template_filter('ctime')
|
|
|
|
def timectime(s):
|
|
|
|
if s:
|
2022-03-13 02:09:35 +00:00
|
|
|
return time.ctime(s) # or datetime.datetime.fromtimestamp(s)
|
2022-01-16 18:22:00 +00:00
|
|
|
else:
|
|
|
|
return "Never"
|
|
|
|
|
|
|
|
@app.template_filter('check_perm_map')
|
|
|
|
def check_perm_map(perm_map, bit):
|
|
|
|
if perm_map:
|
|
|
|
return perm_map & (1 << bit)
|
|
|
|
else:
|
|
|
|
return 0 & (1 << bit)
|
|
|
|
|
2022-02-12 02:31:57 +00:00
|
|
|
@app.template_filter('debug')
|
|
|
|
def debug(text):
|
|
|
|
print(text)
|
|
|
|
|
2022-01-16 18:22:00 +00:00
|
|
|
@app.teardown_appcontext
|
|
|
|
def close_connection(exception):
|
|
|
|
cdclient = getattr(g, '_cdclient', None)
|
|
|
|
if cdclient is not None:
|
|
|
|
cdclient.close()
|
|
|
|
|
2022-12-17 06:59:47 +00:00
|
|
|
@app.template_filter()
|
|
|
|
def numberFormat(value):
|
|
|
|
return format(int(value), ',d')
|
|
|
|
|
|
|
|
|
2022-01-16 18:22:00 +00:00
|
|
|
# add the commands to flask cli
|
|
|
|
app.cli.add_command(init_db)
|
|
|
|
app.cli.add_command(init_accounts)
|
2022-01-27 05:11:49 +00:00
|
|
|
app.cli.add_command(load_property)
|
2022-02-10 18:01:51 +00:00
|
|
|
app.cli.add_command(gen_image_cache)
|
|
|
|
app.cli.add_command(gen_model_cache)
|
2022-04-02 21:33:23 +00:00
|
|
|
app.cli.add_command(fix_clone_ids)
|
2022-01-16 18:22:00 +00:00
|
|
|
|
2022-02-12 02:31:57 +00:00
|
|
|
register_logging(app)
|
2022-01-16 18:22:00 +00:00
|
|
|
register_settings(app)
|
|
|
|
register_extensions(app)
|
|
|
|
register_blueprints(app)
|
|
|
|
register_luclient_jinja_helpers(app)
|
|
|
|
|
2022-12-17 07:47:10 +00:00
|
|
|
# Extract the brickdb if it's not already extracted
|
|
|
|
materials = pathlib.Path(f'{app.config["CACHE_LOCATION"]}Materials.xml')
|
|
|
|
if not materials.is_file():
|
2022-12-17 07:55:55 +00:00
|
|
|
# unzip the brickdb, and remove the import after
|
2022-12-17 07:47:10 +00:00
|
|
|
from zipfile import ZipFile
|
|
|
|
with ZipFile(f"{app.config['CLIENT_LOCATION']}res/brickdb.zip","r") as zip_ref:
|
|
|
|
zip_ref.extractall(app.config["CACHE_LOCATION"])
|
2022-12-17 07:55:55 +00:00
|
|
|
del ZipFile
|
|
|
|
# copy over the brick primitives, and remove the import after
|
2022-12-17 07:47:10 +00:00
|
|
|
from shutil import copytree
|
|
|
|
copytree(f"{app.config['CLIENT_LOCATION']}res/brickprimitives", f"{app.config['CACHE_LOCATION']}brickprimitives")
|
2022-12-17 07:55:55 +00:00
|
|
|
del copytree
|
2022-12-17 07:47:10 +00:00
|
|
|
|
2022-01-16 18:22:00 +00:00
|
|
|
return app
|
|
|
|
|
|
|
|
|
|
|
|
def register_extensions(app):
|
|
|
|
"""Register extensions for Flask app
|
|
|
|
|
|
|
|
Args:
|
|
|
|
app (Flask): Flask app to register for
|
|
|
|
"""
|
|
|
|
db.init_app(app)
|
|
|
|
migrate.init_app(app, db)
|
|
|
|
ma.init_app(app)
|
|
|
|
|
|
|
|
scheduler.init_app(app)
|
|
|
|
scheduler.start()
|
|
|
|
|
|
|
|
csrf_protect.init_app(app)
|
|
|
|
|
|
|
|
user_manager = CustomUserManager(
|
|
|
|
app, db, Account, UserInvitationClass=AccountInvitation
|
|
|
|
)
|
|
|
|
|
|
|
|
assets = Environment(app)
|
|
|
|
assets.url = app.static_url_path
|
2022-12-17 04:57:05 +00:00
|
|
|
scss = Bundle('scss/site.scss', filters='libsass', output='css/site.css')
|
2022-01-16 18:22:00 +00:00
|
|
|
assets.register('scss_all', scss)
|
|
|
|
|
|
|
|
|
|
|
|
def register_blueprints(app):
|
|
|
|
"""Register blueprints for Flask app
|
|
|
|
|
|
|
|
Args:
|
|
|
|
app (Flask): Flask app to register for
|
|
|
|
"""
|
|
|
|
|
|
|
|
from .main import main_blueprint
|
|
|
|
app.register_blueprint(main_blueprint)
|
|
|
|
from .play_keys import play_keys_blueprint
|
|
|
|
app.register_blueprint(play_keys_blueprint, url_prefix='/play_keys')
|
|
|
|
from .accounts import accounts_blueprint
|
|
|
|
app.register_blueprint(accounts_blueprint, url_prefix='/accounts')
|
|
|
|
from .characters import character_blueprint
|
|
|
|
app.register_blueprint(character_blueprint, url_prefix='/characters')
|
|
|
|
from .properties import property_blueprint
|
|
|
|
app.register_blueprint(property_blueprint, url_prefix='/properties')
|
|
|
|
from .moderation import moderation_blueprint
|
|
|
|
app.register_blueprint(moderation_blueprint, url_prefix='/moderation')
|
|
|
|
from .log import log_blueprint
|
|
|
|
app.register_blueprint(log_blueprint, url_prefix='/log')
|
|
|
|
from .bug_reports import bug_report_blueprint
|
|
|
|
app.register_blueprint(bug_report_blueprint, url_prefix='/bug_reports')
|
|
|
|
from .mail import mail_blueprint
|
|
|
|
app.register_blueprint(mail_blueprint, url_prefix='/mail')
|
|
|
|
from .luclient import luclient_blueprint
|
|
|
|
app.register_blueprint(luclient_blueprint, url_prefix='/luclient')
|
|
|
|
from .reports import reports_blueprint
|
|
|
|
app.register_blueprint(reports_blueprint, url_prefix='/reports')
|
2022-03-28 16:01:46 +00:00
|
|
|
from .api import api_blueprint
|
|
|
|
app.register_blueprint(api_blueprint, url_prefix='/api')
|
2022-01-16 18:22:00 +00:00
|
|
|
|
|
|
|
|
2022-02-12 02:31:57 +00:00
|
|
|
def register_logging(app):
|
|
|
|
# file logger
|
2022-12-17 04:50:43 +00:00
|
|
|
file_handler = RotatingFileHandler('logs/nexus_dashboard.log', maxBytes=1024 * 1024 * 100, backupCount=20)
|
2022-02-12 02:31:57 +00:00
|
|
|
file_handler.setLevel(logging.INFO)
|
|
|
|
formatter = logging.Formatter("%(asctime)s - %(name)s - %(levelname)s - %(message)s")
|
|
|
|
file_handler.setFormatter(formatter)
|
|
|
|
app.logger.addHandler(file_handler)
|
|
|
|
|
2022-01-16 18:22:00 +00:00
|
|
|
|
|
|
|
def register_settings(app):
|
|
|
|
"""Register setting from setting and env
|
|
|
|
|
|
|
|
Args:
|
|
|
|
app (Flask): Flask app to register for
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Load common settings
|
2022-03-13 02:38:24 +00:00
|
|
|
try:
|
|
|
|
app.config.from_object('app.settings')
|
|
|
|
except Exception:
|
2022-03-14 00:27:23 +00:00
|
|
|
app.logger.info("No settings.py, loading from example")
|
|
|
|
app.config.from_object('app.settings_example')
|
2022-01-16 18:22:00 +00:00
|
|
|
|
|
|
|
# Load environment specific settings
|
|
|
|
app.config['TESTING'] = False
|
|
|
|
app.config['DEBUG'] = False
|
|
|
|
|
|
|
|
# always pull these two from the env
|
2022-03-13 02:38:24 +00:00
|
|
|
app.config['SECRET_KEY'] = os.getenv(
|
|
|
|
'APP_SECRET_KEY',
|
|
|
|
app.config['APP_SECRET_KEY']
|
|
|
|
|
|
|
|
)
|
|
|
|
app.config['SQLALCHEMY_DATABASE_URI'] = os.getenv(
|
|
|
|
'APP_DATABASE_URI',
|
|
|
|
app.config['APP_DATABASE_URI']
|
|
|
|
)
|
2022-03-14 23:47:19 +00:00
|
|
|
|
2022-01-16 18:22:00 +00:00
|
|
|
# try to get overides, otherwise just use what we have already
|
|
|
|
app.config['USER_ENABLE_REGISTER'] = os.getenv(
|
|
|
|
'USER_ENABLE_REGISTER',
|
|
|
|
app.config['USER_ENABLE_REGISTER']
|
|
|
|
)
|
|
|
|
app.config['USER_ENABLE_EMAIL'] = os.getenv(
|
|
|
|
'USER_ENABLE_EMAIL',
|
|
|
|
app.config['USER_ENABLE_EMAIL']
|
|
|
|
)
|
|
|
|
app.config['USER_ENABLE_CONFIRM_EMAIL'] = os.getenv(
|
|
|
|
'USER_ENABLE_CONFIRM_EMAIL',
|
|
|
|
app.config['USER_ENABLE_CONFIRM_EMAIL']
|
|
|
|
)
|
|
|
|
app.config['REQUIRE_PLAY_KEY'] = os.getenv(
|
|
|
|
'REQUIRE_PLAY_KEY',
|
|
|
|
app.config['REQUIRE_PLAY_KEY']
|
|
|
|
)
|
|
|
|
app.config['USER_ENABLE_INVITE_USER'] = os.getenv(
|
|
|
|
'USER_ENABLE_INVITE_USER',
|
|
|
|
app.config['USER_ENABLE_INVITE_USER']
|
|
|
|
)
|
|
|
|
app.config['USER_REQUIRE_INVITATION'] = os.getenv(
|
|
|
|
'USER_REQUIRE_INVITATION',
|
|
|
|
app.config['USER_REQUIRE_INVITATION']
|
|
|
|
)
|
2022-01-18 03:55:48 +00:00
|
|
|
app.config['ALLOW_ANALYTICS'] = os.getenv(
|
|
|
|
'ALLOW_ANALYTICS',
|
|
|
|
app.config['ALLOW_ANALYTICS']
|
|
|
|
)
|
2022-01-16 18:22:00 +00:00
|
|
|
app.config['SQLALCHEMY_ENGINE_OPTIONS'] = {
|
|
|
|
"pool_pre_ping": True,
|
|
|
|
"pool_size": 10,
|
|
|
|
"max_overflow": 2,
|
|
|
|
"pool_recycle": 300,
|
|
|
|
"pool_pre_ping": True,
|
|
|
|
"pool_use_lifo": True
|
|
|
|
}
|
2022-03-13 20:09:34 +00:00
|
|
|
app.config['MAIL_SERVER'] = os.getenv(
|
|
|
|
'MAIL_SERVER',
|
|
|
|
app.config['MAIL_SERVER']
|
|
|
|
)
|
|
|
|
app.config['MAIL_PORT'] = os.getenv(
|
|
|
|
'MAIL_USE_SSL',
|
|
|
|
app.config['MAIL_PORT']
|
|
|
|
)
|
|
|
|
app.config['MAIL_USE_SSL'] = os.getenv(
|
|
|
|
'MAIL_USE_SSL',
|
|
|
|
app.config['MAIL_USE_SSL']
|
|
|
|
)
|
|
|
|
app.config['MAIL_USE_TLS'] = os.getenv(
|
|
|
|
'MAIL_USE_TLS',
|
|
|
|
app.config['MAIL_USE_TLS']
|
|
|
|
)
|
|
|
|
app.config['MAIL_USERNAME'] = os.getenv(
|
|
|
|
'MAIL_USERNAME',
|
|
|
|
app.config['MAIL_USERNAME']
|
|
|
|
)
|
|
|
|
app.config['MAIL_PASSWORD'] = os.getenv(
|
|
|
|
'MAIL_PASSWORD',
|
|
|
|
app.config['MAIL_PASSWORD']
|
|
|
|
)
|
|
|
|
app.config['USER_EMAIL_SENDER_NAME'] = os.getenv(
|
|
|
|
'USER_EMAIL_SENDER_NAME',
|
|
|
|
app.config['USER_EMAIL_SENDER_NAME']
|
|
|
|
)
|
|
|
|
app.config['USER_EMAIL_SENDER_EMAIL'] = os.getenv(
|
|
|
|
'USER_EMAIL_SENDER_EMAIL',
|
|
|
|
app.config['USER_EMAIL_SENDER_EMAIL']
|
|
|
|
)
|
2022-12-17 07:47:10 +00:00
|
|
|
|
2022-12-17 07:21:39 +00:00
|
|
|
if "ENABLE_CHAR_XML_UPLOAD" not in app.config:
|
|
|
|
app.config['ENABLE_CHAR_XML_UPLOAD'] = False
|
|
|
|
app.config['ENABLE_CHAR_XML_UPLOAD'] = os.getenv(
|
|
|
|
'ENABLE_CHAR_XML_UPLOAD',
|
|
|
|
app.config['ENABLE_CHAR_XML_UPLOAD']
|
|
|
|
)
|
|
|
|
|
|
|
|
if "CLIENT_LOCATION" not in app.config:
|
2022-12-17 06:59:47 +00:00
|
|
|
app.config['CLIENT_LOCATION'] = 'app/luclient/'
|
2022-12-17 07:21:39 +00:00
|
|
|
app.config['CLIENT_LOCATION'] = os.getenv(
|
|
|
|
'CLIENT_LOCATION',
|
|
|
|
app.config['CLIENT_LOCATION']
|
|
|
|
)
|
2022-12-17 06:59:47 +00:00
|
|
|
|
2022-12-17 07:21:39 +00:00
|
|
|
if "CD_SQLITE_LOCATION" not in app.config:
|
2022-12-17 06:59:47 +00:00
|
|
|
app.config['CD_SQLITE_LOCATION'] = 'app/luclient/res/'
|
2022-12-17 07:21:39 +00:00
|
|
|
app.config['CD_SQLITE_LOCATION'] = os.getenv(
|
|
|
|
'CD_SQLITE_LOCATION',
|
|
|
|
app.config['CD_SQLITE_LOCATION']
|
|
|
|
)
|
|
|
|
|
2022-12-17 07:47:10 +00:00
|
|
|
if "CACHE_LOCATION" not in app.config:
|
|
|
|
app.config['CACHE_LOCATION'] = 'app/cache/'
|
|
|
|
app.config['CACHE_LOCATION'] = os.getenv(
|
|
|
|
'CACHE_LOCATION',
|
|
|
|
app.config['CACHE_LOCATION']
|
|
|
|
)
|
|
|
|
|
2022-12-17 06:59:47 +00:00
|
|
|
|
2022-01-16 18:22:00 +00:00
|
|
|
|
|
|
|
def gm_level(gm_level):
|
|
|
|
"""Decorator for handling permissions based on the user's GM Level
|
|
|
|
|
|
|
|
Args:
|
|
|
|
gm_level (int): 0-9
|
|
|
|
"""
|
|
|
|
def decorator(func):
|
|
|
|
@wraps(func)
|
|
|
|
def wrapper(*args, **kwargs):
|
|
|
|
if current_user.gm_level < gm_level:
|
|
|
|
return redirect(url_for('main.index'))
|
|
|
|
return func(*args, **kwargs)
|
|
|
|
return wrapper
|
|
|
|
return decorator
|
2022-02-12 05:05:00 +00:00
|
|
|
|
2022-03-13 02:09:35 +00:00
|
|
|
|
2022-02-12 05:05:00 +00:00
|
|
|
def log_audit(message):
|
|
|
|
AuditLog(
|
|
|
|
account_id=current_user.id,
|
|
|
|
action=message
|
|
|
|
).save()
|