blob: 342e77505973865b41f2c6f0063d6ee8acd05e7c [file] [log] [blame]
Walter Dörwald59b23e82004-09-30 13:46:00 +00001import unittest
2from test import test_support
3
4import sys, UserDict, cStringIO
5
6
7class DictTest(unittest.TestCase):
8 def test_constructor(self):
9 # calling built-in types without argument must return empty
10 self.assertEqual(dict(), {})
11 self.assert_(dict() is not {})
12
13 def test_bool(self):
14 self.assert_(not {})
15 self.assert_({1: 2})
16 self.assert_(bool({}) is False)
17 self.assert_(bool({1: 2}) is True)
18
19 def test_keys(self):
20 d = {}
21 self.assertEqual(d.keys(), [])
22 d = {'a': 1, 'b': 2}
23 k = d.keys()
24 self.assert_(d.has_key('a'))
25 self.assert_(d.has_key('b'))
26
27 self.assertRaises(TypeError, d.keys, None)
28
29 def test_values(self):
30 d = {}
31 self.assertEqual(d.values(), [])
32 d = {1:2}
33 self.assertEqual(d.values(), [2])
34
35 self.assertRaises(TypeError, d.values, None)
36
37 def test_items(self):
38 d = {}
39 self.assertEqual(d.items(), [])
40
41 d = {1:2}
42 self.assertEqual(d.items(), [(1, 2)])
43
44 self.assertRaises(TypeError, d.items, None)
45
46 def test_has_key(self):
47 d = {}
48 self.assert_(not d.has_key('a'))
49 d = {'a': 1, 'b': 2}
50 k = d.keys()
51 k.sort()
52 self.assertEqual(k, ['a', 'b'])
53
54 self.assertRaises(TypeError, d.has_key)
55
56 def test_contains(self):
57 d = {}
58 self.assert_(not ('a' in d))
59 self.assert_('a' not in d)
60 d = {'a': 1, 'b': 2}
61 self.assert_('a' in d)
62 self.assert_('b' in d)
63 self.assert_('c' not in d)
64
65 self.assertRaises(TypeError, d.__contains__)
66
67 def test_len(self):
68 d = {}
69 self.assertEqual(len(d), 0)
70 d = {'a': 1, 'b': 2}
71 self.assertEqual(len(d), 2)
72
73 def test_getitem(self):
74 d = {'a': 1, 'b': 2}
75 self.assertEqual(d['a'], 1)
76 self.assertEqual(d['b'], 2)
77 d['c'] = 3
78 d['a'] = 4
79 self.assertEqual(d['c'], 3)
80 self.assertEqual(d['a'], 4)
81 del d['b']
82 self.assertEqual(d, {'a': 4, 'c': 3})
83
84 self.assertRaises(TypeError, d.__getitem__)
85
86 class BadEq(object):
87 def __eq__(self, other):
88 raise Exc()
Guido van Rossum64c06e32007-11-22 00:55:51 +000089 def __hash__(self):
90 return 24
Walter Dörwald59b23e82004-09-30 13:46:00 +000091
92 d = {}
93 d[BadEq()] = 42
94 self.assertRaises(KeyError, d.__getitem__, 23)
95
96 class Exc(Exception): pass
97
98 class BadHash(object):
99 fail = False
100 def __hash__(self):
101 if self.fail:
102 raise Exc()
103 else:
104 return 42
105
106 x = BadHash()
107 d[x] = 42
108 x.fail = True
109 self.assertRaises(Exc, d.__getitem__, x)
110
111 def test_clear(self):
112 d = {1:1, 2:2, 3:3}
113 d.clear()
114 self.assertEqual(d, {})
115
116 self.assertRaises(TypeError, d.clear, None)
117
118 def test_update(self):
119 d = {}
120 d.update({1:100})
121 d.update({2:20})
122 d.update({1:1, 2:2, 3:3})
123 self.assertEqual(d, {1:1, 2:2, 3:3})
124
125 d.update()
126 self.assertEqual(d, {1:1, 2:2, 3:3})
127
128 self.assertRaises((TypeError, AttributeError), d.update, None)
129
130 class SimpleUserDict:
131 def __init__(self):
132 self.d = {1:1, 2:2, 3:3}
133 def keys(self):
134 return self.d.keys()
135 def __getitem__(self, i):
136 return self.d[i]
137 d.clear()
138 d.update(SimpleUserDict())
139 self.assertEqual(d, {1:1, 2:2, 3:3})
140
141 class Exc(Exception): pass
142
143 d.clear()
144 class FailingUserDict:
145 def keys(self):
146 raise Exc
147 self.assertRaises(Exc, d.update, FailingUserDict())
148
149 class FailingUserDict:
150 def keys(self):
151 class BogonIter:
152 def __init__(self):
153 self.i = 1
154 def __iter__(self):
155 return self
156 def next(self):
157 if self.i:
158 self.i = 0
159 return 'a'
160 raise Exc
161 return BogonIter()
162 def __getitem__(self, key):
163 return key
164 self.assertRaises(Exc, d.update, FailingUserDict())
165
166 class FailingUserDict:
167 def keys(self):
168 class BogonIter:
169 def __init__(self):
170 self.i = ord('a')
171 def __iter__(self):
172 return self
173 def next(self):
174 if self.i <= ord('z'):
175 rtn = chr(self.i)
176 self.i += 1
177 return rtn
178 raise StopIteration
179 return BogonIter()
180 def __getitem__(self, key):
181 raise Exc
182 self.assertRaises(Exc, d.update, FailingUserDict())
183
184 class badseq(object):
185 def __iter__(self):
186 return self
187 def next(self):
188 raise Exc()
189
190 self.assertRaises(Exc, {}.update, badseq())
191
192 self.assertRaises(ValueError, {}.update, [(1, 2, 3)])
193
Raymond Hettinger0922d712007-02-07 20:08:22 +0000194 # SF #1615701: make d.update(m) honor __getitem__() and keys() in dict subclasses
195 class KeyUpperDict(dict):
196 def __getitem__(self, key):
197 return key.upper()
198 d.clear()
199 d.update(KeyUpperDict.fromkeys('abc'))
200 self.assertEqual(d, {'a':'A', 'b':'B', 'c':'C'})
201
Walter Dörwald59b23e82004-09-30 13:46:00 +0000202 def test_fromkeys(self):
203 self.assertEqual(dict.fromkeys('abc'), {'a':None, 'b':None, 'c':None})
204 d = {}
205 self.assert_(not(d.fromkeys('abc') is d))
206 self.assertEqual(d.fromkeys('abc'), {'a':None, 'b':None, 'c':None})
207 self.assertEqual(d.fromkeys((4,5),0), {4:0, 5:0})
208 self.assertEqual(d.fromkeys([]), {})
209 def g():
210 yield 1
211 self.assertEqual(d.fromkeys(g()), {1:None})
212 self.assertRaises(TypeError, {}.fromkeys, 3)
213 class dictlike(dict): pass
214 self.assertEqual(dictlike.fromkeys('a'), {'a':None})
215 self.assertEqual(dictlike().fromkeys('a'), {'a':None})
216 self.assert_(type(dictlike.fromkeys('a')) is dictlike)
217 self.assert_(type(dictlike().fromkeys('a')) is dictlike)
218 class mydict(dict):
219 def __new__(cls):
220 return UserDict.UserDict()
221 ud = mydict.fromkeys('ab')
222 self.assertEqual(ud, {'a':None, 'b':None})
223 self.assert_(isinstance(ud, UserDict.UserDict))
224 self.assertRaises(TypeError, dict.fromkeys)
225
226 class Exc(Exception): pass
227
228 class baddict1(dict):
229 def __init__(self):
230 raise Exc()
231
232 self.assertRaises(Exc, baddict1.fromkeys, [1])
233
234 class BadSeq(object):
235 def __iter__(self):
236 return self
237 def next(self):
238 raise Exc()
239
240 self.assertRaises(Exc, dict.fromkeys, BadSeq())
241
242 class baddict2(dict):
243 def __setitem__(self, key, value):
244 raise Exc()
245
246 self.assertRaises(Exc, baddict2.fromkeys, [1])
247
Raymond Hettingercdcf8872007-11-07 02:26:17 +0000248 # test fast path for dictionary inputs
249 d = dict(zip(range(6), range(6)))
250 self.assertEqual(dict.fromkeys(d, 0), dict(zip(range(6), [0]*6)))
251
Walter Dörwald59b23e82004-09-30 13:46:00 +0000252 def test_copy(self):
253 d = {1:1, 2:2, 3:3}
254 self.assertEqual(d.copy(), {1:1, 2:2, 3:3})
255 self.assertEqual({}.copy(), {})
256 self.assertRaises(TypeError, d.copy, None)
257
258 def test_get(self):
259 d = {}
260 self.assert_(d.get('c') is None)
261 self.assertEqual(d.get('c', 3), 3)
262 d = {'a' : 1, 'b' : 2}
263 self.assert_(d.get('c') is None)
264 self.assertEqual(d.get('c', 3), 3)
265 self.assertEqual(d.get('a'), 1)
266 self.assertEqual(d.get('a', 3), 1)
267 self.assertRaises(TypeError, d.get)
268 self.assertRaises(TypeError, d.get, None, None, None)
269
270 def test_setdefault(self):
271 # dict.setdefault()
272 d = {}
273 self.assert_(d.setdefault('key0') is None)
274 d.setdefault('key0', [])
275 self.assert_(d.setdefault('key0') is None)
276 d.setdefault('key', []).append(3)
277 self.assertEqual(d['key'][0], 3)
278 d.setdefault('key', []).append(4)
279 self.assertEqual(len(d['key']), 2)
280 self.assertRaises(TypeError, d.setdefault)
281
282 class Exc(Exception): pass
283
284 class BadHash(object):
285 fail = False
286 def __hash__(self):
287 if self.fail:
288 raise Exc()
289 else:
290 return 42
291
292 x = BadHash()
293 d[x] = 42
294 x.fail = True
295 self.assertRaises(Exc, d.setdefault, x, [])
296
297 def test_popitem(self):
298 # dict.popitem()
299 for copymode in -1, +1:
300 # -1: b has same structure as a
301 # +1: b is a.copy()
302 for log2size in range(12):
303 size = 2**log2size
304 a = {}
305 b = {}
306 for i in range(size):
307 a[repr(i)] = i
308 if copymode < 0:
309 b[repr(i)] = i
310 if copymode > 0:
311 b = a.copy()
312 for i in range(size):
313 ka, va = ta = a.popitem()
314 self.assertEqual(va, int(ka))
315 kb, vb = tb = b.popitem()
316 self.assertEqual(vb, int(kb))
317 self.assert_(not(copymode < 0 and ta != tb))
318 self.assert_(not a)
319 self.assert_(not b)
320
321 d = {}
322 self.assertRaises(KeyError, d.popitem)
323
324 def test_pop(self):
325 # Tests for pop with specified key
326 d = {}
327 k, v = 'abc', 'def'
328 d[k] = v
329 self.assertRaises(KeyError, d.pop, 'ghi')
330
331 self.assertEqual(d.pop(k), v)
332 self.assertEqual(len(d), 0)
333
334 self.assertRaises(KeyError, d.pop, k)
335
336 # verify longs/ints get same value when key > 32 bits (for 64-bit archs)
337 # see SF bug #689659
338 x = 4503599627370496L
339 y = 4503599627370496
340 h = {x: 'anything', y: 'something else'}
341 self.assertEqual(h[x], h[y])
342
343 self.assertEqual(d.pop(k, v), v)
344 d[k] = v
345 self.assertEqual(d.pop(k, 1), v)
346
347 self.assertRaises(TypeError, d.pop)
348
349 class Exc(Exception): pass
350
351 class BadHash(object):
352 fail = False
353 def __hash__(self):
354 if self.fail:
355 raise Exc()
356 else:
357 return 42
358
359 x = BadHash()
360 d[x] = 42
361 x.fail = True
362 self.assertRaises(Exc, d.pop, x)
363
364 def test_mutatingiteration(self):
365 d = {}
366 d[1] = 1
367 try:
368 for i in d:
369 d[i+1] = 1
370 except RuntimeError:
371 pass
372 else:
373 self.fail("changing dict size during iteration doesn't raise Error")
374
375 def test_repr(self):
376 d = {}
377 self.assertEqual(repr(d), '{}')
378 d[1] = 2
379 self.assertEqual(repr(d), '{1: 2}')
380 d = {}
381 d[1] = d
382 self.assertEqual(repr(d), '{1: {...}}')
383
384 class Exc(Exception): pass
385
386 class BadRepr(object):
387 def __repr__(self):
388 raise Exc()
389
390 d = {1: BadRepr()}
391 self.assertRaises(Exc, repr, d)
392
393 def test_le(self):
394 self.assert_(not ({} < {}))
395 self.assert_(not ({1: 2} < {1L: 2L}))
396
397 class Exc(Exception): pass
398
399 class BadCmp(object):
400 def __eq__(self, other):
401 raise Exc()
Guido van Rossum64c06e32007-11-22 00:55:51 +0000402 def __hash__(self):
403 return 42
Walter Dörwald59b23e82004-09-30 13:46:00 +0000404
405 d1 = {BadCmp(): 1}
406 d2 = {1: 1}
407 try:
408 d1 < d2
409 except Exc:
410 pass
411 else:
412 self.fail("< didn't raise Exc")
413
Guido van Rossum1968ad32006-02-25 22:38:04 +0000414 def test_missing(self):
415 # Make sure dict doesn't have a __missing__ method
416 self.assertEqual(hasattr(dict, "__missing__"), False)
417 self.assertEqual(hasattr({}, "__missing__"), False)
418 # Test several cases:
419 # (D) subclass defines __missing__ method returning a value
420 # (E) subclass defines __missing__ method raising RuntimeError
421 # (F) subclass sets __missing__ instance variable (no effect)
422 # (G) subclass doesn't define __missing__ at a all
423 class D(dict):
424 def __missing__(self, key):
425 return 42
426 d = D({1: 2, 3: 4})
427 self.assertEqual(d[1], 2)
428 self.assertEqual(d[3], 4)
429 self.assert_(2 not in d)
430 self.assert_(2 not in d.keys())
431 self.assertEqual(d[2], 42)
432 class E(dict):
433 def __missing__(self, key):
434 raise RuntimeError(key)
435 e = E()
436 try:
437 e[42]
438 except RuntimeError, err:
439 self.assertEqual(err.args, (42,))
440 else:
Georg Brandl8905bb12007-03-04 17:18:54 +0000441 self.fail("e[42] didn't raise RuntimeError")
Guido van Rossum1968ad32006-02-25 22:38:04 +0000442 class F(dict):
443 def __init__(self):
444 # An instance variable __missing__ should have no effect
445 self.__missing__ = lambda key: None
446 f = F()
447 try:
448 f[42]
449 except KeyError, err:
450 self.assertEqual(err.args, (42,))
451 else:
Georg Brandl8905bb12007-03-04 17:18:54 +0000452 self.fail("f[42] didn't raise KeyError")
Guido van Rossum1968ad32006-02-25 22:38:04 +0000453 class G(dict):
454 pass
455 g = G()
456 try:
457 g[42]
458 except KeyError, err:
459 self.assertEqual(err.args, (42,))
460 else:
Georg Brandl8905bb12007-03-04 17:18:54 +0000461 self.fail("g[42] didn't raise KeyError")
Guido van Rossum1968ad32006-02-25 22:38:04 +0000462
Georg Brandlb9f4ad32006-10-29 18:31:42 +0000463 def test_tuple_keyerror(self):
464 # SF #1576657
465 d = {}
466 try:
467 d[(1,)]
468 except KeyError, e:
469 self.assertEqual(e.args, ((1,),))
470 else:
471 self.fail("missing KeyError")
Tim Petersea5962f2007-03-12 18:07:52 +0000472
Collin Winterf567ca32007-03-12 15:57:19 +0000473 def test_bad_key(self):
474 # Dictionary lookups should fail if __cmp__() raises an exception.
475 class CustomException(Exception):
476 pass
Tim Petersea5962f2007-03-12 18:07:52 +0000477
Collin Winterf567ca32007-03-12 15:57:19 +0000478 class BadDictKey:
479 def __hash__(self):
480 return hash(self.__class__)
481
482 def __cmp__(self, other):
483 if isinstance(other, self.__class__):
484 raise CustomException
485 return other
Tim Petersea5962f2007-03-12 18:07:52 +0000486
Collin Winterf567ca32007-03-12 15:57:19 +0000487 d = {}
488 x1 = BadDictKey()
489 x2 = BadDictKey()
490 d[x1] = 1
491 for stmt in ['d[x2] = 2',
492 'z = d[x2]',
493 'x2 in d',
494 'd.has_key(x2)',
495 'd.get(x2)',
496 'd.setdefault(x2, 42)',
497 'd.pop(x2)',
498 'd.update({x2: 2})']:
499 try:
500 exec stmt in locals()
501 except CustomException:
502 pass
503 else:
504 self.fail("Statement didn't raise exception")
505
506 def test_resize1(self):
507 # Dict resizing bug, found by Jack Jansen in 2.2 CVS development.
508 # This version got an assert failure in debug build, infinite loop in
509 # release build. Unfortunately, provoking this kind of stuff requires
510 # a mix of inserts and deletes hitting exactly the right hash codes in
511 # exactly the right order, and I can't think of a randomized approach
512 # that would be *likely* to hit a failing case in reasonable time.
Tim Petersea5962f2007-03-12 18:07:52 +0000513
Collin Winterf567ca32007-03-12 15:57:19 +0000514 d = {}
515 for i in range(5):
516 d[i] = i
517 for i in range(5):
518 del d[i]
519 for i in range(5, 9): # i==8 was the problem
520 d[i] = i
521
522 def test_resize2(self):
523 # Another dict resizing bug (SF bug #1456209).
524 # This caused Segmentation faults or Illegal instructions.
Tim Petersea5962f2007-03-12 18:07:52 +0000525
Collin Winterf567ca32007-03-12 15:57:19 +0000526 class X(object):
527 def __hash__(self):
528 return 5
529 def __eq__(self, other):
530 if resizing:
531 d.clear()
532 return False
533 d = {}
534 resizing = False
535 d[X()] = 1
536 d[X()] = 2
537 d[X()] = 3
538 d[X()] = 4
539 d[X()] = 5
540 # now trigger a resize
541 resizing = True
542 d[9] = 6
Georg Brandlb9f4ad32006-10-29 18:31:42 +0000543
Guido van Rossum1968ad32006-02-25 22:38:04 +0000544
Neal Norwitzb902f4e2006-04-03 04:45:34 +0000545from test import mapping_tests
Raymond Hettinger49c522b2004-09-30 15:07:29 +0000546
547class GeneralMappingTests(mapping_tests.BasicTestMappingProtocol):
548 type2test = dict
549
550class Dict(dict):
551 pass
552
553class SubclassMappingTests(mapping_tests.BasicTestMappingProtocol):
554 type2test = Dict
555
Walter Dörwald59b23e82004-09-30 13:46:00 +0000556def test_main():
557 test_support.run_unittest(
558 DictTest,
Raymond Hettinger49c522b2004-09-30 15:07:29 +0000559 GeneralMappingTests,
560 SubclassMappingTests,
Walter Dörwald59b23e82004-09-30 13:46:00 +0000561 )
562
563if __name__ == "__main__":
564 test_main()