2016-10-31 15:43:11 +00:00
|
|
|
import json
|
2017-12-14 17:12:26 +00:00
|
|
|
|
|
|
|
|
from flask import jsonify, current_app, request
|
2020-03-13 17:12:53 +00:00
|
|
|
from jsonschema import ValidationError as JsonSchemaValidationError
|
2018-01-19 12:23:07 +00:00
|
|
|
from notifications_utils.recipients import InvalidEmailError
|
2016-10-31 15:43:11 +00:00
|
|
|
from sqlalchemy.exc import DataError
|
2016-10-31 12:22:26 +00:00
|
|
|
from sqlalchemy.orm.exc import NoResultFound
|
2017-12-14 17:12:26 +00:00
|
|
|
|
2016-10-31 12:22:26 +00:00
|
|
|
from app.authentication.auth import AuthError
|
2016-10-25 18:04:03 +01:00
|
|
|
from app.errors import InvalidRequest
|
|
|
|
|
|
|
|
|
|
|
2016-10-27 11:46:37 +01:00
|
|
|
class TooManyRequestsError(InvalidRequest):
|
|
|
|
|
status_code = 429
|
|
|
|
|
message_template = 'Exceeded send limits ({}) for today'
|
|
|
|
|
|
|
|
|
|
def __init__(self, sending_limit):
|
|
|
|
|
self.message = self.message_template.format(sending_limit)
|
|
|
|
|
|
|
|
|
|
|
2017-04-24 14:15:08 +01:00
|
|
|
class RateLimitError(InvalidRequest):
|
|
|
|
|
status_code = 429
|
|
|
|
|
message_template = 'Exceeded rate limit for key type {} of {} requests per {} seconds'
|
|
|
|
|
|
|
|
|
|
def __init__(self, sending_limit, interval, key_type):
|
|
|
|
|
# normal keys are spoken of as "live" in the documentation
|
|
|
|
|
# so using this in the error messaging
|
|
|
|
|
if key_type == 'normal':
|
|
|
|
|
key_type = 'live'
|
|
|
|
|
|
|
|
|
|
self.message = self.message_template.format(key_type.upper(), sending_limit, interval)
|
|
|
|
|
|
|
|
|
|
|
2016-10-27 11:46:37 +01:00
|
|
|
class BadRequestError(InvalidRequest):
|
|
|
|
|
message = "An error occurred"
|
2016-10-25 18:04:03 +01:00
|
|
|
|
2020-12-22 15:46:31 +00:00
|
|
|
def __init__(self, fields=None, message=None, status_code=400):
|
2017-07-31 18:28:00 +01:00
|
|
|
self.status_code = status_code
|
2020-12-22 15:46:31 +00:00
|
|
|
self.fields = fields or []
|
2016-10-27 11:46:37 +01:00
|
|
|
self.message = message if message else self.message
|
2016-10-25 18:04:03 +01:00
|
|
|
|
|
|
|
|
|
2020-03-13 17:12:53 +00:00
|
|
|
class ValidationError(InvalidRequest):
|
|
|
|
|
message = "Your notification has failed validation"
|
|
|
|
|
|
2020-12-22 15:46:31 +00:00
|
|
|
def __init__(self, fields=None, message=None, status_code=400):
|
2020-03-13 17:12:53 +00:00
|
|
|
self.status_code = status_code
|
2020-12-22 15:46:31 +00:00
|
|
|
self.fields = fields or []
|
2020-03-13 17:12:53 +00:00
|
|
|
self.message = message if message else self.message
|
|
|
|
|
|
|
|
|
|
|
2019-09-17 12:16:24 +01:00
|
|
|
class PDFNotReadyError(BadRequestError):
|
|
|
|
|
def __init__(self):
|
|
|
|
|
super().__init__(message='PDF not available yet, try again later', status_code=400)
|
|
|
|
|
|
|
|
|
|
|
2016-10-25 18:04:03 +01:00
|
|
|
def register_errors(blueprint):
|
2018-01-19 12:23:07 +00:00
|
|
|
@blueprint.errorhandler(InvalidEmailError)
|
|
|
|
|
def invalid_format(error):
|
|
|
|
|
# Please not that InvalidEmailError is re-raised for InvalidEmail or InvalidPhone,
|
|
|
|
|
# work should be done in the utils app to tidy up these errors.
|
2018-02-08 13:38:32 +00:00
|
|
|
current_app.logger.info(error)
|
2018-01-19 12:23:07 +00:00
|
|
|
return jsonify(status_code=400,
|
|
|
|
|
errors=[{"error": error.__class__.__name__, "message": str(error)}]), 400
|
|
|
|
|
|
2016-10-27 11:46:37 +01:00
|
|
|
@blueprint.errorhandler(InvalidRequest)
|
|
|
|
|
def invalid_data(error):
|
2018-02-08 13:38:32 +00:00
|
|
|
current_app.logger.info(error)
|
2016-10-27 11:46:37 +01:00
|
|
|
response = jsonify(error.to_dict_v2()), error.status_code
|
|
|
|
|
return response
|
|
|
|
|
|
2020-03-13 17:12:53 +00:00
|
|
|
@blueprint.errorhandler(JsonSchemaValidationError)
|
2016-10-31 15:43:11 +00:00
|
|
|
def validation_error(error):
|
2018-02-08 13:38:32 +00:00
|
|
|
current_app.logger.info(error)
|
2016-10-31 15:43:11 +00:00
|
|
|
return jsonify(json.loads(error.message)), 400
|
|
|
|
|
|
2016-10-31 12:22:26 +00:00
|
|
|
@blueprint.errorhandler(NoResultFound)
|
|
|
|
|
@blueprint.errorhandler(DataError)
|
|
|
|
|
def no_result_found(e):
|
2018-02-08 13:38:32 +00:00
|
|
|
current_app.logger.info(e)
|
2016-11-10 14:53:39 +00:00
|
|
|
return jsonify(status_code=404,
|
|
|
|
|
errors=[{"error": e.__class__.__name__, "message": "No result found"}]), 404
|
2016-10-31 12:22:26 +00:00
|
|
|
|
|
|
|
|
@blueprint.errorhandler(AuthError)
|
|
|
|
|
def auth_error(error):
|
2017-12-14 17:12:26 +00:00
|
|
|
current_app.logger.info('API AuthError, client: {} error: {}'.format(request.headers.get('User-Agent'), error))
|
2016-11-01 10:33:34 +00:00
|
|
|
return jsonify(error.to_dict_v2()), error.code
|
2016-10-31 12:22:26 +00:00
|
|
|
|
2016-10-27 11:46:37 +01:00
|
|
|
@blueprint.errorhandler(Exception)
|
2016-10-31 12:22:26 +00:00
|
|
|
def internal_server_error(error):
|
|
|
|
|
current_app.logger.exception(error)
|
2016-11-10 14:53:39 +00:00
|
|
|
return jsonify(status_code=500,
|
|
|
|
|
errors=[{"error": error.__class__.__name__, "message": 'Internal server error'}]), 500
|