mirror of
https://github.com/alerta/alerta.git
synced 2025-01-24 17:29:39 +00:00
135 lines
3.5 KiB
Python
135 lines
3.5 KiB
Python
import traceback
|
|
from typing import Any, Dict, Tuple, Union
|
|
|
|
from flask import Response, current_app, g, jsonify
|
|
from werkzeug.exceptions import HTTPException
|
|
from werkzeug.routing import RoutingException
|
|
|
|
|
|
class BaseError(Exception):
|
|
code = 500
|
|
description = 'Unhandled exception'
|
|
|
|
def __init__(self, message, code=None, errors=None):
|
|
super().__init__(message)
|
|
self.message = message
|
|
if code is not None:
|
|
self.code = code
|
|
self.errors = errors
|
|
self.request_id = None
|
|
|
|
|
|
class AlertaException(BaseError):
|
|
pass
|
|
|
|
|
|
class RejectException(BaseError):
|
|
"""The alert was rejected because the format did not meet the required policy."""
|
|
pass
|
|
|
|
|
|
class RateLimit(BaseError):
|
|
"""Too many alerts have been received for a resource or from an origin."""
|
|
pass
|
|
|
|
|
|
class HeartbeatReceived(BaseError):
|
|
"""Alert was not processed because it was converted into a heartbeat."""
|
|
|
|
def __init__(self, id):
|
|
self.id = id
|
|
super().__init__('Alert converted to Heartbeat')
|
|
|
|
|
|
class BlackoutPeriod(BaseError):
|
|
"""Alert was not processed because it was sent during a blackout period."""
|
|
pass
|
|
|
|
|
|
class ForwardingLoop(BaseError):
|
|
"""Forwarding loop detected."""
|
|
pass
|
|
|
|
|
|
class InvalidAction(BaseError):
|
|
"""Invalid or redundant action for the current alert status or severity."""
|
|
pass
|
|
|
|
|
|
class NoCustomerMatch(BaseError):
|
|
"""There was no customer lookup found for the user or group."""
|
|
pass
|
|
|
|
|
|
class ApiError(BaseError):
|
|
pass
|
|
|
|
|
|
class BasicAuthError(BaseError):
|
|
pass
|
|
|
|
|
|
class ExceptionHandlers:
|
|
|
|
def register(self, app):
|
|
from werkzeug.exceptions import default_exceptions
|
|
for code in default_exceptions.keys():
|
|
app.register_error_handler(code, handle_http_error)
|
|
app.register_error_handler(ApiError, handle_api_error)
|
|
app.register_error_handler(BasicAuthError, handle_basic_auth_error)
|
|
app.register_error_handler(Exception, handle_exception)
|
|
|
|
|
|
def handle_http_error(error: HTTPException) -> Tuple[Response, int]:
|
|
error.code = error.code or 500
|
|
if error.code >= 500:
|
|
current_app.logger.exception(error)
|
|
return jsonify({
|
|
'status': 'error',
|
|
'message': str(error),
|
|
'code': error.code,
|
|
'errors': [
|
|
error.description
|
|
],
|
|
'requestId': g.request_id
|
|
}), error.code
|
|
|
|
|
|
def handle_api_error(error: ApiError) -> Tuple[Response, int]:
|
|
if error.code >= 500:
|
|
current_app.logger.exception(error)
|
|
return jsonify({
|
|
'status': 'error',
|
|
'message': error.message,
|
|
'code': error.code,
|
|
'errors': error.errors,
|
|
'requestId': g.request_id
|
|
}), error.code
|
|
|
|
|
|
def handle_basic_auth_error(error: BasicAuthError) -> Tuple[Response, int, Dict[str, Any]]:
|
|
return jsonify({
|
|
'status': 'error',
|
|
'message': error.message,
|
|
'code': error.code,
|
|
'errors': error.errors,
|
|
'requestId': g.request_id
|
|
}), error.code, {'WWW-Authenticate': f"Basic realm={current_app.config['BASIC_AUTH_REALM']}"}
|
|
|
|
|
|
def handle_exception(error: Exception) -> Union[Tuple[Response, int], Exception]:
|
|
# RoutingExceptions are used internally to trigger routing
|
|
# actions, such as slash redirects raising RequestRedirect.
|
|
if isinstance(error, RoutingException):
|
|
return error
|
|
|
|
current_app.logger.exception(error)
|
|
return jsonify({
|
|
'status': 'error',
|
|
'message': str(error),
|
|
'code': 500,
|
|
'errors': [
|
|
traceback.format_exc()
|
|
],
|
|
'requestId': g.request_id
|
|
}), 500
|