Raymond Hettinger | d1ef854 | 2008-01-11 00:23:13 +0000 | [diff] [blame] | 1 | import unittest, doctest |
Raymond Hettinger | c37e5e0 | 2007-03-01 06:16:43 +0000 | [diff] [blame] | 2 | from test import test_support |
Raymond Hettinger | 01a0957 | 2007-10-23 20:37:41 +0000 | [diff] [blame] | 3 | from collections import namedtuple |
Guido van Rossum | 64c06e3 | 2007-11-22 00:55:51 +0000 | [diff] [blame] | 4 | from collections import Hashable, Iterable, Iterator |
| 5 | from collections import Sized, Container, Callable |
| 6 | from collections import Set, MutableSet |
| 7 | from collections import Mapping, MutableMapping |
| 8 | from collections import Sequence, MutableSequence |
| 9 | |
Raymond Hettinger | c37e5e0 | 2007-03-01 06:16:43 +0000 | [diff] [blame] | 10 | |
| 11 | class TestNamedTuple(unittest.TestCase): |
| 12 | |
| 13 | def test_factory(self): |
Raymond Hettinger | 01a0957 | 2007-10-23 20:37:41 +0000 | [diff] [blame] | 14 | Point = namedtuple('Point', 'x y') |
Raymond Hettinger | c37e5e0 | 2007-03-01 06:16:43 +0000 | [diff] [blame] | 15 | self.assertEqual(Point.__name__, 'Point') |
| 16 | self.assertEqual(Point.__doc__, 'Point(x, y)') |
| 17 | self.assertEqual(Point.__slots__, ()) |
| 18 | self.assertEqual(Point.__module__, __name__) |
| 19 | self.assertEqual(Point.__getitem__, tuple.__getitem__) |
Raymond Hettinger | e0734e7 | 2008-01-04 03:22:53 +0000 | [diff] [blame] | 20 | self.assertEqual(Point._fields, ('x', 'y')) |
Raymond Hettinger | abfd8df | 2007-10-16 21:28:32 +0000 | [diff] [blame] | 21 | |
Raymond Hettinger | 01a0957 | 2007-10-23 20:37:41 +0000 | [diff] [blame] | 22 | self.assertRaises(ValueError, namedtuple, 'abc%', 'efg ghi') # type has non-alpha char |
| 23 | self.assertRaises(ValueError, namedtuple, 'class', 'efg ghi') # type has keyword |
| 24 | self.assertRaises(ValueError, namedtuple, '9abc', 'efg ghi') # type starts with digit |
Raymond Hettinger | abfd8df | 2007-10-16 21:28:32 +0000 | [diff] [blame] | 25 | |
Raymond Hettinger | 01a0957 | 2007-10-23 20:37:41 +0000 | [diff] [blame] | 26 | self.assertRaises(ValueError, namedtuple, 'abc', 'efg g%hi') # field with non-alpha char |
| 27 | self.assertRaises(ValueError, namedtuple, 'abc', 'abc class') # field has keyword |
| 28 | self.assertRaises(ValueError, namedtuple, 'abc', '8efg 9ghi') # field starts with digit |
Raymond Hettinger | 42da874 | 2007-12-14 02:49:47 +0000 | [diff] [blame] | 29 | self.assertRaises(ValueError, namedtuple, 'abc', '_efg ghi') # field with leading underscore |
Raymond Hettinger | 01a0957 | 2007-10-23 20:37:41 +0000 | [diff] [blame] | 30 | self.assertRaises(ValueError, namedtuple, 'abc', 'efg efg ghi') # duplicate field |
Raymond Hettinger | abfd8df | 2007-10-16 21:28:32 +0000 | [diff] [blame] | 31 | |
Raymond Hettinger | 01a0957 | 2007-10-23 20:37:41 +0000 | [diff] [blame] | 32 | namedtuple('Point0', 'x1 y2') # Verify that numbers are allowed in names |
Raymond Hettinger | 42da874 | 2007-12-14 02:49:47 +0000 | [diff] [blame] | 33 | namedtuple('_', 'a b c') # Test leading underscores in a typename |
Raymond Hettinger | c37e5e0 | 2007-03-01 06:16:43 +0000 | [diff] [blame] | 34 | |
Raymond Hettinger | 02740f7 | 2008-01-05 01:35:43 +0000 | [diff] [blame] | 35 | self.assertRaises(TypeError, Point._make, [11]) # catch too few args |
| 36 | self.assertRaises(TypeError, Point._make, [11, 22, 33]) # catch too many args |
| 37 | |
Raymond Hettinger | c37e5e0 | 2007-03-01 06:16:43 +0000 | [diff] [blame] | 38 | def test_instance(self): |
Raymond Hettinger | 01a0957 | 2007-10-23 20:37:41 +0000 | [diff] [blame] | 39 | Point = namedtuple('Point', 'x y') |
Raymond Hettinger | c37e5e0 | 2007-03-01 06:16:43 +0000 | [diff] [blame] | 40 | p = Point(11, 22) |
| 41 | self.assertEqual(p, Point(x=11, y=22)) |
| 42 | self.assertEqual(p, Point(11, y=22)) |
| 43 | self.assertEqual(p, Point(y=22, x=11)) |
| 44 | self.assertEqual(p, Point(*(11, 22))) |
| 45 | self.assertEqual(p, Point(**dict(x=11, y=22))) |
| 46 | self.assertRaises(TypeError, Point, 1) # too few args |
| 47 | self.assertRaises(TypeError, Point, 1, 2, 3) # too many args |
| 48 | self.assertRaises(TypeError, eval, 'Point(XXX=1, y=2)', locals()) # wrong keyword argument |
| 49 | self.assertRaises(TypeError, eval, 'Point(x=1)', locals()) # missing keyword argument |
| 50 | self.assertEqual(repr(p), 'Point(x=11, y=22)') |
| 51 | self.assert_('__dict__' not in dir(p)) # verify instance has no dict |
| 52 | self.assert_('__weakref__' not in dir(p)) |
Raymond Hettinger | 02740f7 | 2008-01-05 01:35:43 +0000 | [diff] [blame] | 53 | self.assertEqual(p, Point._make([11, 22])) # test _make classmethod |
Raymond Hettinger | 42da874 | 2007-12-14 02:49:47 +0000 | [diff] [blame] | 54 | self.assertEqual(p._fields, ('x', 'y')) # test _fields attribute |
| 55 | self.assertEqual(p._replace(x=1), (1, 22)) # test _replace method |
| 56 | self.assertEqual(p._asdict(), dict(x=11, y=22)) # test _asdict method |
Raymond Hettinger | d36a60e | 2007-09-17 00:55:00 +0000 | [diff] [blame] | 57 | |
Raymond Hettinger | 1b50fd7 | 2008-01-05 02:17:24 +0000 | [diff] [blame] | 58 | try: |
| 59 | p._replace(x=1, error=2) |
| 60 | except ValueError: |
| 61 | pass |
| 62 | else: |
| 63 | self._fail('Did not detect an incorrect fieldname') |
| 64 | |
Raymond Hettinger | d36a60e | 2007-09-17 00:55:00 +0000 | [diff] [blame] | 65 | # verify that field string can have commas |
Raymond Hettinger | 01a0957 | 2007-10-23 20:37:41 +0000 | [diff] [blame] | 66 | Point = namedtuple('Point', 'x, y') |
Raymond Hettinger | d36a60e | 2007-09-17 00:55:00 +0000 | [diff] [blame] | 67 | p = Point(x=11, y=22) |
| 68 | self.assertEqual(repr(p), 'Point(x=11, y=22)') |
Raymond Hettinger | c37e5e0 | 2007-03-01 06:16:43 +0000 | [diff] [blame] | 69 | |
Raymond Hettinger | 2115bbc | 2007-10-08 09:14:28 +0000 | [diff] [blame] | 70 | # verify that fieldspec can be a non-string sequence |
Raymond Hettinger | 01a0957 | 2007-10-23 20:37:41 +0000 | [diff] [blame] | 71 | Point = namedtuple('Point', ('x', 'y')) |
Raymond Hettinger | 2115bbc | 2007-10-08 09:14:28 +0000 | [diff] [blame] | 72 | p = Point(x=11, y=22) |
| 73 | self.assertEqual(repr(p), 'Point(x=11, y=22)') |
| 74 | |
Raymond Hettinger | c37e5e0 | 2007-03-01 06:16:43 +0000 | [diff] [blame] | 75 | def test_tupleness(self): |
Raymond Hettinger | 01a0957 | 2007-10-23 20:37:41 +0000 | [diff] [blame] | 76 | Point = namedtuple('Point', 'x y') |
Raymond Hettinger | c37e5e0 | 2007-03-01 06:16:43 +0000 | [diff] [blame] | 77 | p = Point(11, 22) |
| 78 | |
| 79 | self.assert_(isinstance(p, tuple)) |
| 80 | self.assertEqual(p, (11, 22)) # matches a real tuple |
| 81 | self.assertEqual(tuple(p), (11, 22)) # coercable to a real tuple |
| 82 | self.assertEqual(list(p), [11, 22]) # coercable to a list |
| 83 | self.assertEqual(max(p), 22) # iterable |
| 84 | self.assertEqual(max(*p), 22) # star-able |
| 85 | x, y = p |
| 86 | self.assertEqual(p, (x, y)) # unpacks like a tuple |
| 87 | self.assertEqual((p[0], p[1]), (11, 22)) # indexable like a tuple |
| 88 | self.assertRaises(IndexError, p.__getitem__, 3) |
| 89 | |
| 90 | self.assertEqual(p.x, x) |
| 91 | self.assertEqual(p.y, y) |
| 92 | self.assertRaises(AttributeError, eval, 'p.z', locals()) |
| 93 | |
Raymond Hettinger | 2b03d45 | 2007-09-18 03:33:19 +0000 | [diff] [blame] | 94 | def test_odd_sizes(self): |
Raymond Hettinger | 01a0957 | 2007-10-23 20:37:41 +0000 | [diff] [blame] | 95 | Zero = namedtuple('Zero', '') |
Raymond Hettinger | 2b03d45 | 2007-09-18 03:33:19 +0000 | [diff] [blame] | 96 | self.assertEqual(Zero(), ()) |
Raymond Hettinger | 02740f7 | 2008-01-05 01:35:43 +0000 | [diff] [blame] | 97 | self.assertEqual(Zero._make([]), ()) |
Raymond Hettinger | 88880b2 | 2007-12-18 00:13:45 +0000 | [diff] [blame] | 98 | self.assertEqual(repr(Zero()), 'Zero()') |
| 99 | self.assertEqual(Zero()._asdict(), {}) |
| 100 | self.assertEqual(Zero()._fields, ()) |
| 101 | |
Raymond Hettinger | 01a0957 | 2007-10-23 20:37:41 +0000 | [diff] [blame] | 102 | Dot = namedtuple('Dot', 'd') |
Raymond Hettinger | 2b03d45 | 2007-09-18 03:33:19 +0000 | [diff] [blame] | 103 | self.assertEqual(Dot(1), (1,)) |
Raymond Hettinger | 02740f7 | 2008-01-05 01:35:43 +0000 | [diff] [blame] | 104 | self.assertEqual(Dot._make([1]), (1,)) |
Raymond Hettinger | 88880b2 | 2007-12-18 00:13:45 +0000 | [diff] [blame] | 105 | self.assertEqual(Dot(1).d, 1) |
| 106 | self.assertEqual(repr(Dot(1)), 'Dot(d=1)') |
| 107 | self.assertEqual(Dot(1)._asdict(), {'d':1}) |
| 108 | self.assertEqual(Dot(1)._replace(d=999), (999,)) |
| 109 | self.assertEqual(Dot(1)._fields, ('d',)) |
Raymond Hettinger | 2b03d45 | 2007-09-18 03:33:19 +0000 | [diff] [blame] | 110 | |
Raymond Hettinger | 88880b2 | 2007-12-18 00:13:45 +0000 | [diff] [blame] | 111 | n = 10000 |
| 112 | import string, random |
| 113 | names = [''.join([random.choice(string.letters) for j in range(10)]) for i in range(n)] |
| 114 | Big = namedtuple('Big', names) |
| 115 | b = Big(*range(n)) |
| 116 | self.assertEqual(b, tuple(range(n))) |
Raymond Hettinger | 02740f7 | 2008-01-05 01:35:43 +0000 | [diff] [blame] | 117 | self.assertEqual(Big._make(range(n)), tuple(range(n))) |
Raymond Hettinger | 88880b2 | 2007-12-18 00:13:45 +0000 | [diff] [blame] | 118 | for pos, name in enumerate(names): |
| 119 | self.assertEqual(getattr(b, name), pos) |
| 120 | repr(b) # make sure repr() doesn't blow-up |
| 121 | d = b._asdict() |
| 122 | d_expected = dict(zip(names, range(n))) |
| 123 | self.assertEqual(d, d_expected) |
| 124 | b2 = b._replace(**dict([(names[1], 999),(names[-5], 42)])) |
| 125 | b2_expected = range(n) |
| 126 | b2_expected[1] = 999 |
| 127 | b2_expected[-5] = 42 |
| 128 | self.assertEqual(b2, tuple(b2_expected)) |
| 129 | self.assertEqual(b._fields, tuple(names)) |
Guido van Rossum | 64c06e3 | 2007-11-22 00:55:51 +0000 | [diff] [blame] | 130 | |
| 131 | class TestOneTrickPonyABCs(unittest.TestCase): |
| 132 | |
| 133 | def test_Hashable(self): |
| 134 | # Check some non-hashables |
| 135 | non_samples = [list(), set(), dict()] |
| 136 | for x in non_samples: |
| 137 | self.failIf(isinstance(x, Hashable), repr(x)) |
| 138 | self.failIf(issubclass(type(x), Hashable), repr(type(x))) |
| 139 | # Check some hashables |
| 140 | samples = [None, |
| 141 | int(), float(), complex(), |
| 142 | str(), |
| 143 | tuple(), frozenset(), |
| 144 | int, list, object, type, |
| 145 | ] |
| 146 | for x in samples: |
| 147 | self.failUnless(isinstance(x, Hashable), repr(x)) |
| 148 | self.failUnless(issubclass(type(x), Hashable), repr(type(x))) |
| 149 | self.assertRaises(TypeError, Hashable) |
| 150 | # Check direct subclassing |
| 151 | class H(Hashable): |
| 152 | def __hash__(self): |
| 153 | return super(H, self).__hash__() |
| 154 | self.assertEqual(hash(H()), 0) |
| 155 | self.failIf(issubclass(int, H)) |
| 156 | |
| 157 | def test_Iterable(self): |
| 158 | # Check some non-iterables |
| 159 | non_samples = [None, 42, 3.14, 1j] |
| 160 | for x in non_samples: |
| 161 | self.failIf(isinstance(x, Iterable), repr(x)) |
| 162 | self.failIf(issubclass(type(x), Iterable), repr(type(x))) |
| 163 | # Check some iterables |
| 164 | samples = [str(), |
| 165 | tuple(), list(), set(), frozenset(), dict(), |
| 166 | dict().keys(), dict().items(), dict().values(), |
| 167 | (lambda: (yield))(), |
| 168 | (x for x in []), |
| 169 | ] |
| 170 | for x in samples: |
| 171 | self.failUnless(isinstance(x, Iterable), repr(x)) |
| 172 | self.failUnless(issubclass(type(x), Iterable), repr(type(x))) |
| 173 | # Check direct subclassing |
| 174 | class I(Iterable): |
| 175 | def __iter__(self): |
| 176 | return super(I, self).__iter__() |
| 177 | self.assertEqual(list(I()), []) |
| 178 | self.failIf(issubclass(str, I)) |
| 179 | |
| 180 | def test_Iterator(self): |
| 181 | non_samples = [None, 42, 3.14, 1j, "".encode('ascii'), "", (), [], |
| 182 | {}, set()] |
| 183 | for x in non_samples: |
| 184 | self.failIf(isinstance(x, Iterator), repr(x)) |
| 185 | self.failIf(issubclass(type(x), Iterator), repr(type(x))) |
| 186 | samples = [iter(str()), |
| 187 | iter(tuple()), iter(list()), iter(dict()), |
| 188 | iter(set()), iter(frozenset()), |
| 189 | iter(dict().keys()), iter(dict().items()), |
| 190 | iter(dict().values()), |
| 191 | (lambda: (yield))(), |
| 192 | (x for x in []), |
| 193 | ] |
| 194 | for x in samples: |
| 195 | self.failUnless(isinstance(x, Iterator), repr(x)) |
| 196 | self.failUnless(issubclass(type(x), Iterator), repr(type(x))) |
| 197 | |
| 198 | def test_Sized(self): |
| 199 | non_samples = [None, 42, 3.14, 1j, |
| 200 | (lambda: (yield))(), |
| 201 | (x for x in []), |
| 202 | ] |
| 203 | for x in non_samples: |
| 204 | self.failIf(isinstance(x, Sized), repr(x)) |
| 205 | self.failIf(issubclass(type(x), Sized), repr(type(x))) |
| 206 | samples = [str(), |
| 207 | tuple(), list(), set(), frozenset(), dict(), |
| 208 | dict().keys(), dict().items(), dict().values(), |
| 209 | ] |
| 210 | for x in samples: |
| 211 | self.failUnless(isinstance(x, Sized), repr(x)) |
| 212 | self.failUnless(issubclass(type(x), Sized), repr(type(x))) |
| 213 | |
| 214 | def test_Container(self): |
| 215 | non_samples = [None, 42, 3.14, 1j, |
| 216 | (lambda: (yield))(), |
| 217 | (x for x in []), |
| 218 | ] |
| 219 | for x in non_samples: |
| 220 | self.failIf(isinstance(x, Container), repr(x)) |
| 221 | self.failIf(issubclass(type(x), Container), repr(type(x))) |
| 222 | samples = [str(), |
| 223 | tuple(), list(), set(), frozenset(), dict(), |
| 224 | dict().keys(), dict().items(), |
| 225 | ] |
| 226 | for x in samples: |
| 227 | self.failUnless(isinstance(x, Container), repr(x)) |
| 228 | self.failUnless(issubclass(type(x), Container), repr(type(x))) |
| 229 | |
| 230 | def test_Callable(self): |
| 231 | non_samples = [None, 42, 3.14, 1j, |
| 232 | "", "".encode('ascii'), (), [], {}, set(), |
| 233 | (lambda: (yield))(), |
| 234 | (x for x in []), |
| 235 | ] |
| 236 | for x in non_samples: |
| 237 | self.failIf(isinstance(x, Callable), repr(x)) |
| 238 | self.failIf(issubclass(type(x), Callable), repr(type(x))) |
| 239 | samples = [lambda: None, |
| 240 | type, int, object, |
| 241 | len, |
| 242 | list.append, [].append, |
| 243 | ] |
| 244 | for x in samples: |
| 245 | self.failUnless(isinstance(x, Callable), repr(x)) |
| 246 | self.failUnless(issubclass(type(x), Callable), repr(type(x))) |
| 247 | |
| 248 | def test_direct_subclassing(self): |
| 249 | for B in Hashable, Iterable, Iterator, Sized, Container, Callable: |
| 250 | class C(B): |
| 251 | pass |
| 252 | self.failUnless(issubclass(C, B)) |
| 253 | self.failIf(issubclass(int, C)) |
| 254 | |
| 255 | def test_registration(self): |
| 256 | for B in Hashable, Iterable, Iterator, Sized, Container, Callable: |
| 257 | class C: |
| 258 | __metaclass__ = type |
| 259 | __hash__ = None # Make sure it isn't hashable by default |
| 260 | self.failIf(issubclass(C, B), B.__name__) |
| 261 | B.register(C) |
| 262 | self.failUnless(issubclass(C, B)) |
| 263 | |
| 264 | |
| 265 | class TestCollectionABCs(unittest.TestCase): |
| 266 | |
| 267 | # XXX For now, we only test some virtual inheritance properties. |
| 268 | # We should also test the proper behavior of the collection ABCs |
| 269 | # as real base classes or mix-in classes. |
| 270 | |
| 271 | def test_Set(self): |
| 272 | for sample in [set, frozenset]: |
| 273 | self.failUnless(isinstance(sample(), Set)) |
| 274 | self.failUnless(issubclass(sample, Set)) |
| 275 | |
| 276 | def test_MutableSet(self): |
| 277 | self.failUnless(isinstance(set(), MutableSet)) |
| 278 | self.failUnless(issubclass(set, MutableSet)) |
| 279 | self.failIf(isinstance(frozenset(), MutableSet)) |
| 280 | self.failIf(issubclass(frozenset, MutableSet)) |
| 281 | |
| 282 | def test_Mapping(self): |
| 283 | for sample in [dict]: |
| 284 | self.failUnless(isinstance(sample(), Mapping)) |
| 285 | self.failUnless(issubclass(sample, Mapping)) |
| 286 | |
| 287 | def test_MutableMapping(self): |
| 288 | for sample in [dict]: |
| 289 | self.failUnless(isinstance(sample(), MutableMapping)) |
| 290 | self.failUnless(issubclass(sample, MutableMapping)) |
| 291 | |
| 292 | def test_Sequence(self): |
| 293 | for sample in [tuple, list, str]: |
| 294 | self.failUnless(isinstance(sample(), Sequence)) |
| 295 | self.failUnless(issubclass(sample, Sequence)) |
| 296 | self.failUnless(issubclass(basestring, Sequence)) |
| 297 | |
| 298 | def test_MutableSequence(self): |
| 299 | for sample in [tuple, str]: |
| 300 | self.failIf(isinstance(sample(), MutableSequence)) |
| 301 | self.failIf(issubclass(sample, MutableSequence)) |
| 302 | for sample in [list]: |
| 303 | self.failUnless(isinstance(sample(), MutableSequence)) |
| 304 | self.failUnless(issubclass(sample, MutableSequence)) |
| 305 | self.failIf(issubclass(basestring, MutableSequence)) |
| 306 | |
Raymond Hettinger | d1ef854 | 2008-01-11 00:23:13 +0000 | [diff] [blame] | 307 | import doctest, collections |
Guido van Rossum | 64c06e3 | 2007-11-22 00:55:51 +0000 | [diff] [blame] | 308 | |
Raymond Hettinger | c37e5e0 | 2007-03-01 06:16:43 +0000 | [diff] [blame] | 309 | def test_main(verbose=None): |
Amaury Forgeot d'Arc | cb0f2ad | 2008-04-02 00:55:04 +0000 | [diff] [blame^] | 310 | NamedTupleDocs = doctest.DocTestSuite(module=collections) |
Raymond Hettinger | d1ef854 | 2008-01-11 00:23:13 +0000 | [diff] [blame] | 311 | test_classes = [TestNamedTuple, NamedTupleDocs, TestOneTrickPonyABCs, TestCollectionABCs] |
Raymond Hettinger | c37e5e0 | 2007-03-01 06:16:43 +0000 | [diff] [blame] | 312 | test_support.run_unittest(*test_classes) |
Amaury Forgeot d'Arc | cb0f2ad | 2008-04-02 00:55:04 +0000 | [diff] [blame^] | 313 | test_support.run_doctest(collections, verbose) |
Raymond Hettinger | c37e5e0 | 2007-03-01 06:16:43 +0000 | [diff] [blame] | 314 | |
| 315 | if __name__ == "__main__": |
| 316 | test_main(verbose=True) |