|  | # Test case for DynamicClassAttribute | 
|  | # more tests are in test_descr | 
|  |  | 
|  | import abc | 
|  | import sys | 
|  | import unittest | 
|  | from types import DynamicClassAttribute | 
|  |  | 
|  | class PropertyBase(Exception): | 
|  | pass | 
|  |  | 
|  | class PropertyGet(PropertyBase): | 
|  | pass | 
|  |  | 
|  | class PropertySet(PropertyBase): | 
|  | pass | 
|  |  | 
|  | class PropertyDel(PropertyBase): | 
|  | pass | 
|  |  | 
|  | class BaseClass(object): | 
|  | def __init__(self): | 
|  | self._spam = 5 | 
|  |  | 
|  | @DynamicClassAttribute | 
|  | def spam(self): | 
|  | """BaseClass.getter""" | 
|  | return self._spam | 
|  |  | 
|  | @spam.setter | 
|  | def spam(self, value): | 
|  | self._spam = value | 
|  |  | 
|  | @spam.deleter | 
|  | def spam(self): | 
|  | del self._spam | 
|  |  | 
|  | class SubClass(BaseClass): | 
|  |  | 
|  | spam = BaseClass.__dict__['spam'] | 
|  |  | 
|  | @spam.getter | 
|  | def spam(self): | 
|  | """SubClass.getter""" | 
|  | raise PropertyGet(self._spam) | 
|  |  | 
|  | @spam.setter | 
|  | def spam(self, value): | 
|  | raise PropertySet(self._spam) | 
|  |  | 
|  | @spam.deleter | 
|  | def spam(self): | 
|  | raise PropertyDel(self._spam) | 
|  |  | 
|  | class PropertyDocBase(object): | 
|  | _spam = 1 | 
|  | def _get_spam(self): | 
|  | return self._spam | 
|  | spam = DynamicClassAttribute(_get_spam, doc="spam spam spam") | 
|  |  | 
|  | class PropertyDocSub(PropertyDocBase): | 
|  | spam = PropertyDocBase.__dict__['spam'] | 
|  | @spam.getter | 
|  | def spam(self): | 
|  | """The decorator does not use this doc string""" | 
|  | return self._spam | 
|  |  | 
|  | class PropertySubNewGetter(BaseClass): | 
|  | spam = BaseClass.__dict__['spam'] | 
|  | @spam.getter | 
|  | def spam(self): | 
|  | """new docstring""" | 
|  | return 5 | 
|  |  | 
|  | class PropertyNewGetter(object): | 
|  | @DynamicClassAttribute | 
|  | def spam(self): | 
|  | """original docstring""" | 
|  | return 1 | 
|  | @spam.getter | 
|  | def spam(self): | 
|  | """new docstring""" | 
|  | return 8 | 
|  |  | 
|  | class ClassWithAbstractVirtualProperty(metaclass=abc.ABCMeta): | 
|  | @DynamicClassAttribute | 
|  | @abc.abstractmethod | 
|  | def color(): | 
|  | pass | 
|  |  | 
|  | class ClassWithPropertyAbstractVirtual(metaclass=abc.ABCMeta): | 
|  | @abc.abstractmethod | 
|  | @DynamicClassAttribute | 
|  | def color(): | 
|  | pass | 
|  |  | 
|  | class PropertyTests(unittest.TestCase): | 
|  | def test_property_decorator_baseclass(self): | 
|  | # see #1620 | 
|  | base = BaseClass() | 
|  | self.assertEqual(base.spam, 5) | 
|  | self.assertEqual(base._spam, 5) | 
|  | base.spam = 10 | 
|  | self.assertEqual(base.spam, 10) | 
|  | self.assertEqual(base._spam, 10) | 
|  | delattr(base, "spam") | 
|  | self.assertTrue(not hasattr(base, "spam")) | 
|  | self.assertTrue(not hasattr(base, "_spam")) | 
|  | base.spam = 20 | 
|  | self.assertEqual(base.spam, 20) | 
|  | self.assertEqual(base._spam, 20) | 
|  |  | 
|  | def test_property_decorator_subclass(self): | 
|  | # see #1620 | 
|  | sub = SubClass() | 
|  | self.assertRaises(PropertyGet, getattr, sub, "spam") | 
|  | self.assertRaises(PropertySet, setattr, sub, "spam", None) | 
|  | self.assertRaises(PropertyDel, delattr, sub, "spam") | 
|  |  | 
|  | @unittest.skipIf(sys.flags.optimize >= 2, | 
|  | "Docstrings are omitted with -O2 and above") | 
|  | def test_property_decorator_subclass_doc(self): | 
|  | sub = SubClass() | 
|  | self.assertEqual(sub.__class__.__dict__['spam'].__doc__, "SubClass.getter") | 
|  |  | 
|  | @unittest.skipIf(sys.flags.optimize >= 2, | 
|  | "Docstrings are omitted with -O2 and above") | 
|  | def test_property_decorator_baseclass_doc(self): | 
|  | base = BaseClass() | 
|  | self.assertEqual(base.__class__.__dict__['spam'].__doc__, "BaseClass.getter") | 
|  |  | 
|  | def test_property_decorator_doc(self): | 
|  | base = PropertyDocBase() | 
|  | sub = PropertyDocSub() | 
|  | self.assertEqual(base.__class__.__dict__['spam'].__doc__, "spam spam spam") | 
|  | self.assertEqual(sub.__class__.__dict__['spam'].__doc__, "spam spam spam") | 
|  |  | 
|  | @unittest.skipIf(sys.flags.optimize >= 2, | 
|  | "Docstrings are omitted with -O2 and above") | 
|  | def test_property_getter_doc_override(self): | 
|  | newgettersub = PropertySubNewGetter() | 
|  | self.assertEqual(newgettersub.spam, 5) | 
|  | self.assertEqual(newgettersub.__class__.__dict__['spam'].__doc__, "new docstring") | 
|  | newgetter = PropertyNewGetter() | 
|  | self.assertEqual(newgetter.spam, 8) | 
|  | self.assertEqual(newgetter.__class__.__dict__['spam'].__doc__, "new docstring") | 
|  |  | 
|  | def test_property___isabstractmethod__descriptor(self): | 
|  | for val in (True, False, [], [1], '', '1'): | 
|  | class C(object): | 
|  | def foo(self): | 
|  | pass | 
|  | foo.__isabstractmethod__ = val | 
|  | foo = DynamicClassAttribute(foo) | 
|  | self.assertIs(C.__dict__['foo'].__isabstractmethod__, bool(val)) | 
|  |  | 
|  | # check that the DynamicClassAttribute's __isabstractmethod__ descriptor does the | 
|  | # right thing when presented with a value that fails truth testing: | 
|  | class NotBool(object): | 
|  | def __bool__(self): | 
|  | raise ValueError() | 
|  | __len__ = __bool__ | 
|  | with self.assertRaises(ValueError): | 
|  | class C(object): | 
|  | def foo(self): | 
|  | pass | 
|  | foo.__isabstractmethod__ = NotBool() | 
|  | foo = DynamicClassAttribute(foo) | 
|  |  | 
|  | def test_abstract_virtual(self): | 
|  | self.assertRaises(TypeError, ClassWithAbstractVirtualProperty) | 
|  | self.assertRaises(TypeError, ClassWithPropertyAbstractVirtual) | 
|  | class APV(ClassWithPropertyAbstractVirtual): | 
|  | pass | 
|  | self.assertRaises(TypeError, APV) | 
|  | class AVP(ClassWithAbstractVirtualProperty): | 
|  | pass | 
|  | self.assertRaises(TypeError, AVP) | 
|  | class Okay1(ClassWithAbstractVirtualProperty): | 
|  | @DynamicClassAttribute | 
|  | def color(self): | 
|  | return self._color | 
|  | def __init__(self): | 
|  | self._color = 'cyan' | 
|  | with self.assertRaises(AttributeError): | 
|  | Okay1.color | 
|  | self.assertEqual(Okay1().color, 'cyan') | 
|  | class Okay2(ClassWithAbstractVirtualProperty): | 
|  | @DynamicClassAttribute | 
|  | def color(self): | 
|  | return self._color | 
|  | def __init__(self): | 
|  | self._color = 'magenta' | 
|  | with self.assertRaises(AttributeError): | 
|  | Okay2.color | 
|  | self.assertEqual(Okay2().color, 'magenta') | 
|  |  | 
|  |  | 
|  | # Issue 5890: subclasses of DynamicClassAttribute do not preserve method __doc__ strings | 
|  | class PropertySub(DynamicClassAttribute): | 
|  | """This is a subclass of DynamicClassAttribute""" | 
|  |  | 
|  | class PropertySubSlots(DynamicClassAttribute): | 
|  | """This is a subclass of DynamicClassAttribute that defines __slots__""" | 
|  | __slots__ = () | 
|  |  | 
|  | class PropertySubclassTests(unittest.TestCase): | 
|  |  | 
|  | @unittest.skipIf(hasattr(PropertySubSlots, '__doc__'), | 
|  | "__doc__ is already present, __slots__ will have no effect") | 
|  | def test_slots_docstring_copy_exception(self): | 
|  | try: | 
|  | class Foo(object): | 
|  | @PropertySubSlots | 
|  | def spam(self): | 
|  | """Trying to copy this docstring will raise an exception""" | 
|  | return 1 | 
|  | print('\n',spam.__doc__) | 
|  | except AttributeError: | 
|  | pass | 
|  | else: | 
|  | raise Exception("AttributeError not raised") | 
|  |  | 
|  | @unittest.skipIf(sys.flags.optimize >= 2, | 
|  | "Docstrings are omitted with -O2 and above") | 
|  | def test_docstring_copy(self): | 
|  | class Foo(object): | 
|  | @PropertySub | 
|  | def spam(self): | 
|  | """spam wrapped in DynamicClassAttribute subclass""" | 
|  | return 1 | 
|  | self.assertEqual( | 
|  | Foo.__dict__['spam'].__doc__, | 
|  | "spam wrapped in DynamicClassAttribute subclass") | 
|  |  | 
|  | @unittest.skipIf(sys.flags.optimize >= 2, | 
|  | "Docstrings are omitted with -O2 and above") | 
|  | def test_property_setter_copies_getter_docstring(self): | 
|  | class Foo(object): | 
|  | def __init__(self): self._spam = 1 | 
|  | @PropertySub | 
|  | def spam(self): | 
|  | """spam wrapped in DynamicClassAttribute subclass""" | 
|  | return self._spam | 
|  | @spam.setter | 
|  | def spam(self, value): | 
|  | """this docstring is ignored""" | 
|  | self._spam = value | 
|  | foo = Foo() | 
|  | self.assertEqual(foo.spam, 1) | 
|  | foo.spam = 2 | 
|  | self.assertEqual(foo.spam, 2) | 
|  | self.assertEqual( | 
|  | Foo.__dict__['spam'].__doc__, | 
|  | "spam wrapped in DynamicClassAttribute subclass") | 
|  | class FooSub(Foo): | 
|  | spam = Foo.__dict__['spam'] | 
|  | @spam.setter | 
|  | def spam(self, value): | 
|  | """another ignored docstring""" | 
|  | self._spam = 'eggs' | 
|  | foosub = FooSub() | 
|  | self.assertEqual(foosub.spam, 1) | 
|  | foosub.spam = 7 | 
|  | self.assertEqual(foosub.spam, 'eggs') | 
|  | self.assertEqual( | 
|  | FooSub.__dict__['spam'].__doc__, | 
|  | "spam wrapped in DynamicClassAttribute subclass") | 
|  |  | 
|  | @unittest.skipIf(sys.flags.optimize >= 2, | 
|  | "Docstrings are omitted with -O2 and above") | 
|  | def test_property_new_getter_new_docstring(self): | 
|  |  | 
|  | class Foo(object): | 
|  | @PropertySub | 
|  | def spam(self): | 
|  | """a docstring""" | 
|  | return 1 | 
|  | @spam.getter | 
|  | def spam(self): | 
|  | """a new docstring""" | 
|  | return 2 | 
|  | self.assertEqual(Foo.__dict__['spam'].__doc__, "a new docstring") | 
|  | class FooBase(object): | 
|  | @PropertySub | 
|  | def spam(self): | 
|  | """a docstring""" | 
|  | return 1 | 
|  | class Foo2(FooBase): | 
|  | spam = FooBase.__dict__['spam'] | 
|  | @spam.getter | 
|  | def spam(self): | 
|  | """a new docstring""" | 
|  | return 2 | 
|  | self.assertEqual(Foo.__dict__['spam'].__doc__, "a new docstring") | 
|  |  | 
|  |  | 
|  |  | 
|  | if __name__ == '__main__': | 
|  | unittest.main() |