J'ai un problème avec AttributeErrors
élevé dans une @property
en combinaison avec __getattr__()
en python:AttributeErrors: interaction indésirable entre @property et __getattr__
Exemple Code:
>>> def deeply_nested_factory_fn():
... a = 2
... return a.invalid_attr
...
>>> class Test(object):
... def __getattr__(self, name):
... if name == 'abc':
... return 'abc'
... raise AttributeError("'Test' object has no attribute '%s'" % name)
... @property
... def my_prop(self):
... return deeply_nested_factory_fn()
...
>>> test = Test()
>>> test.my_prop
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 5, in __getattr__
AttributeError: 'Test' object has no attribute 'my_prop'
Dans mon cas, c'est un message d'erreur très trompeur, car il cache le fait que deeply_nested_factory_fn()
a une erreur. Sur la base de l'idée contenue dans la réponse de Tadhg McDonald-Jensen, ma meilleure solution actuelle est la suivante. Tous les conseils sur la façon de se débarrasser du préfixe __main__.
à AttributeError
et la référence à attributeErrorCatcher
dans le retraçage serait très appréciée.
>>> def catchAttributeErrors(func):
... AttributeError_org = AttributeError
... def attributeErrorCatcher(*args, **kwargs):
... try:
... return func(*args, **kwargs)
... except AttributeError_org as e:
... import sys
... class AttributeError(Exception):
... pass
... etype, value, tb = sys.exc_info()
... raise AttributeError(e).with_traceback(tb.tb_next) from None
... return attributeErrorCatcher
...
>>> def deeply_nested_factory_fn():
... a = 2
... return a.invalid_attr
...
>>> class Test(object):
... def __getattr__(self, name):
... if name == 'abc':
... # computing come other attributes
... return 'abc'
... raise AttributeError("'Test' object has no attribute '%s'" % name)
... @property
... @catchAttributeErrors
... def my_prop(self):
... return deeply_nested_factory_fn()
...
>>> class Test1(object):
... def __init__(self):
... test = Test()
... test.my_prop
...
>>> test1 = Test1()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 4, in __init__
File "<stdin>", line 11, in attributeErrorCatcher
File "<stdin>", line 10, in my_prop
File "<stdin>", line 3, in deeply_nested_factory_fn
__main__.AttributeError: 'int' object has no attribute 'invalid_attr'
vous devez faire '__qualname__ =" AttributeError "' dans la définition de la classe pour supprimer la partie '__main__' mais croyez-moi, vous ne ** voulez pas que l'erreur indique simplement" AttributeError "car vous risquez d'être dérouté pourquoi le diable 'except AttributeError' n'a pas attrapé une AttributeError. –