Source code for app.lib.exceptions
"""Litestar-saqlalchemy exception types.
Also, defines functions that translate service and repository exceptions
into HTTP exceptions.
"""
from __future__ import annotations
import sys
from typing import TYPE_CHECKING
from advanced_alchemy.exceptions import IntegrityError
from litestar.exceptions import (
HTTPException,
InternalServerException,
NotFoundException,
PermissionDeniedException,
)
from litestar.exceptions.responses import create_debug_response, create_exception_response
from litestar.repository.exceptions import ConflictError, NotFoundError, RepositoryError
from litestar.status_codes import HTTP_409_CONFLICT, HTTP_500_INTERNAL_SERVER_ERROR
from structlog.contextvars import bind_contextvars
if TYPE_CHECKING:
from typing import Any
from litestar.connection import Request
from litestar.middleware.exceptions.middleware import ExceptionResponseContent
from litestar.response import Response
from litestar.types import Scope
__all__ = (
"ApplicationError",
"AuthorizationError",
"HealthCheckConfigurationError",
"after_exception_hook_handler",
)
[docs]
class ApplicationError(Exception):
"""Base exception type for the lib's custom exception types."""
detail: str
[docs]
def __init__(self, *args: Any, detail: str = "") -> None:
"""Initialize ``AdvancedAlchemyException``.
Args:
*args: args are converted to :class:`str` before passing to :class:`Exception`
detail: detail of the exception.
"""
str_args = [str(arg) for arg in args if arg]
if not detail:
if str_args:
detail, *str_args = str_args
elif hasattr(self, "detail"):
detail = self.detail
self.detail = detail
super().__init__(*str_args)
def __repr__(self) -> str:
if self.detail:
return f"{self.__class__.__name__} - {self.detail}"
return self.__class__.__name__
def __str__(self) -> str:
return " ".join((*self.args, self.detail)).strip()
class MissingDependencyError(ApplicationError, ImportError):
"""Missing optional dependency.
This exception is raised only when a module depends on a dependency that has not been installed.
"""
class ApplicationClientError(ApplicationError):
"""Base exception type for client errors."""
[docs]
class AuthorizationError(ApplicationClientError):
"""A user tried to do something they shouldn't have."""
[docs]
class HealthCheckConfigurationError(ApplicationError):
"""An error occurred while registering an health check."""
class _HTTPConflictException(HTTPException):
"""Request conflict with the current state of the target resource."""
status_code = HTTP_409_CONFLICT
[docs]
async def after_exception_hook_handler(exc: Exception, _scope: Scope) -> None:
"""Binds `exc_info` key with exception instance as value to structlog
context vars.
This must be a coroutine so that it is not wrapped in a thread where we'll lose context.
Args:
exc: the exception that was raised.
_scope: scope of the request
"""
if isinstance(exc, ApplicationError):
return
if isinstance(exc, HTTPException) and exc.status_code < HTTP_500_INTERNAL_SERVER_ERROR:
return
bind_contextvars(exc_info=sys.exc_info())
def exception_to_http_response(
request: Request[Any, Any, Any],
exc: ApplicationError | RepositoryError,
) -> Response[ExceptionResponseContent]:
"""Transform repository exceptions to HTTP exceptions.
Args:
request: The request that experienced the exception.
exc: Exception raised during handling of the request.
Returns:
Exception response appropriate to the type of original exception.
"""
http_exc: type[HTTPException]
if isinstance(exc, NotFoundError):
http_exc = NotFoundException
elif isinstance(exc, ConflictError | RepositoryError | IntegrityError):
http_exc = _HTTPConflictException
elif isinstance(exc, AuthorizationError):
http_exc = PermissionDeniedException
else:
http_exc = InternalServerException
if request.app.debug and http_exc not in (PermissionDeniedException, NotFoundError, AuthorizationError):
return create_debug_response(request, exc)
return create_exception_response(request, http_exc(detail=str(exc.__cause__)))