Python-flask обрабатывает ошибки приложения

Я создал фляжное приложение и пытаюсь поймать необработанные ошибки приложения из своих маршрутов, используя декоратор обработчика ошибок.

У меня есть main.py, который выглядит так:

app = Flask(__name__)
api = Api(app)

api.add_resource(Ping, '/ping')

@app.errorhandler(500)
def internal_server_error(error):
    print "caught internal server error"
    return "This page does not exist", 500

Пинг маршрута находится в другом файле, вот примерная версия файла

class Ping(Resource):
    def get(self):

        raise
        return {}, 200

Я сделал рейз, чтобы попытаться воспроизвести внутреннюю ошибку сервера 500. Вот пример ошибки, вызванной моим приложением

[2019-01-26 10:37:36,449] ERROR in app: Exception on /events/v1/monitoring/ping [GET]
Traceback (most recent call last):
  File "/usr/local/lib/python2.7/site-packages/flask/app.py", line 1813, in full_dispatch_request
    rv = self.dispatch_request()
  File "/usr/local/lib/python2.7/site-packages/flask/app.py", line 1799, in dispatch_request
    return self.view_functions[rule.endpoint](**req.view_args)
  File "/usr/local/lib64/python2.7/site-packages/flask_restful/__init__.py", line 480, in wrapper
    resp = resource(*args, **kwargs)
  File "/usr/local/lib/python2.7/site-packages/flask/views.py", line 88, in view
    return self.dispatch_request(*args, **kwargs)
  File "/usr/local/lib64/python2.7/site-packages/flask_restful/__init__.py", line 595, in dispatch_request
    resp = meth(*args, **kwargs)
  File "/usr/lib/python2.7/site-packages/myapi/ping.py", line 17, in get
    raise
TypeError: exceptions must be old-style classes or derived from BaseException, not NoneType
127.0.0.1 - - [26/Jan/2019 10:37:36] "GET /ping HTTP/1.0" 500 -

По причинам, которые я не смог понять, декоратор @app.errorhandler не улавливает ошибки 500, возникающие в приложении.

Каково значение «DEBUG» в конфигурации вашего приложения Flask?

DoctorAgon 26.01.2019 16:33

его значение по умолчанию, False

user6434902 27.01.2019 12:20
Почему в Python есть оператор "pass"?
Почему в Python есть оператор "pass"?
Оператор pass в Python - это простая концепция, которую могут быстро освоить даже новички без опыта программирования.
Некоторые методы, о которых вы не знали, что они существуют в Python
Некоторые методы, о которых вы не знали, что они существуют в Python
Python - самый известный и самый простой в изучении язык в наши дни. Имея широкий спектр применения в области машинного обучения, Data Science,...
Основы Python Часть I
Основы Python Часть I
Вы когда-нибудь задумывались, почему в программах на Python вы видите приведенный ниже код?
LeetCode - 1579. Удаление максимального числа ребер для сохранения полной проходимости графа
LeetCode - 1579. Удаление максимального числа ребер для сохранения полной проходимости графа
Алиса и Боб имеют неориентированный граф из n узлов и трех типов ребер:
Оптимизация кода с помощью тернарного оператора Python
Оптимизация кода с помощью тернарного оператора Python
И последнее, что мы хотели бы показать вам, прежде чем двигаться дальше, это
Советы по эффективной веб-разработке с помощью Python
Советы по эффективной веб-разработке с помощью Python
Как веб-разработчик, Python может стать мощным инструментом для создания эффективных и масштабируемых веб-приложений.
0
2
2 539
2
Перейти к ответу Данный вопрос помечен как решенный

Ответы 2

Вместо этого попробуйте это:

@app.errorhandler(Exception)

В моем приложении у меня есть несколько обработчиков ошибок. Один для общего случая исключения, как это сделал здесь Александр. Затем несколько дополнительных обработчиков для 500, 403, 404 и т. д. Может быть, вам больше повезет с обоими?

DoctorAgon 26.01.2019 16:36
Ответ принят как подходящий

Я думаю, вы используете фляжка-успокоительный. Чтобы ваш обработчик ошибок был пойман, вы должны установить параметр конфигурации PROPAGATE_EXCEPTIONS на True в вашем файле конфигурации.

Flask restful имеет собственный обработчик ошибок, отличный от того, что есть в Flask. Это выдержка:

[...]

if not isinstance(e, HTTPException) and current_app.propagate_exceptions:
    exc_type, exc_value, tb = sys.exc_info()
    if exc_value is e:
        raise
    else:
        raise e

headers = Headers()
if isinstance(e, HTTPException):
    code = e.code
    default_data = {
        'message': getattr(e, 'description', http_status_message(code))
    }
    headers = e.get_response().headers
else:
    code = 500
    default_data = {
        'message': http_status_message(code),
    }
[...]

После установки PROPAGATE_EXCEPTIONS на True вы должны добавить по крайней мере эти два обработчика ошибок:

@app.errorhandler(Exception)
def internal_server_error(e):
    return jsonify({'msg': 'Internal server error'}), 500

@app.errorhandler(500)
def internal_server_error_500(e):
    return jsonify({'msg': 'Internal server error'}), 500

Таким образом, первый будет перехватывать любое исключение, поднятое на уровне приложения, даже ваши собственные пользовательские исключения, а второй перехватит любую ошибку 500, например, возникшую на abort(500).

Да, я использую flask-restful Как только я настрою конфиг, вы знаете, как отловить все 500 внутренних ошибок сервера и корректно с ними справиться?

user6434902 30.01.2019 06:12

Другие вопросы по теме