blob: de400aa580d6a28a8900bca077ac6c36b4ea4ed3 [file] [log] [blame]
Walter Dörwald59b23e82004-09-30 13:46:00 +00001import unittest
2from test import test_support
3
Christian Heimesc5f05e42008-02-23 17:40:11 +00004import UserDict, random, string
Antoine Pitrouaa687902009-01-01 14:11:22 +00005import gc, weakref
Walter Dörwald59b23e82004-09-30 13:46:00 +00006
7
8class DictTest(unittest.TestCase):
9 def test_constructor(self):
10 # calling built-in types without argument must return empty
11 self.assertEqual(dict(), {})
Benjamin Peterson5c8da862009-06-30 22:57:08 +000012 self.assertTrue(dict() is not {})
Walter Dörwald59b23e82004-09-30 13:46:00 +000013
Raymond Hettinger3b635562007-12-19 00:21:06 +000014 def test_literal_constructor(self):
15 # check literal constructor for different sized dicts (to exercise the BUILD_MAP oparg
Jeffrey Yasskined414652008-03-18 05:12:41 +000016 for n in (0, 1, 6, 256, 400):
17 items = [(''.join([random.choice(string.letters)
18 for j in range(8)]),
19 i)
20 for i in range(n)]
21 random.shuffle(items)
Raymond Hettinger3b635562007-12-19 00:21:06 +000022 dictliteral = '{' + ', '.join('%r: %d' % item for item in items) + '}'
23 self.assertEqual(eval(dictliteral), dict(items))
Raymond Hettinger3b635562007-12-19 00:21:06 +000024
Walter Dörwald59b23e82004-09-30 13:46:00 +000025 def test_bool(self):
Benjamin Peterson5c8da862009-06-30 22:57:08 +000026 self.assertTrue(not {})
27 self.assertTrue({1: 2})
28 self.assertTrue(bool({}) is False)
29 self.assertTrue(bool({1: 2}) is True)
Walter Dörwald59b23e82004-09-30 13:46:00 +000030
31 def test_keys(self):
32 d = {}
33 self.assertEqual(d.keys(), [])
34 d = {'a': 1, 'b': 2}
35 k = d.keys()
Senthil Kumaran3ddc4352010-01-08 18:41:40 +000036 self.assertTrue('a' in d)
37 self.assertTrue('b' in d)
38 # Silence Py3k warning
39 with test_support.check_warnings():
40 self.assertTrue(d.has_key('a'))
41 self.assertTrue(d.has_key('b'))
Walter Dörwald59b23e82004-09-30 13:46:00 +000042
43 self.assertRaises(TypeError, d.keys, None)
44
45 def test_values(self):
46 d = {}
47 self.assertEqual(d.values(), [])
48 d = {1:2}
49 self.assertEqual(d.values(), [2])
50
51 self.assertRaises(TypeError, d.values, None)
52
53 def test_items(self):
54 d = {}
55 self.assertEqual(d.items(), [])
56
57 d = {1:2}
58 self.assertEqual(d.items(), [(1, 2)])
59
60 self.assertRaises(TypeError, d.items, None)
61
62 def test_has_key(self):
63 d = {}
Senthil Kumaran3ddc4352010-01-08 18:41:40 +000064 self.assertTrue('a' not in d)
65 # Silence Py3k warning
66 with test_support.check_warnings():
67 self.assertTrue(not d.has_key('a'))
68 self.assertRaises(TypeError, d.has_key)
Walter Dörwald59b23e82004-09-30 13:46:00 +000069 d = {'a': 1, 'b': 2}
70 k = d.keys()
71 k.sort()
72 self.assertEqual(k, ['a', 'b'])
73
Walter Dörwald59b23e82004-09-30 13:46:00 +000074 def test_contains(self):
75 d = {}
Benjamin Peterson5c8da862009-06-30 22:57:08 +000076 self.assertTrue(not ('a' in d))
77 self.assertTrue('a' not in d)
Walter Dörwald59b23e82004-09-30 13:46:00 +000078 d = {'a': 1, 'b': 2}
Benjamin Peterson5c8da862009-06-30 22:57:08 +000079 self.assertTrue('a' in d)
80 self.assertTrue('b' in d)
81 self.assertTrue('c' not in d)
Walter Dörwald59b23e82004-09-30 13:46:00 +000082
83 self.assertRaises(TypeError, d.__contains__)
84
85 def test_len(self):
86 d = {}
87 self.assertEqual(len(d), 0)
88 d = {'a': 1, 'b': 2}
89 self.assertEqual(len(d), 2)
90
91 def test_getitem(self):
92 d = {'a': 1, 'b': 2}
93 self.assertEqual(d['a'], 1)
94 self.assertEqual(d['b'], 2)
95 d['c'] = 3
96 d['a'] = 4
97 self.assertEqual(d['c'], 3)
98 self.assertEqual(d['a'], 4)
99 del d['b']
100 self.assertEqual(d, {'a': 4, 'c': 3})
101
102 self.assertRaises(TypeError, d.__getitem__)
103
104 class BadEq(object):
105 def __eq__(self, other):
106 raise Exc()
Guido van Rossum64c06e32007-11-22 00:55:51 +0000107 def __hash__(self):
108 return 24
Walter Dörwald59b23e82004-09-30 13:46:00 +0000109
110 d = {}
111 d[BadEq()] = 42
112 self.assertRaises(KeyError, d.__getitem__, 23)
113
114 class Exc(Exception): pass
115
116 class BadHash(object):
117 fail = False
118 def __hash__(self):
119 if self.fail:
120 raise Exc()
121 else:
122 return 42
123
124 x = BadHash()
125 d[x] = 42
126 x.fail = True
127 self.assertRaises(Exc, d.__getitem__, x)
128
129 def test_clear(self):
130 d = {1:1, 2:2, 3:3}
131 d.clear()
132 self.assertEqual(d, {})
133
134 self.assertRaises(TypeError, d.clear, None)
135
136 def test_update(self):
137 d = {}
138 d.update({1:100})
139 d.update({2:20})
140 d.update({1:1, 2:2, 3:3})
141 self.assertEqual(d, {1:1, 2:2, 3:3})
142
143 d.update()
144 self.assertEqual(d, {1:1, 2:2, 3:3})
145
146 self.assertRaises((TypeError, AttributeError), d.update, None)
147
148 class SimpleUserDict:
149 def __init__(self):
150 self.d = {1:1, 2:2, 3:3}
151 def keys(self):
152 return self.d.keys()
153 def __getitem__(self, i):
154 return self.d[i]
155 d.clear()
156 d.update(SimpleUserDict())
157 self.assertEqual(d, {1:1, 2:2, 3:3})
158
159 class Exc(Exception): pass
160
161 d.clear()
162 class FailingUserDict:
163 def keys(self):
164 raise Exc
165 self.assertRaises(Exc, d.update, FailingUserDict())
166
167 class FailingUserDict:
168 def keys(self):
169 class BogonIter:
170 def __init__(self):
171 self.i = 1
172 def __iter__(self):
173 return self
174 def next(self):
175 if self.i:
176 self.i = 0
177 return 'a'
178 raise Exc
179 return BogonIter()
180 def __getitem__(self, key):
181 return key
182 self.assertRaises(Exc, d.update, FailingUserDict())
183
184 class FailingUserDict:
185 def keys(self):
186 class BogonIter:
187 def __init__(self):
188 self.i = ord('a')
189 def __iter__(self):
190 return self
191 def next(self):
192 if self.i <= ord('z'):
193 rtn = chr(self.i)
194 self.i += 1
195 return rtn
196 raise StopIteration
197 return BogonIter()
198 def __getitem__(self, key):
199 raise Exc
200 self.assertRaises(Exc, d.update, FailingUserDict())
201
202 class badseq(object):
203 def __iter__(self):
204 return self
205 def next(self):
206 raise Exc()
207
208 self.assertRaises(Exc, {}.update, badseq())
209
210 self.assertRaises(ValueError, {}.update, [(1, 2, 3)])
211
212 def test_fromkeys(self):
213 self.assertEqual(dict.fromkeys('abc'), {'a':None, 'b':None, 'c':None})
214 d = {}
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000215 self.assertTrue(not(d.fromkeys('abc') is d))
Walter Dörwald59b23e82004-09-30 13:46:00 +0000216 self.assertEqual(d.fromkeys('abc'), {'a':None, 'b':None, 'c':None})
217 self.assertEqual(d.fromkeys((4,5),0), {4:0, 5:0})
218 self.assertEqual(d.fromkeys([]), {})
219 def g():
220 yield 1
221 self.assertEqual(d.fromkeys(g()), {1:None})
222 self.assertRaises(TypeError, {}.fromkeys, 3)
223 class dictlike(dict): pass
224 self.assertEqual(dictlike.fromkeys('a'), {'a':None})
225 self.assertEqual(dictlike().fromkeys('a'), {'a':None})
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000226 self.assertTrue(type(dictlike.fromkeys('a')) is dictlike)
227 self.assertTrue(type(dictlike().fromkeys('a')) is dictlike)
Walter Dörwald59b23e82004-09-30 13:46:00 +0000228 class mydict(dict):
229 def __new__(cls):
230 return UserDict.UserDict()
231 ud = mydict.fromkeys('ab')
232 self.assertEqual(ud, {'a':None, 'b':None})
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000233 self.assertTrue(isinstance(ud, UserDict.UserDict))
Walter Dörwald59b23e82004-09-30 13:46:00 +0000234 self.assertRaises(TypeError, dict.fromkeys)
235
236 class Exc(Exception): pass
237
238 class baddict1(dict):
239 def __init__(self):
240 raise Exc()
241
242 self.assertRaises(Exc, baddict1.fromkeys, [1])
243
244 class BadSeq(object):
245 def __iter__(self):
246 return self
247 def next(self):
248 raise Exc()
249
250 self.assertRaises(Exc, dict.fromkeys, BadSeq())
251
252 class baddict2(dict):
253 def __setitem__(self, key, value):
254 raise Exc()
255
256 self.assertRaises(Exc, baddict2.fromkeys, [1])
257
Raymond Hettingercdcf8872007-11-07 02:26:17 +0000258 # test fast path for dictionary inputs
259 d = dict(zip(range(6), range(6)))
260 self.assertEqual(dict.fromkeys(d, 0), dict(zip(range(6), [0]*6)))
261
Walter Dörwald59b23e82004-09-30 13:46:00 +0000262 def test_copy(self):
263 d = {1:1, 2:2, 3:3}
264 self.assertEqual(d.copy(), {1:1, 2:2, 3:3})
265 self.assertEqual({}.copy(), {})
266 self.assertRaises(TypeError, d.copy, None)
267
268 def test_get(self):
269 d = {}
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000270 self.assertTrue(d.get('c') is None)
Walter Dörwald59b23e82004-09-30 13:46:00 +0000271 self.assertEqual(d.get('c', 3), 3)
272 d = {'a' : 1, 'b' : 2}
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000273 self.assertTrue(d.get('c') is None)
Walter Dörwald59b23e82004-09-30 13:46:00 +0000274 self.assertEqual(d.get('c', 3), 3)
275 self.assertEqual(d.get('a'), 1)
276 self.assertEqual(d.get('a', 3), 1)
277 self.assertRaises(TypeError, d.get)
278 self.assertRaises(TypeError, d.get, None, None, None)
279
280 def test_setdefault(self):
281 # dict.setdefault()
282 d = {}
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000283 self.assertTrue(d.setdefault('key0') is None)
Walter Dörwald59b23e82004-09-30 13:46:00 +0000284 d.setdefault('key0', [])
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000285 self.assertTrue(d.setdefault('key0') is None)
Walter Dörwald59b23e82004-09-30 13:46:00 +0000286 d.setdefault('key', []).append(3)
287 self.assertEqual(d['key'][0], 3)
288 d.setdefault('key', []).append(4)
289 self.assertEqual(len(d['key']), 2)
290 self.assertRaises(TypeError, d.setdefault)
291
292 class Exc(Exception): pass
293
294 class BadHash(object):
295 fail = False
296 def __hash__(self):
297 if self.fail:
298 raise Exc()
299 else:
300 return 42
301
302 x = BadHash()
303 d[x] = 42
304 x.fail = True
305 self.assertRaises(Exc, d.setdefault, x, [])
306
307 def test_popitem(self):
308 # dict.popitem()
309 for copymode in -1, +1:
310 # -1: b has same structure as a
311 # +1: b is a.copy()
312 for log2size in range(12):
313 size = 2**log2size
314 a = {}
315 b = {}
316 for i in range(size):
317 a[repr(i)] = i
318 if copymode < 0:
319 b[repr(i)] = i
320 if copymode > 0:
321 b = a.copy()
322 for i in range(size):
323 ka, va = ta = a.popitem()
324 self.assertEqual(va, int(ka))
325 kb, vb = tb = b.popitem()
326 self.assertEqual(vb, int(kb))
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000327 self.assertTrue(not(copymode < 0 and ta != tb))
328 self.assertTrue(not a)
329 self.assertTrue(not b)
Walter Dörwald59b23e82004-09-30 13:46:00 +0000330
331 d = {}
332 self.assertRaises(KeyError, d.popitem)
333
334 def test_pop(self):
335 # Tests for pop with specified key
336 d = {}
337 k, v = 'abc', 'def'
338 d[k] = v
339 self.assertRaises(KeyError, d.pop, 'ghi')
340
341 self.assertEqual(d.pop(k), v)
342 self.assertEqual(len(d), 0)
343
344 self.assertRaises(KeyError, d.pop, k)
345
346 # verify longs/ints get same value when key > 32 bits (for 64-bit archs)
347 # see SF bug #689659
348 x = 4503599627370496L
349 y = 4503599627370496
350 h = {x: 'anything', y: 'something else'}
351 self.assertEqual(h[x], h[y])
352
353 self.assertEqual(d.pop(k, v), v)
354 d[k] = v
355 self.assertEqual(d.pop(k, 1), v)
356
357 self.assertRaises(TypeError, d.pop)
358
359 class Exc(Exception): pass
360
361 class BadHash(object):
362 fail = False
363 def __hash__(self):
364 if self.fail:
365 raise Exc()
366 else:
367 return 42
368
369 x = BadHash()
370 d[x] = 42
371 x.fail = True
372 self.assertRaises(Exc, d.pop, x)
373
374 def test_mutatingiteration(self):
375 d = {}
376 d[1] = 1
377 try:
378 for i in d:
379 d[i+1] = 1
380 except RuntimeError:
381 pass
382 else:
383 self.fail("changing dict size during iteration doesn't raise Error")
384
385 def test_repr(self):
386 d = {}
387 self.assertEqual(repr(d), '{}')
388 d[1] = 2
389 self.assertEqual(repr(d), '{1: 2}')
390 d = {}
391 d[1] = d
392 self.assertEqual(repr(d), '{1: {...}}')
393
394 class Exc(Exception): pass
395
396 class BadRepr(object):
397 def __repr__(self):
398 raise Exc()
399
400 d = {1: BadRepr()}
401 self.assertRaises(Exc, repr, d)
402
403 def test_le(self):
Walter Dörwald59b23e82004-09-30 13:46:00 +0000404
405 class Exc(Exception): pass
406
407 class BadCmp(object):
408 def __eq__(self, other):
409 raise Exc()
Guido van Rossum64c06e32007-11-22 00:55:51 +0000410 def __hash__(self):
411 return 42
Walter Dörwald59b23e82004-09-30 13:46:00 +0000412
413 d1 = {BadCmp(): 1}
414 d2 = {1: 1}
Senthil Kumaran3ddc4352010-01-08 18:41:40 +0000415
416 # Silence Py3k warning
417 with test_support.check_warnings():
418 self.assertTrue(not ({} < {}))
419 self.assertTrue(not ({1: 2} < {1L: 2L}))
420
421 try:
422 d1 < d2
423 except Exc:
424 pass
425 else:
426 self.fail("< didn't raise Exc")
Walter Dörwald59b23e82004-09-30 13:46:00 +0000427
Guido van Rossum1968ad32006-02-25 22:38:04 +0000428 def test_missing(self):
429 # Make sure dict doesn't have a __missing__ method
430 self.assertEqual(hasattr(dict, "__missing__"), False)
431 self.assertEqual(hasattr({}, "__missing__"), False)
432 # Test several cases:
433 # (D) subclass defines __missing__ method returning a value
434 # (E) subclass defines __missing__ method raising RuntimeError
435 # (F) subclass sets __missing__ instance variable (no effect)
436 # (G) subclass doesn't define __missing__ at a all
437 class D(dict):
438 def __missing__(self, key):
439 return 42
440 d = D({1: 2, 3: 4})
441 self.assertEqual(d[1], 2)
442 self.assertEqual(d[3], 4)
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000443 self.assertTrue(2 not in d)
444 self.assertTrue(2 not in d.keys())
Guido van Rossum1968ad32006-02-25 22:38:04 +0000445 self.assertEqual(d[2], 42)
446 class E(dict):
447 def __missing__(self, key):
448 raise RuntimeError(key)
449 e = E()
450 try:
451 e[42]
452 except RuntimeError, err:
453 self.assertEqual(err.args, (42,))
454 else:
Georg Brandl8905bb12007-03-04 17:18:54 +0000455 self.fail("e[42] didn't raise RuntimeError")
Guido van Rossum1968ad32006-02-25 22:38:04 +0000456 class F(dict):
457 def __init__(self):
458 # An instance variable __missing__ should have no effect
459 self.__missing__ = lambda key: None
460 f = F()
461 try:
462 f[42]
463 except KeyError, err:
464 self.assertEqual(err.args, (42,))
465 else:
Georg Brandl8905bb12007-03-04 17:18:54 +0000466 self.fail("f[42] didn't raise KeyError")
Guido van Rossum1968ad32006-02-25 22:38:04 +0000467 class G(dict):
468 pass
469 g = G()
470 try:
471 g[42]
472 except KeyError, err:
473 self.assertEqual(err.args, (42,))
474 else:
Georg Brandl8905bb12007-03-04 17:18:54 +0000475 self.fail("g[42] didn't raise KeyError")
Guido van Rossum1968ad32006-02-25 22:38:04 +0000476
Georg Brandlb9f4ad32006-10-29 18:31:42 +0000477 def test_tuple_keyerror(self):
478 # SF #1576657
479 d = {}
480 try:
481 d[(1,)]
482 except KeyError, e:
483 self.assertEqual(e.args, ((1,),))
484 else:
485 self.fail("missing KeyError")
Tim Petersea5962f2007-03-12 18:07:52 +0000486
Collin Winterf567ca32007-03-12 15:57:19 +0000487 def test_bad_key(self):
488 # Dictionary lookups should fail if __cmp__() raises an exception.
489 class CustomException(Exception):
490 pass
Tim Petersea5962f2007-03-12 18:07:52 +0000491
Collin Winterf567ca32007-03-12 15:57:19 +0000492 class BadDictKey:
493 def __hash__(self):
494 return hash(self.__class__)
495
496 def __cmp__(self, other):
497 if isinstance(other, self.__class__):
498 raise CustomException
499 return other
Tim Petersea5962f2007-03-12 18:07:52 +0000500
Collin Winterf567ca32007-03-12 15:57:19 +0000501 d = {}
502 x1 = BadDictKey()
503 x2 = BadDictKey()
504 d[x1] = 1
505 for stmt in ['d[x2] = 2',
506 'z = d[x2]',
507 'x2 in d',
508 'd.has_key(x2)',
509 'd.get(x2)',
510 'd.setdefault(x2, 42)',
511 'd.pop(x2)',
512 'd.update({x2: 2})']:
513 try:
Senthil Kumaran3ddc4352010-01-08 18:41:40 +0000514 # Silence Py3k warning
515 with test_support.check_warnings():
516 exec stmt in locals()
Collin Winterf567ca32007-03-12 15:57:19 +0000517 except CustomException:
518 pass
519 else:
520 self.fail("Statement didn't raise exception")
521
522 def test_resize1(self):
523 # Dict resizing bug, found by Jack Jansen in 2.2 CVS development.
524 # This version got an assert failure in debug build, infinite loop in
525 # release build. Unfortunately, provoking this kind of stuff requires
526 # a mix of inserts and deletes hitting exactly the right hash codes in
527 # exactly the right order, and I can't think of a randomized approach
528 # that would be *likely* to hit a failing case in reasonable time.
Tim Petersea5962f2007-03-12 18:07:52 +0000529
Collin Winterf567ca32007-03-12 15:57:19 +0000530 d = {}
531 for i in range(5):
532 d[i] = i
533 for i in range(5):
534 del d[i]
535 for i in range(5, 9): # i==8 was the problem
536 d[i] = i
537
538 def test_resize2(self):
539 # Another dict resizing bug (SF bug #1456209).
540 # This caused Segmentation faults or Illegal instructions.
Tim Petersea5962f2007-03-12 18:07:52 +0000541
Collin Winterf567ca32007-03-12 15:57:19 +0000542 class X(object):
543 def __hash__(self):
544 return 5
545 def __eq__(self, other):
546 if resizing:
547 d.clear()
548 return False
549 d = {}
550 resizing = False
551 d[X()] = 1
552 d[X()] = 2
553 d[X()] = 3
554 d[X()] = 4
555 d[X()] = 5
556 # now trigger a resize
557 resizing = True
558 d[9] = 6
Georg Brandlb9f4ad32006-10-29 18:31:42 +0000559
Georg Brandl1e13ea92008-08-11 09:07:59 +0000560 def test_empty_presized_dict_in_freelist(self):
561 # Bug #3537: if an empty but presized dict with a size larger
562 # than 7 was in the freelist, it triggered an assertion failure
563 try:
Senthil Kumaran3ddc4352010-01-08 18:41:40 +0000564 d = {'a': 1 // 0, 'b': None, 'c': None, 'd': None, 'e': None,
Georg Brandl1e13ea92008-08-11 09:07:59 +0000565 'f': None, 'g': None, 'h': None}
566 except ZeroDivisionError:
567 pass
568 d = {}
569
Antoine Pitrouaa687902009-01-01 14:11:22 +0000570 def test_container_iterator(self):
Antoine Pitrou733dc742009-01-01 15:38:03 +0000571 # Bug #3680: tp_traverse was not implemented for dictiter objects
Antoine Pitrouaa687902009-01-01 14:11:22 +0000572 class C(object):
573 pass
574 iterators = (dict.iteritems, dict.itervalues, dict.iterkeys)
575 for i in iterators:
576 obj = C()
577 ref = weakref.ref(obj)
578 container = {obj: 1}
579 obj.x = i(container)
580 del obj, container
581 gc.collect()
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000582 self.assertTrue(ref() is None, "Cycle was not collected")
Georg Brandl1e13ea92008-08-11 09:07:59 +0000583
Antoine Pitrouf8387af2009-03-23 18:41:45 +0000584 def _not_tracked(self, t):
585 # Nested containers can take several collections to untrack
586 gc.collect()
587 gc.collect()
588 self.assertFalse(gc.is_tracked(t), t)
589
590 def _tracked(self, t):
591 self.assertTrue(gc.is_tracked(t), t)
592 gc.collect()
593 gc.collect()
594 self.assertTrue(gc.is_tracked(t), t)
595
596 def test_track_literals(self):
597 # Test GC-optimization of dict literals
598 x, y, z, w = 1.5, "a", (1, None), []
599
600 self._not_tracked({})
601 self._not_tracked({x:(), y:x, z:1})
602 self._not_tracked({1: "a", "b": 2})
603 self._not_tracked({1: 2, (None, True, False, ()): int})
604 self._not_tracked({1: object()})
605
606 # Dicts with mutable elements are always tracked, even if those
607 # elements are not tracked right now.
608 self._tracked({1: []})
609 self._tracked({1: ([],)})
610 self._tracked({1: {}})
611 self._tracked({1: set()})
612
613 def test_track_dynamic(self):
614 # Test GC-optimization of dynamically-created dicts
615 class MyObject(object):
616 pass
617 x, y, z, w, o = 1.5, "a", (1, object()), [], MyObject()
618
619 d = dict()
620 self._not_tracked(d)
621 d[1] = "a"
622 self._not_tracked(d)
623 d[y] = 2
624 self._not_tracked(d)
625 d[z] = 3
626 self._not_tracked(d)
627 self._not_tracked(d.copy())
628 d[4] = w
629 self._tracked(d)
630 self._tracked(d.copy())
631 d[4] = None
632 self._not_tracked(d)
633 self._not_tracked(d.copy())
634
635 # dd isn't tracked right now, but it may mutate and therefore d
636 # which contains it must be tracked.
637 d = dict()
638 dd = dict()
639 d[1] = dd
640 self._not_tracked(dd)
641 self._tracked(d)
642 dd[1] = d
643 self._tracked(dd)
644
645 d = dict.fromkeys([x, y, z])
646 self._not_tracked(d)
647 dd = dict()
648 dd.update(d)
649 self._not_tracked(dd)
650 d = dict.fromkeys([x, y, z, o])
651 self._tracked(d)
652 dd = dict()
653 dd.update(d)
654 self._tracked(dd)
655
656 d = dict(x=x, y=y, z=z)
657 self._not_tracked(d)
658 d = dict(x=x, y=y, z=z, w=w)
659 self._tracked(d)
660 d = dict()
661 d.update(x=x, y=y, z=z)
662 self._not_tracked(d)
663 d.update(w=w)
664 self._tracked(d)
665
666 d = dict([(x, y), (z, 1)])
667 self._not_tracked(d)
668 d = dict([(x, y), (z, w)])
669 self._tracked(d)
670 d = dict()
671 d.update([(x, y), (z, 1)])
672 self._not_tracked(d)
673 d.update([(x, y), (z, w)])
674 self._tracked(d)
675
676 def test_track_subtypes(self):
677 # Dict subtypes are always tracked
678 class MyDict(dict):
679 pass
680 self._tracked(MyDict())
681
Guido van Rossum1968ad32006-02-25 22:38:04 +0000682
Neal Norwitzb902f4e2006-04-03 04:45:34 +0000683from test import mapping_tests
Raymond Hettinger49c522b2004-09-30 15:07:29 +0000684
685class GeneralMappingTests(mapping_tests.BasicTestMappingProtocol):
686 type2test = dict
687
688class Dict(dict):
689 pass
690
691class SubclassMappingTests(mapping_tests.BasicTestMappingProtocol):
692 type2test = Dict
693
Walter Dörwald59b23e82004-09-30 13:46:00 +0000694def test_main():
695 test_support.run_unittest(
696 DictTest,
Raymond Hettinger49c522b2004-09-30 15:07:29 +0000697 GeneralMappingTests,
698 SubclassMappingTests,
Walter Dörwald59b23e82004-09-30 13:46:00 +0000699 )
700
701if __name__ == "__main__":
702 test_main()