BrickTracker/bricktracker/views/error.py
2025-01-17 11:03:00 +01:00

145 lines
3.7 KiB
Python

import logging
from sqlite3 import Error, OperationalError
import traceback
from typing import Tuple
from flask import jsonify, redirect, request, render_template, url_for
from werkzeug.wrappers.response import Response
from ..exceptions import DatabaseException, ErrorException, NotFoundException
logger = logging.getLogger(__name__)
# Get the cleaned exception
def cleaned_exception(e: Exception, /) -> str:
trace = traceback.TracebackException.from_exception(e)
cleaned: list[str] = []
# Hacky: stripped from the call to the decorator wrapper() or outer()
for line in trace.format():
if 'in wrapper' not in line and 'in outer' not in line:
cleaned.append(line)
return ''.join(cleaned)
# Generic error
def error(
error: Exception | None,
file: str,
/,
json: bool = False,
post_redirect: str | None = None,
**kwargs,
) -> str | Tuple[str | Response, int] | Response:
# Back to the index if no error (not sure if this can happen)
if error is None:
if json:
return jsonify({'error': 'error() called without an error'})
else:
return redirect(url_for('index.index'))
# Convert SQLite errors
if isinstance(error, (Error, OperationalError)):
error = DatabaseException(error)
# Clear redirect if not POST or json
if json or request.method != 'POST':
post_redirect = None
# Not found
if isinstance(error, NotFoundException):
return error_404(
error,
json=json,
post_redirect=post_redirect,
**kwargs
)
# Common error
elif isinstance(error, ErrorException):
# Error
logger.error('{title}: {error}'.format(
title=error.title,
error=str(error),
))
# Debug
logger.debug(cleaned_exception(error))
if json:
return jsonify({'error': str(error)})
elif post_redirect is not None:
return redirect(url_for(
post_redirect,
error=str(error),
**kwargs,
))
else:
return render_template(
'error.html',
title=error.title,
error=str(error)
)
# Exception
else:
# Error
logger.error(cleaned_exception(error))
if error.__traceback__ is not None:
line = error.__traceback__.tb_lineno
else:
line = None
if json:
return jsonify({
'error': 'Exception: {error}'.format(error=str(error)),
'name': type(error).__name__,
'line': line,
'file': file,
}), 500
elif post_redirect is not None:
return redirect(url_for(
post_redirect,
error=str(error),
**kwargs,
))
else:
return render_template(
'exception.html',
error=str(error),
name=type(error).__name__,
line=line,
file=file,
)
# Error 404
def error_404(
error: Exception,
/,
json: bool = False,
post_redirect: str | None = None,
**kwargs,
) -> Tuple[str | Response, int]:
# Warning
logger.warning('Not found: {error}'.format(
error=str(error),
))
if json:
return jsonify({
'error': 'Not found: {error}'.format(error=str(error))
}), 404
elif post_redirect is not None:
return redirect(url_for(
post_redirect,
error=str(error),
**kwargs
)), 404
else:
return render_template('404.html', error=str(error)), 404