Source code for genro_asgi.exceptions

# Copyright 2025 Softwell S.r.l.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

"""HTTP and WebSocket exception classes.

Classes:
    HTTPException — HTTP error response (status_code, detail, headers)
    HTTPForbidden, HTTPNotFound, HTTPUnauthorized — common HTTP errors
    WebSocketException — close WebSocket with error code
    WebSocketDisconnect — signal: client disconnected (not an error)
    Redirect — HTTP redirect (301/302/307/308)
"""


[docs] class HTTPException(Exception): """ HTTP exception with status code and detail. Raise this in handlers to return an HTTP error response. The framework will catch this and convert it to an appropriate HTTP response with the given status code, detail, and headers. Attributes: status_code: HTTP status code (expected 4xx or 5xx, not validated) detail: Error detail message headers: Response headers as list of tuples (supports duplicate names) Example: >>> raise HTTPException(404, detail="User not found") >>> raise HTTPException(401, headers={"WWW-Authenticate": "Bearer"}) >>> raise HTTPException(400, headers=[("Set-Cookie", "a=1"), ("Set-Cookie", "b=2")]) """
[docs] def __init__( self, status_code: int, detail: str = "", headers: dict[str, str] | list[tuple[str, str]] | None = None, ) -> None: """ Initialize HTTP exception. Args: status_code: HTTP status code (4xx, 5xx expected) detail: Error detail message (default: "") headers: Response headers as dict or list of tuples (default: None). Dict is converted to list internally to support duplicate names. """ self.status_code = status_code self.detail = detail # Normalize headers to list[tuple[str, str]] for consistent internal format if headers is None: self.headers: list[tuple[str, str]] | None = None elif isinstance(headers, dict): self.headers = list(headers.items()) else: self.headers = list(headers) super().__init__(detail)
def __repr__(self) -> str: """Return detailed string representation.""" return f"HTTPException(status_code={self.status_code}, detail={self.detail!r})"
[docs] class WebSocketException(Exception): """ WebSocket exception with close code and reason. Raise this to close a WebSocket connection with an error code. The framework will catch this and send a close frame with the given code and reason. Attributes: code: WebSocket close code (1000-4999, not validated) reason: Close reason message Example: >>> raise WebSocketException(code=4000, reason="Invalid message") """
[docs] def __init__( self, code: int = 1000, reason: str = "", ) -> None: """ Initialize WebSocket exception. Args: code: WebSocket close code (default: 1000) reason: Close reason message (default: "") """ self.code = code self.reason = reason super().__init__(reason)
def __repr__(self) -> str: """Return detailed string representation.""" return f"WebSocketException(code={self.code}, reason={self.reason!r})"
[docs] class WebSocketDisconnect(Exception): """ Raised when a WebSocket is disconnected by the client. This is not an error, just a signal that the connection was closed. The framework raises this when a receive operation fails because the client has disconnected. Attributes: code: WebSocket close code from client reason: Close reason from client (if any) Example: >>> try: ... data = await websocket.receive_text() ... except WebSocketDisconnect as e: ... print(f"Client disconnected: {e.code}") """
[docs] def __init__( self, code: int = 1000, reason: str = "", ) -> None: """ Initialize disconnect exception. Args: code: WebSocket close code (default: 1000) reason: Close reason (default: "") """ self.code = code self.reason = reason super().__init__(f"WebSocket disconnected with code {code}")
def __repr__(self) -> str: """Return detailed string representation.""" return f"WebSocketDisconnect(code={self.code}, reason={self.reason!r})"
[docs] class Redirect(HTTPException): """HTTP redirect exception (302 by default). Attributes: url: Target redirect URL (set as Location header). status_code: HTTP status (302, 301, 307, 308). """
[docs] def __init__(self, url: str, status_code: int = 302) -> None: """Args: url: Target URL for the Location header. status_code: HTTP redirect status code (default: 302). """ super().__init__(status_code, headers={"Location": url}) self.url = url
def __repr__(self) -> str: """Return detailed string representation.""" return f"Redirect(url={self.url!r}, status_code={self.status_code})"
[docs] class HTTPNotFound(HTTPException): """HTTP 404 Not Found exception."""
[docs] def __init__(self, detail: str = "Not found") -> None: """Args: detail: Error message (default: "Not found"). """ super().__init__(404, detail=detail)
[docs] class HTTPUnauthorized(HTTPException): """HTTP 401 Unauthorized exception."""
[docs] def __init__(self, detail: str = "Unauthorized") -> None: """Args: detail: Error message (default: "Unauthorized"). """ super().__init__(401, detail=detail)
[docs] class HTTPForbidden(HTTPException): """HTTP 403 Forbidden exception."""
[docs] def __init__(self, detail: str = "Forbidden") -> None: """Args: detail: Error message (default: "Forbidden"). """ super().__init__(403, detail=detail)
[docs] class HTTPBadRequest(HTTPException): """HTTP 400 Bad Request exception."""
[docs] def __init__(self, detail: str = "Bad request") -> None: """Args: detail: Error message (default: "Bad request"). """ super().__init__(400, detail=detail)
[docs] class HTTPServiceUnavailable(HTTPException): """HTTP 503 Service Unavailable exception."""
[docs] def __init__(self, detail: str = "Service unavailable") -> None: """Args: detail: Error message (default: "Service unavailable"). """ super().__init__(503, detail=detail)