test_property.py 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. # Test case for property
  2. # more tests are in test_descr
  3. import sys
  4. import unittest
  5. from test.test_support import run_unittest
  6. class PropertyBase(Exception):
  7. pass
  8. class PropertyGet(PropertyBase):
  9. pass
  10. class PropertySet(PropertyBase):
  11. pass
  12. class PropertyDel(PropertyBase):
  13. pass
  14. class BaseClass(object):
  15. def __init__(self):
  16. self._spam = 5
  17. @property
  18. def spam(self):
  19. """BaseClass.getter"""
  20. return self._spam
  21. @spam.setter
  22. def spam(self, value):
  23. self._spam = value
  24. @spam.deleter
  25. def spam(self):
  26. del self._spam
  27. class SubClass(BaseClass):
  28. @BaseClass.spam.getter
  29. def spam(self):
  30. """SubClass.getter"""
  31. raise PropertyGet(self._spam)
  32. @spam.setter
  33. def spam(self, value):
  34. raise PropertySet(self._spam)
  35. @spam.deleter
  36. def spam(self):
  37. raise PropertyDel(self._spam)
  38. class PropertyDocBase(object):
  39. _spam = 1
  40. def _get_spam(self):
  41. return self._spam
  42. spam = property(_get_spam, doc="spam spam spam")
  43. class PropertyDocSub(PropertyDocBase):
  44. @PropertyDocBase.spam.getter
  45. def spam(self):
  46. """The decorator does not use this doc string"""
  47. return self._spam
  48. class PropertySubNewGetter(BaseClass):
  49. @BaseClass.spam.getter
  50. def spam(self):
  51. """new docstring"""
  52. return 5
  53. class PropertyNewGetter(object):
  54. @property
  55. def spam(self):
  56. """original docstring"""
  57. return 1
  58. @spam.getter
  59. def spam(self):
  60. """new docstring"""
  61. return 8
  62. class PropertyTests(unittest.TestCase):
  63. def test_property_decorator_baseclass(self):
  64. # see #1620
  65. base = BaseClass()
  66. self.assertEqual(base.spam, 5)
  67. self.assertEqual(base._spam, 5)
  68. base.spam = 10
  69. self.assertEqual(base.spam, 10)
  70. self.assertEqual(base._spam, 10)
  71. delattr(base, "spam")
  72. self.assertTrue(not hasattr(base, "spam"))
  73. self.assertTrue(not hasattr(base, "_spam"))
  74. base.spam = 20
  75. self.assertEqual(base.spam, 20)
  76. self.assertEqual(base._spam, 20)
  77. def test_property_decorator_subclass(self):
  78. # see #1620
  79. sub = SubClass()
  80. self.assertRaises(PropertyGet, getattr, sub, "spam")
  81. self.assertRaises(PropertySet, setattr, sub, "spam", None)
  82. self.assertRaises(PropertyDel, delattr, sub, "spam")
  83. @unittest.skipIf(sys.flags.optimize >= 2,
  84. "Docstrings are omitted with -O2 and above")
  85. def test_property_decorator_subclass_doc(self):
  86. sub = SubClass()
  87. self.assertEqual(sub.__class__.spam.__doc__, "SubClass.getter")
  88. @unittest.skipIf(sys.flags.optimize >= 2,
  89. "Docstrings are omitted with -O2 and above")
  90. def test_property_decorator_baseclass_doc(self):
  91. base = BaseClass()
  92. self.assertEqual(base.__class__.spam.__doc__, "BaseClass.getter")
  93. def test_property_decorator_doc(self):
  94. base = PropertyDocBase()
  95. sub = PropertyDocSub()
  96. self.assertEqual(base.__class__.spam.__doc__, "spam spam spam")
  97. self.assertEqual(sub.__class__.spam.__doc__, "spam spam spam")
  98. @unittest.skipIf(sys.flags.optimize >= 2,
  99. "Docstrings are omitted with -O2 and above")
  100. def test_property_getter_doc_override(self):
  101. newgettersub = PropertySubNewGetter()
  102. self.assertEqual(newgettersub.spam, 5)
  103. self.assertEqual(newgettersub.__class__.spam.__doc__, "new docstring")
  104. newgetter = PropertyNewGetter()
  105. self.assertEqual(newgetter.spam, 8)
  106. self.assertEqual(newgetter.__class__.spam.__doc__, "new docstring")
  107. # Issue 5890: subclasses of property do not preserve method __doc__ strings
  108. class PropertySub(property):
  109. """This is a subclass of property"""
  110. class PropertySubSlots(property):
  111. """This is a subclass of property that defines __slots__"""
  112. __slots__ = ()
  113. class PropertySubclassTests(unittest.TestCase):
  114. def test_slots_docstring_copy_exception(self):
  115. try:
  116. class Foo(object):
  117. @PropertySubSlots
  118. def spam(self):
  119. """Trying to copy this docstring will raise an exception"""
  120. return 1
  121. except AttributeError:
  122. pass
  123. else:
  124. raise Exception("AttributeError not raised")
  125. @unittest.skipIf(sys.flags.optimize >= 2,
  126. "Docstrings are omitted with -O2 and above")
  127. def test_docstring_copy(self):
  128. class Foo(object):
  129. @PropertySub
  130. def spam(self):
  131. """spam wrapped in property subclass"""
  132. return 1
  133. self.assertEqual(
  134. Foo.spam.__doc__,
  135. "spam wrapped in property subclass")
  136. @unittest.skipIf(sys.flags.optimize >= 2,
  137. "Docstrings are omitted with -O2 and above")
  138. def test_property_setter_copies_getter_docstring(self):
  139. class Foo(object):
  140. def __init__(self): self._spam = 1
  141. @PropertySub
  142. def spam(self):
  143. """spam wrapped in property subclass"""
  144. return self._spam
  145. @spam.setter
  146. def spam(self, value):
  147. """this docstring is ignored"""
  148. self._spam = value
  149. foo = Foo()
  150. self.assertEqual(foo.spam, 1)
  151. foo.spam = 2
  152. self.assertEqual(foo.spam, 2)
  153. self.assertEqual(
  154. Foo.spam.__doc__,
  155. "spam wrapped in property subclass")
  156. class FooSub(Foo):
  157. @Foo.spam.setter
  158. def spam(self, value):
  159. """another ignored docstring"""
  160. self._spam = 'eggs'
  161. foosub = FooSub()
  162. self.assertEqual(foosub.spam, 1)
  163. foosub.spam = 7
  164. self.assertEqual(foosub.spam, 'eggs')
  165. self.assertEqual(
  166. FooSub.spam.__doc__,
  167. "spam wrapped in property subclass")
  168. @unittest.skipIf(sys.flags.optimize >= 2,
  169. "Docstrings are omitted with -O2 and above")
  170. def test_property_new_getter_new_docstring(self):
  171. class Foo(object):
  172. @PropertySub
  173. def spam(self):
  174. """a docstring"""
  175. return 1
  176. @spam.getter
  177. def spam(self):
  178. """a new docstring"""
  179. return 2
  180. self.assertEqual(Foo.spam.__doc__, "a new docstring")
  181. class FooBase(object):
  182. @PropertySub
  183. def spam(self):
  184. """a docstring"""
  185. return 1
  186. class Foo2(FooBase):
  187. @FooBase.spam.getter
  188. def spam(self):
  189. """a new docstring"""
  190. return 2
  191. self.assertEqual(Foo.spam.__doc__, "a new docstring")
  192. def test_main():
  193. run_unittest(PropertyTests, PropertySubclassTests)
  194. if __name__ == '__main__':
  195. test_main()