You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

108 lines
3.1 KiB

"""Фабрика Flask-приложения.
Этап 1:
E1.0 — фундамент (БД-пул, sessions, base.html).
E1.1 — авторизация (cookie-сессии Flask, bcrypt + Werkzeug, опц. HR_AUTH).
"""
from __future__ import annotations
import os
import secrets
from datetime import timedelta
from flask import Flask, jsonify, render_template, request
_ROLE_LABELS = {
'employee': 'Сотрудник',
'manager': 'Руководитель',
'hr': 'HR',
}
def _format_role(role: str | None) -> str:
return _ROLE_LABELS.get((role or '').strip().lower(), '')
def _format_surname_with_initials(full_name: str | None, fallback: str | None = None) -> str:
name = (full_name or '').strip()
if not name:
return (fallback or '').strip() or ''
parts = [p for p in name.replace('\xa0', ' ').split(' ') if p]
if len(parts) < 2:
return name
surname = parts[0]
initials = []
for p in parts[1:3]:
initials.append(f'{p[0].upper()}.')
return f"{surname} {' '.join(initials)}".strip()
def create_app() -> Flask:
app = Flask(
__name__,
instance_relative_config=True,
template_folder='templates',
static_folder='static',
static_url_path='/static',
)
sk = (os.environ.get('SECRET_KEY') or '').strip()
app.config['SECRET_KEY'] = sk or secrets.token_hex(32)
app.config['MAX_CONTENT_LENGTH'] = 16 * 1024 * 1024 # 16 MB upload limit
app.config.update(
SESSION_COOKIE_NAME='testing_session',
SESSION_COOKIE_HTTPONLY=True,
SESSION_COOKIE_SAMESITE='Lax',
SESSION_COOKIE_SECURE=(os.environ.get('FLASK_ENV') == 'production'),
PERMANENT_SESSION_LIFETIME=timedelta(days=7),
)
from .blueprints.main import main_bp
from .blueprints.settings import settings_bp
from .auth import auth_bp
from .tests import tests_bp
app.register_blueprint(main_bp)
app.register_blueprint(auth_bp)
app.register_blueprint(tests_bp)
app.register_blueprint(settings_bp)
from . import db as _db
@app.teardown_appcontext
def _shutdown_session(exc): # noqa: ARG001
_db.remove_session()
from .config import is_assignment_feature_enabled, is_dev_ui, is_hr_auth_enabled
from .auth.decorators import current_user as _current_user
@app.context_processor
def _inject_globals():
return {
'current_user': _current_user(),
'hr_auth_enabled': is_hr_auth_enabled(),
'dev_ui': is_dev_ui(),
'assignment_ui': is_assignment_feature_enabled(),
'format_name_short': _format_surname_with_initials,
'format_role': _format_role,
}
@app.errorhandler(404)
def _not_found(_e):
if _is_api_path():
return jsonify(error='not_found'), 404
return render_template('404.html'), 404
@app.errorhandler(500)
def _internal_error(_e):
if _is_api_path():
return jsonify(error='internal_error'), 500
return render_template('500.html'), 500
return app
def _is_api_path() -> bool:
return request.path.startswith('/api/')