2026-03-11 15:35:40 +00:00
|
|
|
import logging
|
|
|
|
|
from logging.handlers import RotatingFileHandler
|
|
|
|
|
import os
|
2026-03-13 12:07:11 +00:00
|
|
|
from flask import Flask, render_template, request
|
2026-03-10 12:16:41 +00:00
|
|
|
from flask_login import LoginManager
|
|
|
|
|
from flask_wtf.csrf import CSRFProtect
|
|
|
|
|
from config import Config
|
2026-03-11 10:15:22 +00:00
|
|
|
from app.models import db, Config as ConfigModel, User
|
|
|
|
|
from datetime import datetime
|
2026-03-11 15:35:40 +00:00
|
|
|
import secrets
|
2026-03-10 12:16:41 +00:00
|
|
|
|
|
|
|
|
login_manager = LoginManager()
|
|
|
|
|
login_manager.login_view = 'auth.login'
|
|
|
|
|
login_manager.login_message = 'Please log in to access this page.'
|
2026-03-11 15:35:40 +00:00
|
|
|
# Enhanced session security
|
|
|
|
|
login_manager.session_protection = "strong"
|
2026-03-10 12:16:41 +00:00
|
|
|
|
2026-03-11 10:15:22 +00:00
|
|
|
@login_manager.user_loader
|
|
|
|
|
def load_user(user_id):
|
|
|
|
|
"""Load user for Flask-Login"""
|
|
|
|
|
return User.query.get(int(user_id))
|
|
|
|
|
|
|
|
|
|
def get_logo_filename():
|
|
|
|
|
"""Get the logo filename from database configuration"""
|
|
|
|
|
try:
|
|
|
|
|
logo_config = ConfigModel.query.filter_by(key='logo_filename').first()
|
|
|
|
|
return logo_config.value if logo_config else None
|
|
|
|
|
except:
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
def format_date(value, format='%Y'):
|
|
|
|
|
"""Format date for Jinja2 templates"""
|
|
|
|
|
if value:
|
|
|
|
|
return value.strftime(format)
|
|
|
|
|
return ''
|
|
|
|
|
|
2026-03-10 12:16:41 +00:00
|
|
|
def create_app(config_class=Config):
|
|
|
|
|
app = Flask(__name__)
|
|
|
|
|
app.config.from_object(config_class)
|
|
|
|
|
|
2026-03-11 15:35:40 +00:00
|
|
|
# Enhanced security configuration
|
|
|
|
|
app.config['SESSION_COOKIE_SECURE'] = True # HTTPS only
|
|
|
|
|
app.config['SESSION_COOKIE_HTTPONLY'] = True # Prevent XSS
|
|
|
|
|
app.config['SESSION_COOKIE_SAMESITE'] = 'Lax' # CSRF protection
|
|
|
|
|
app.config['PERMANENT_SESSION_LIFETIME'] = 3600 # 1 hour session timeout
|
|
|
|
|
|
|
|
|
|
# Setup logging
|
|
|
|
|
if not app.debug and not app.testing:
|
|
|
|
|
if not os.path.exists('logs'):
|
|
|
|
|
os.mkdir('logs')
|
|
|
|
|
file_handler = RotatingFileHandler('logs/ep_inspection_tool.log', maxBytes=10240, backupCount=10)
|
|
|
|
|
file_handler.setFormatter(logging.Formatter(
|
|
|
|
|
'%(asctime)s %(levelname)s: %(message)s [in %(pathname)s:%(lineno)d]'))
|
|
|
|
|
file_handler.setLevel(logging.INFO)
|
|
|
|
|
app.logger.addHandler(file_handler)
|
|
|
|
|
app.logger.setLevel(logging.INFO)
|
|
|
|
|
app.logger.info('EP Inspection Tool startup')
|
|
|
|
|
|
2026-03-10 12:16:41 +00:00
|
|
|
# Initialize extensions
|
|
|
|
|
db.init_app(app)
|
|
|
|
|
login_manager.init_app(app)
|
|
|
|
|
csrf = CSRFProtect(app)
|
|
|
|
|
|
|
|
|
|
# Create uploads directory if it doesn't exist
|
|
|
|
|
os.makedirs(app.config['UPLOAD_FOLDER'], exist_ok=True)
|
|
|
|
|
|
|
|
|
|
# Create certs directory if it doesn't exist
|
|
|
|
|
os.makedirs(os.path.dirname(app.config['CERT_PATH']), exist_ok=True)
|
|
|
|
|
|
|
|
|
|
# Register blueprints
|
|
|
|
|
from app.routes.auth import auth_bp
|
|
|
|
|
from app.routes.inspections import inspections_bp
|
|
|
|
|
from app.routes.admin import admin_bp
|
|
|
|
|
from app.routes.export import export_bp
|
|
|
|
|
|
|
|
|
|
app.register_blueprint(auth_bp)
|
|
|
|
|
app.register_blueprint(inspections_bp)
|
|
|
|
|
app.register_blueprint(admin_bp)
|
|
|
|
|
app.register_blueprint(export_bp)
|
|
|
|
|
|
2026-03-11 10:15:22 +00:00
|
|
|
# Add logo filename to template context
|
|
|
|
|
@app.context_processor
|
|
|
|
|
def inject_logo():
|
|
|
|
|
return dict(logo_filename=get_logo_filename())
|
|
|
|
|
|
|
|
|
|
# Add custom filters
|
|
|
|
|
@app.template_filter('format_date')
|
|
|
|
|
def format_date_filter(value, format='%Y'):
|
|
|
|
|
return format_date(value, format)
|
|
|
|
|
|
|
|
|
|
# Add current date function
|
|
|
|
|
@app.context_processor
|
|
|
|
|
def inject_current_date():
|
|
|
|
|
return dict(moment=lambda: datetime.now().strftime('%Y-%m-%d %H:%M'))
|
|
|
|
|
|
2026-03-10 12:16:41 +00:00
|
|
|
# Error handlers
|
|
|
|
|
@app.errorhandler(404)
|
|
|
|
|
def not_found_error(error):
|
2026-03-11 15:35:40 +00:00
|
|
|
app.logger.warning(f'Page not found: {request.url}')
|
2026-03-10 12:16:41 +00:00
|
|
|
return render_template('errors/404.html'), 404
|
|
|
|
|
|
|
|
|
|
@app.errorhandler(500)
|
|
|
|
|
def internal_error(error):
|
|
|
|
|
db.session.rollback()
|
2026-03-11 15:35:40 +00:00
|
|
|
app.logger.error(f'Server Error: {error}')
|
2026-03-10 12:16:41 +00:00
|
|
|
return render_template('errors/500.html'), 500
|
|
|
|
|
|
2026-03-11 15:35:40 +00:00
|
|
|
# Security headers
|
|
|
|
|
@app.after_request
|
|
|
|
|
def after_request(response):
|
|
|
|
|
response.headers['X-Content-Type-Options'] = 'nosniff'
|
|
|
|
|
response.headers['X-Frame-Options'] = 'DENY'
|
|
|
|
|
response.headers['X-XSS-Protection'] = '1; mode=block'
|
|
|
|
|
return response
|
|
|
|
|
|
2026-03-10 12:16:41 +00:00
|
|
|
return app
|