Наилучшая практика для обработки исключений из библиотек, импортированных другими библиотеками в Python? - PullRequest
4 голосов
/ 16 июня 2011

Каков подходящий способ обработки исключений из библиотек, импортированных другими библиотеками в Python?

Например, у меня есть библиотека с именем "pycontrol", которую я импортирую в основную программу.«pycontrol» импортирует библиотеку «suds».Библиотека "suds", в свою очередь, импортирует библиотеку "urllib2".Я заметил, что когда библиотека «suds» испытывает проблемы с подключением к удаленным ресурсам, она получает доступ через «urllib2», эти исключения возникают в моей основной программе.

На данный момент я предпочитаю импортировать urllib2 и suds в мое глобальное пространство имен и отлавливать типичные исключения, которые генерируются ими и не обрабатываются в "pycontrol".

Есть ли какой-то другой лучший способ, как можно подойти к этому?

Основная идея о том, как выглядит фрагмент кода (без импорта suds или urllib2 в глобальное пространство имен):

    import pycontrol.pycontrol as pc

    print "Connecting to iControl API on LTM %s..." % ltm
    try:
        b = pc.BIGIP(hostname=ltm, username=user, password=pw, 
            wsdls=wsdl_list, fromurl=True,
            debug=soap_debug)
    except (<whattocatch>), detail:
        print "Error: could not connect to iControl API on LTM %s... aborting!" % ltm
        print "Details: %s" % detail
        exitcode = 1
    else:
        print "Connection successfully established."

Вот пример трассировки:

Connecting to iControl API on LTM s0-bigip1-lb2.lab.zynga.com...
Traceback (most recent call last):
  File "./register.py", line 507, in <module>
    main()
  File "./register.py", line 415, in main
    b = build_bigip_object(ltm, user, pw, WSDLS, soap_debug = False)
  File "./register.py", line 85, in build_bigip_object
    debug=soap_debug)
  File "build/bdist.macosx-10.6-universal/egg/pycontrol/pycontrol.py", line 81, in __init__
  File "build/bdist.macosx-10.6-universal/egg/pycontrol/pycontrol.py", line 103, in _get_clients
  File "build/bdist.macosx-10.6-universal/egg/pycontrol/pycontrol.py", line 149, in _get_suds_client
  File "/Library/Python/2.6/site-packages/suds/client.py", line 111, in __init__
    self.wsdl = reader.open(url)
  File "/Library/Python/2.6/site-packages/suds/reader.py", line 136, in open
    d = self.fn(url, self.options)
  File "/Library/Python/2.6/site-packages/suds/wsdl.py", line 136, in __init__
    d = reader.open(url)
  File "/Library/Python/2.6/site-packages/suds/reader.py", line 73, in open
    d = self.download(url)
  File "/Library/Python/2.6/site-packages/suds/reader.py", line 88, in download
    fp = self.options.transport.open(Request(url))
  File "/Library/Python/2.6/site-packages/suds/transport/https.py", line 60, in open
    return  HttpTransport.open(self, request)
  File "/Library/Python/2.6/site-packages/suds/transport/http.py", line 62, in open
    return self.u2open(u2request)
  File "/Library/Python/2.6/site-packages/suds/transport/http.py", line 118, in u2open
    return url.open(u2request, timeout=tm)
  File "/System/Library/Frameworks/Python.framework/Versions/2.6/lib/python2.6/urllib2.py", line 383, in open
    response = self._open(req, data)
  File "/System/Library/Frameworks/Python.framework/Versions/2.6/lib/python2.6/urllib2.py", line 401, in _open
    '_open', req)
  File "/System/Library/Frameworks/Python.framework/Versions/2.6/lib/python2.6/urllib2.py", line 361, in _call_chain
    result = func(*args)
  File "/System/Library/Frameworks/Python.framework/Versions/2.6/lib/python2.6/urllib2.py", line 1138, in https_open
    return self.do_open(httplib.HTTPSConnection, req)
  File "/System/Library/Frameworks/Python.framework/Versions/2.6/lib/python2.6/urllib2.py", line 1105, in do_open
    raise URLError(err)
urllib2.URLError: <urlopen error [Errno 8] nodename nor servname provided, or not known>

Ответы [ 2 ]

2 голосов
/ 16 июня 2011

Я думаю, вы сами ответили на свой вопрос.Импортируйте urllib2 и перехватывайте исключение в вашем модуле.

from urllib2 import URLError

try:
    # something
except URLError, e:
    # Do something in case of error.
0 голосов
/ 16 июня 2011

Зачем вам вообще нужно ловить определенные исключения? В конце концов, любое исключение (не только URLError), взятое из b = pc.BIGIP(...), означает, что вы не можете продолжать.

Я предлагаю:

import traceback

try:
    b = pc.BIGIP(...)
except:
    traceback.print_exc()
    exitcode = 1
else:
    do_something_with_connection(b)

Другая идея: зачем вообще ловить исключение? Интерпретатор Python выдаст трассировку стека в stderr и выйдет из программы, когда обнаружит необработанное исключение:

b = bc.BIGIP(...)
do_something_with_connection(b)

Или, если вам нужно записать в журнал ошибок:

import logging
import sys

def main():
   b = bc.BIGIP(...)
   do_something_with_connection(b)

if __name__ == "__main__":
    try:
        main()
    except:
        logging.exception("An unexpected error occured")
        sys.exit(1)
...