Получить Errno из запросов Python ConnectionError?

Я ловлю и печатаю запросы Python ConnectionErrors в порядке только с этим:

except requests.exceptions.ConnectionError as e:
    logger.warning(str(e.message))

выводит сообщения типа:

HTTPSConnectionPool(host='10.100.24.16', port=443): Max retries exceeded with url: /api/datastores/06651841-bbdb-472a-bde2-689d8cb8da19 (Caused by <class 'socket.error'>: [Errno 61] Connection refused)

и

HTTPSConnectionPool(host='10.100.24.16', port=443): Max retries exceeded with url: /api/datastores/06651841-bbdb-472a-bde2-689d8cb8da19 (Caused by <class 'socket.error'>: [Errno 65] No route to host)

и многие другие. Мне интересно, какой лучший, самый Питонический способ получить это errno, которое отображается в сообщении? Я хотел бы иметь надежную систему для улавливания проблем и предлагать как можно более полезное и актуальное сообщение об ошибке пользователю. Насколько я могу судить, ConnectionError является косвенным decedent BaseException, без каких-либо новых свойств или методов, добавляемых за пределами предложения BaseException. Я не решаюсь просто использовать regex, потому что мне кажется, что я рискую предположить, что все сообщения об ошибках отформатированы одинаково во всех местах.

1 ответов


Я думаю, вы можете получить доступ к нему с помощью e.args[0].reason.errno.

это, вероятно, где-то задокументировано, но обычно, когда мне нужно отследить что-то вроде этого, я просто пробую это на консоли и немного копаю. (Я использую IPython, поэтому легко сделать tab-inspection, но давайте попробуем без него).

во-первых, давайте сгенерируем ошибку, используя

import requests
try:
    requests.get("http://not.a.real.url/really_not")
except requests.exceptions.ConnectionError as e:
    pass

, который должен дать нам об ошибках в e:

>>> e
ConnectionError(MaxRetryError("HTTPConnectionPool(host='not.a.real.url', port=80): Max retries exceeded with url: /really_not (Caused by <class 'socket.gaierror'>: [Errno -2] Name or service not known)",),)

информация обычно находится в args:

>>> e.args
(MaxRetryError("HTTPConnectionPool(host='not.a.real.url', port=80): Max retries exceeded with url: /really_not (Caused by <class 'socket.gaierror'>: [Errno -2] Name or service not known)",),)
>>> e.args[0]
MaxRetryError("HTTPConnectionPool(host='not.a.real.url', port=80): Max retries exceeded with url: /really_not (Caused by <class 'socket.gaierror'>: [Errno -2] Name or service not known)",)

заглянув внутрь, мы видим:

>>> dir(e.args[0])
['__class__', '__delattr__', '__dict__', '__doc__', '__format__', '__getattribute__',
 '__getitem__', '__getslice__', '__hash__', '__init__', '__module__', '__new__',
 '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__setstate__', '__sizeof__',
 '__str__', '__subclasshook__', '__unicode__', '__weakref__', 'args', 'message', 'pool',
 'reason', 'url']

reason выглядит обнадеживающе:

>>> e.args[0].reason
gaierror(-2, 'Name or service not known')
>>> dir(e.args[0].reason)
['__class__', '__delattr__', '__dict__', '__doc__', '__format__', '__getattribute__',
 '__getitem__', '__getslice__', '__hash__', '__init__', '__module__', '__new__',
 '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__setstate__', '__sizeof__',
 '__str__', '__subclasshook__', '__unicode__', '__weakref__', 'args', 'errno', 'filename',
 'message', 'strerror']
>>> e.args[0].reason.errno
-2