blob: 8fed220d22c9be6fe84c9ac9683c96c9ccc249a8 [file] [log] [blame]
Guido van Rossumb5591132007-09-10 22:36:02 +00001# Copyright 2007 Google, Inc. All Rights Reserved.
2# Licensed to PSF under a Contributor Agreement.
3
4"""Unit tests for abc.py."""
5
Guido van Rossumb5591132007-09-10 22:36:02 +00006import unittest
7from test import test_support
8
9import abc
Christian Heimes608c1d82008-03-03 18:28:04 +000010from inspect import isabstract
Guido van Rossumb5591132007-09-10 22:36:02 +000011
12
13class TestABC(unittest.TestCase):
14
15 def test_abstractmethod_basics(self):
16 @abc.abstractmethod
17 def foo(self): pass
18 self.assertEqual(foo.__isabstractmethod__, True)
19 def bar(self): pass
20 self.assertEqual(hasattr(bar, "__isabstractmethod__"), False)
21
22 def test_abstractproperty_basics(self):
23 @abc.abstractproperty
24 def foo(self): pass
25 self.assertEqual(foo.__isabstractmethod__, True)
26 def bar(self): pass
27 self.assertEqual(hasattr(bar, "__isabstractmethod__"), False)
28
29 class C:
30 __metaclass__ = abc.ABCMeta
31 @abc.abstractproperty
32 def foo(self): return 3
33 class D(C):
34 @property
35 def foo(self): return super(D, self).foo
36 self.assertEqual(D().foo, 3)
37
38 def test_abstractmethod_integration(self):
39 for abstractthing in [abc.abstractmethod, abc.abstractproperty]:
40 class C:
41 __metaclass__ = abc.ABCMeta
42 @abstractthing
43 def foo(self): pass # abstract
44 def bar(self): pass # concrete
45 self.assertEqual(C.__abstractmethods__, set(["foo"]))
46 self.assertRaises(TypeError, C) # because foo is abstract
Christian Heimes608c1d82008-03-03 18:28:04 +000047 self.assert_(isabstract(C))
Guido van Rossumb5591132007-09-10 22:36:02 +000048 class D(C):
49 def bar(self): pass # concrete override of concrete
50 self.assertEqual(D.__abstractmethods__, set(["foo"]))
51 self.assertRaises(TypeError, D) # because foo is still abstract
Christian Heimes608c1d82008-03-03 18:28:04 +000052 self.assert_(isabstract(D))
Guido van Rossumb5591132007-09-10 22:36:02 +000053 class E(D):
54 def foo(self): pass
55 self.assertEqual(E.__abstractmethods__, set())
56 E() # now foo is concrete, too
Christian Heimes608c1d82008-03-03 18:28:04 +000057 self.failIf(isabstract(E))
Guido van Rossumb5591132007-09-10 22:36:02 +000058 class F(E):
59 @abstractthing
60 def bar(self): pass # abstract override of concrete
61 self.assertEqual(F.__abstractmethods__, set(["bar"]))
62 self.assertRaises(TypeError, F) # because bar is abstract now
Christian Heimes608c1d82008-03-03 18:28:04 +000063 self.assert_(isabstract(F))
Guido van Rossumb5591132007-09-10 22:36:02 +000064
Jeffrey Yasskinfd1c2452008-01-07 06:09:40 +000065 def test_subclass_oldstyle_class(self):
66 class A:
67 __metaclass__ = abc.ABCMeta
68 class OldstyleClass:
69 pass
70 self.assertFalse(issubclass(OldstyleClass, A))
71 self.assertFalse(issubclass(A, OldstyleClass))
72
Jeffrey Yasskinb9e15f72008-03-17 16:31:21 +000073 def test_isinstance_class(self):
74 class A:
75 __metaclass__ = abc.ABCMeta
76 class OldstyleClass:
77 pass
78 self.assertFalse(isinstance(OldstyleClass, A))
79 self.assertTrue(isinstance(OldstyleClass, type(OldstyleClass)))
80 self.assertFalse(isinstance(A, OldstyleClass))
81 # This raises a recursion depth error, but is low-priority:
82 # self.assertTrue(isinstance(A, abc.ABCMeta))
83
Guido van Rossumb5591132007-09-10 22:36:02 +000084 def test_registration_basics(self):
85 class A:
86 __metaclass__ = abc.ABCMeta
Jeffrey Yasskinfd1c2452008-01-07 06:09:40 +000087 class B(object):
Guido van Rossumb5591132007-09-10 22:36:02 +000088 pass
89 b = B()
90 self.assertEqual(issubclass(B, A), False)
91 self.assertEqual(isinstance(b, A), False)
92 A.register(B)
93 self.assertEqual(issubclass(B, A), True)
94 self.assertEqual(isinstance(b, A), True)
95 class C(B):
96 pass
97 c = C()
98 self.assertEqual(issubclass(C, A), True)
99 self.assertEqual(isinstance(c, A), True)
100
Jeffrey Yasskin57bd60b2008-02-13 17:58:04 +0000101 def test_isinstance_invalidation(self):
102 class A:
103 __metaclass__ = abc.ABCMeta
104 class B(object):
105 pass
106 b = B()
107 self.assertEqual(isinstance(b, A), False)
108 A.register(B)
109 self.assertEqual(isinstance(b, A), True)
110
Guido van Rossumb5591132007-09-10 22:36:02 +0000111 def test_registration_builtins(self):
112 class A:
113 __metaclass__ = abc.ABCMeta
114 A.register(int)
115 self.assertEqual(isinstance(42, A), True)
116 self.assertEqual(issubclass(int, A), True)
117 class B(A):
118 pass
119 B.register(basestring)
120 self.assertEqual(isinstance("", A), True)
121 self.assertEqual(issubclass(str, A), True)
122
123 def test_registration_edge_cases(self):
124 class A:
125 __metaclass__ = abc.ABCMeta
126 A.register(A) # should pass silently
127 class A1(A):
128 pass
129 self.assertRaises(RuntimeError, A1.register, A) # cycles not allowed
Jeffrey Yasskinfd1c2452008-01-07 06:09:40 +0000130 class B(object):
Guido van Rossumb5591132007-09-10 22:36:02 +0000131 pass
132 A1.register(B) # ok
133 A1.register(B) # should pass silently
134 class C(A):
135 pass
136 A.register(C) # should pass silently
137 self.assertRaises(RuntimeError, C.register, A) # cycles not allowed
138 C.register(B) # ok
139
140 def test_registration_transitiveness(self):
141 class A:
142 __metaclass__ = abc.ABCMeta
143 self.failUnless(issubclass(A, A))
144 class B:
145 __metaclass__ = abc.ABCMeta
146 self.failIf(issubclass(A, B))
147 self.failIf(issubclass(B, A))
148 class C:
149 __metaclass__ = abc.ABCMeta
150 A.register(B)
151 class B1(B):
152 pass
153 self.failUnless(issubclass(B1, A))
154 class C1(C):
155 pass
156 B1.register(C1)
157 self.failIf(issubclass(C, B))
158 self.failIf(issubclass(C, B1))
159 self.failUnless(issubclass(C1, A))
160 self.failUnless(issubclass(C1, B))
161 self.failUnless(issubclass(C1, B1))
162 C1.register(int)
163 class MyInt(int):
164 pass
165 self.failUnless(issubclass(MyInt, A))
166 self.failUnless(isinstance(42, A))
167
Guido van Rossum64c06e32007-11-22 00:55:51 +0000168 def test_all_new_methods_are_called(self):
169 class A:
170 __metaclass__ = abc.ABCMeta
Jeffrey Yasskinfd1c2452008-01-07 06:09:40 +0000171 class B(object):
Guido van Rossum64c06e32007-11-22 00:55:51 +0000172 counter = 0
173 def __new__(cls):
174 B.counter += 1
175 return super(B, cls).__new__(cls)
176 class C(A, B):
177 pass
178 self.assertEqual(B.counter, 0)
179 C()
180 self.assertEqual(B.counter, 1)
181
Guido van Rossumb5591132007-09-10 22:36:02 +0000182
183def test_main():
184 test_support.run_unittest(TestABC)
185
186
187if __name__ == "__main__":
188 unittest.main()