blob: 15db51d3ea389271439d9778293530e4a0c7738f [file] [log] [blame]
Walter Dörwald59b23e82004-09-30 13:46:00 +00001import unittest
Benjamin Petersonee8712c2008-05-20 21:35:26 +00002from test import support
Walter Dörwald59b23e82004-09-30 13:46:00 +00003
Florent Xiclunaa988e422010-03-02 16:06:24 +00004import collections, random, string
Antoine Pitrou7ddda782009-01-01 15:35:33 +00005import gc, weakref
Walter Dörwald59b23e82004-09-30 13:46:00 +00006
7
8class DictTest(unittest.TestCase):
Guido van Rossum47b9ff62006-08-24 00:41:19 +00009
Benjamin Petersonfb886362010-04-24 18:21:17 +000010 def test_invalid_keyword_arguments(self):
Benjamin Petersonf6096542010-11-17 22:33:12 +000011 class Custom(dict):
12 pass
13 for invalid in {1 : 2}, Custom({1 : 2}):
14 with self.assertRaises(TypeError):
15 dict(**invalid)
16 with self.assertRaises(TypeError):
17 {}.update(**invalid)
Benjamin Petersonfb886362010-04-24 18:21:17 +000018
Walter Dörwald59b23e82004-09-30 13:46:00 +000019 def test_constructor(self):
20 # calling built-in types without argument must return empty
21 self.assertEqual(dict(), {})
Florent Xiclunaa988e422010-03-02 16:06:24 +000022 self.assertIsNot(dict(), {})
Walter Dörwald59b23e82004-09-30 13:46:00 +000023
Christian Heimes99170a52007-12-19 02:07:34 +000024 def test_literal_constructor(self):
Florent Xiclunaa988e422010-03-02 16:06:24 +000025 # check literal constructor for different sized dicts
26 # (to exercise the BUILD_MAP oparg).
Christian Heimesb186d002008-03-18 15:15:01 +000027 for n in (0, 1, 6, 256, 400):
Florent Xiclunaa988e422010-03-02 16:06:24 +000028 items = [(''.join(random.sample(string.ascii_letters, 8)), i)
Christian Heimesb186d002008-03-18 15:15:01 +000029 for i in range(n)]
30 random.shuffle(items)
Florent Xiclunaa988e422010-03-02 16:06:24 +000031 formatted_items = ('{!r}: {:d}'.format(k, v) for k, v in items)
32 dictliteral = '{' + ', '.join(formatted_items) + '}'
Christian Heimes99170a52007-12-19 02:07:34 +000033 self.assertEqual(eval(dictliteral), dict(items))
Christian Heimes99170a52007-12-19 02:07:34 +000034
Walter Dörwald59b23e82004-09-30 13:46:00 +000035 def test_bool(self):
Florent Xiclunaa988e422010-03-02 16:06:24 +000036 self.assertIs(not {}, True)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000037 self.assertTrue({1: 2})
Florent Xiclunaa988e422010-03-02 16:06:24 +000038 self.assertIs(bool({}), False)
39 self.assertIs(bool({1: 2}), True)
Walter Dörwald59b23e82004-09-30 13:46:00 +000040
41 def test_keys(self):
42 d = {}
Guido van Rossume34cdd12007-02-14 17:49:04 +000043 self.assertEqual(set(d.keys()), set())
Walter Dörwald59b23e82004-09-30 13:46:00 +000044 d = {'a': 1, 'b': 2}
45 k = d.keys()
Benjamin Peterson577473f2010-01-19 00:09:57 +000046 self.assertIn('a', d)
Benjamin Peterson577473f2010-01-19 00:09:57 +000047 self.assertIn('b', d)
Walter Dörwald59b23e82004-09-30 13:46:00 +000048 self.assertRaises(TypeError, d.keys, None)
Raymond Hettingerb0d56af2009-03-03 10:52:49 +000049 self.assertEqual(repr(dict(a=1).keys()), "dict_keys(['a'])")
Walter Dörwald59b23e82004-09-30 13:46:00 +000050
51 def test_values(self):
52 d = {}
Guido van Rossume34cdd12007-02-14 17:49:04 +000053 self.assertEqual(set(d.values()), set())
Walter Dörwald59b23e82004-09-30 13:46:00 +000054 d = {1:2}
Guido van Rossume34cdd12007-02-14 17:49:04 +000055 self.assertEqual(set(d.values()), {2})
Walter Dörwald59b23e82004-09-30 13:46:00 +000056 self.assertRaises(TypeError, d.values, None)
Raymond Hettingerb0d56af2009-03-03 10:52:49 +000057 self.assertEqual(repr(dict(a=1).values()), "dict_values([1])")
Walter Dörwald59b23e82004-09-30 13:46:00 +000058
59 def test_items(self):
60 d = {}
Guido van Rossume34cdd12007-02-14 17:49:04 +000061 self.assertEqual(set(d.items()), set())
Walter Dörwald59b23e82004-09-30 13:46:00 +000062
63 d = {1:2}
Guido van Rossume34cdd12007-02-14 17:49:04 +000064 self.assertEqual(set(d.items()), {(1, 2)})
Walter Dörwald59b23e82004-09-30 13:46:00 +000065 self.assertRaises(TypeError, d.items, None)
Raymond Hettingerb0d56af2009-03-03 10:52:49 +000066 self.assertEqual(repr(dict(a=1).items()), "dict_items([('a', 1)])")
Walter Dörwald59b23e82004-09-30 13:46:00 +000067
Walter Dörwald59b23e82004-09-30 13:46:00 +000068 def test_contains(self):
69 d = {}
Ezio Melottib58e0bd2010-01-23 15:40:09 +000070 self.assertNotIn('a', d)
Florent Xiclunaa988e422010-03-02 16:06:24 +000071 self.assertFalse('a' in d)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000072 self.assertTrue('a' not in d)
Walter Dörwald59b23e82004-09-30 13:46:00 +000073 d = {'a': 1, 'b': 2}
Benjamin Peterson577473f2010-01-19 00:09:57 +000074 self.assertIn('a', d)
Benjamin Peterson577473f2010-01-19 00:09:57 +000075 self.assertIn('b', d)
Benjamin Peterson577473f2010-01-19 00:09:57 +000076 self.assertNotIn('c', d)
Walter Dörwald59b23e82004-09-30 13:46:00 +000077
78 self.assertRaises(TypeError, d.__contains__)
79
80 def test_len(self):
81 d = {}
82 self.assertEqual(len(d), 0)
83 d = {'a': 1, 'b': 2}
84 self.assertEqual(len(d), 2)
85
86 def test_getitem(self):
87 d = {'a': 1, 'b': 2}
88 self.assertEqual(d['a'], 1)
89 self.assertEqual(d['b'], 2)
90 d['c'] = 3
91 d['a'] = 4
92 self.assertEqual(d['c'], 3)
93 self.assertEqual(d['a'], 4)
94 del d['b']
95 self.assertEqual(d, {'a': 4, 'c': 3})
96
97 self.assertRaises(TypeError, d.__getitem__)
98
99 class BadEq(object):
100 def __eq__(self, other):
101 raise Exc()
Guido van Rossum38938152006-08-21 23:36:26 +0000102 def __hash__(self):
103 return 24
Walter Dörwald59b23e82004-09-30 13:46:00 +0000104
105 d = {}
106 d[BadEq()] = 42
107 self.assertRaises(KeyError, d.__getitem__, 23)
108
109 class Exc(Exception): pass
110
111 class BadHash(object):
112 fail = False
113 def __hash__(self):
114 if self.fail:
115 raise Exc()
116 else:
117 return 42
118
119 x = BadHash()
120 d[x] = 42
121 x.fail = True
122 self.assertRaises(Exc, d.__getitem__, x)
123
124 def test_clear(self):
125 d = {1:1, 2:2, 3:3}
126 d.clear()
127 self.assertEqual(d, {})
128
129 self.assertRaises(TypeError, d.clear, None)
130
131 def test_update(self):
132 d = {}
133 d.update({1:100})
134 d.update({2:20})
135 d.update({1:1, 2:2, 3:3})
136 self.assertEqual(d, {1:1, 2:2, 3:3})
137
138 d.update()
139 self.assertEqual(d, {1:1, 2:2, 3:3})
140
141 self.assertRaises((TypeError, AttributeError), d.update, None)
142
143 class SimpleUserDict:
144 def __init__(self):
145 self.d = {1:1, 2:2, 3:3}
146 def keys(self):
147 return self.d.keys()
148 def __getitem__(self, i):
149 return self.d[i]
150 d.clear()
151 d.update(SimpleUserDict())
152 self.assertEqual(d, {1:1, 2:2, 3:3})
153
154 class Exc(Exception): pass
155
156 d.clear()
157 class FailingUserDict:
158 def keys(self):
159 raise Exc
160 self.assertRaises(Exc, d.update, FailingUserDict())
161
162 class FailingUserDict:
163 def keys(self):
164 class BogonIter:
165 def __init__(self):
166 self.i = 1
167 def __iter__(self):
168 return self
Georg Brandla18af4e2007-04-21 15:47:16 +0000169 def __next__(self):
Walter Dörwald59b23e82004-09-30 13:46:00 +0000170 if self.i:
171 self.i = 0
172 return 'a'
173 raise Exc
174 return BogonIter()
175 def __getitem__(self, key):
176 return key
177 self.assertRaises(Exc, d.update, FailingUserDict())
178
179 class FailingUserDict:
180 def keys(self):
181 class BogonIter:
182 def __init__(self):
183 self.i = ord('a')
184 def __iter__(self):
185 return self
Georg Brandla18af4e2007-04-21 15:47:16 +0000186 def __next__(self):
Walter Dörwald59b23e82004-09-30 13:46:00 +0000187 if self.i <= ord('z'):
188 rtn = chr(self.i)
189 self.i += 1
190 return rtn
191 raise StopIteration
192 return BogonIter()
193 def __getitem__(self, key):
194 raise Exc
195 self.assertRaises(Exc, d.update, FailingUserDict())
196
197 class badseq(object):
198 def __iter__(self):
199 return self
Georg Brandla18af4e2007-04-21 15:47:16 +0000200 def __next__(self):
Walter Dörwald59b23e82004-09-30 13:46:00 +0000201 raise Exc()
202
203 self.assertRaises(Exc, {}.update, badseq())
204
205 self.assertRaises(ValueError, {}.update, [(1, 2, 3)])
206
207 def test_fromkeys(self):
208 self.assertEqual(dict.fromkeys('abc'), {'a':None, 'b':None, 'c':None})
209 d = {}
Florent Xiclunaa988e422010-03-02 16:06:24 +0000210 self.assertIsNot(d.fromkeys('abc'), d)
Walter Dörwald59b23e82004-09-30 13:46:00 +0000211 self.assertEqual(d.fromkeys('abc'), {'a':None, 'b':None, 'c':None})
212 self.assertEqual(d.fromkeys((4,5),0), {4:0, 5:0})
213 self.assertEqual(d.fromkeys([]), {})
214 def g():
215 yield 1
216 self.assertEqual(d.fromkeys(g()), {1:None})
217 self.assertRaises(TypeError, {}.fromkeys, 3)
218 class dictlike(dict): pass
219 self.assertEqual(dictlike.fromkeys('a'), {'a':None})
220 self.assertEqual(dictlike().fromkeys('a'), {'a':None})
Florent Xiclunaa988e422010-03-02 16:06:24 +0000221 self.assertIsInstance(dictlike.fromkeys('a'), dictlike)
222 self.assertIsInstance(dictlike().fromkeys('a'), dictlike)
Walter Dörwald59b23e82004-09-30 13:46:00 +0000223 class mydict(dict):
224 def __new__(cls):
Raymond Hettingerf80680d2008-02-06 00:07:11 +0000225 return collections.UserDict()
Walter Dörwald59b23e82004-09-30 13:46:00 +0000226 ud = mydict.fromkeys('ab')
227 self.assertEqual(ud, {'a':None, 'b':None})
Ezio Melottie9615932010-01-24 19:26:24 +0000228 self.assertIsInstance(ud, collections.UserDict)
Walter Dörwald59b23e82004-09-30 13:46:00 +0000229 self.assertRaises(TypeError, dict.fromkeys)
230
231 class Exc(Exception): pass
232
233 class baddict1(dict):
234 def __init__(self):
235 raise Exc()
236
237 self.assertRaises(Exc, baddict1.fromkeys, [1])
238
239 class BadSeq(object):
240 def __iter__(self):
241 return self
Georg Brandla18af4e2007-04-21 15:47:16 +0000242 def __next__(self):
Walter Dörwald59b23e82004-09-30 13:46:00 +0000243 raise Exc()
244
245 self.assertRaises(Exc, dict.fromkeys, BadSeq())
246
247 class baddict2(dict):
248 def __setitem__(self, key, value):
249 raise Exc()
250
251 self.assertRaises(Exc, baddict2.fromkeys, [1])
252
Guido van Rossum58da9312007-11-10 23:39:45 +0000253 # test fast path for dictionary inputs
254 d = dict(zip(range(6), range(6)))
255 self.assertEqual(dict.fromkeys(d, 0), dict(zip(range(6), [0]*6)))
256
Walter Dörwald59b23e82004-09-30 13:46:00 +0000257 def test_copy(self):
258 d = {1:1, 2:2, 3:3}
259 self.assertEqual(d.copy(), {1:1, 2:2, 3:3})
260 self.assertEqual({}.copy(), {})
261 self.assertRaises(TypeError, d.copy, None)
262
263 def test_get(self):
264 d = {}
Florent Xiclunaa988e422010-03-02 16:06:24 +0000265 self.assertIs(d.get('c'), None)
Walter Dörwald59b23e82004-09-30 13:46:00 +0000266 self.assertEqual(d.get('c', 3), 3)
Florent Xiclunaa988e422010-03-02 16:06:24 +0000267 d = {'a': 1, 'b': 2}
268 self.assertIs(d.get('c'), None)
Walter Dörwald59b23e82004-09-30 13:46:00 +0000269 self.assertEqual(d.get('c', 3), 3)
270 self.assertEqual(d.get('a'), 1)
271 self.assertEqual(d.get('a', 3), 1)
272 self.assertRaises(TypeError, d.get)
273 self.assertRaises(TypeError, d.get, None, None, None)
274
275 def test_setdefault(self):
276 # dict.setdefault()
277 d = {}
Florent Xiclunaa988e422010-03-02 16:06:24 +0000278 self.assertIs(d.setdefault('key0'), None)
Walter Dörwald59b23e82004-09-30 13:46:00 +0000279 d.setdefault('key0', [])
Florent Xiclunaa988e422010-03-02 16:06:24 +0000280 self.assertIs(d.setdefault('key0'), None)
Walter Dörwald59b23e82004-09-30 13:46:00 +0000281 d.setdefault('key', []).append(3)
282 self.assertEqual(d['key'][0], 3)
283 d.setdefault('key', []).append(4)
284 self.assertEqual(len(d['key']), 2)
285 self.assertRaises(TypeError, d.setdefault)
286
287 class Exc(Exception): pass
288
289 class BadHash(object):
290 fail = False
291 def __hash__(self):
292 if self.fail:
293 raise Exc()
294 else:
295 return 42
296
297 x = BadHash()
298 d[x] = 42
299 x.fail = True
300 self.assertRaises(Exc, d.setdefault, x, [])
301
Antoine Pitroue965d972012-02-27 00:45:12 +0100302 def test_setdefault_atomic(self):
303 # Issue #13521: setdefault() calls __hash__ and __eq__ only once.
304 class Hashed(object):
305 def __init__(self):
306 self.hash_count = 0
307 self.eq_count = 0
308 def __hash__(self):
309 self.hash_count += 1
310 return 42
311 def __eq__(self, other):
312 self.eq_count += 1
313 return id(self) == id(other)
314 hashed1 = Hashed()
315 y = {hashed1: 5}
316 hashed2 = Hashed()
317 y.setdefault(hashed2, [])
318 self.assertEqual(hashed1.hash_count, 1)
319 self.assertEqual(hashed2.hash_count, 1)
320 self.assertEqual(hashed1.eq_count + hashed2.eq_count, 1)
321
Walter Dörwald59b23e82004-09-30 13:46:00 +0000322 def test_popitem(self):
323 # dict.popitem()
324 for copymode in -1, +1:
325 # -1: b has same structure as a
326 # +1: b is a.copy()
327 for log2size in range(12):
328 size = 2**log2size
329 a = {}
330 b = {}
331 for i in range(size):
332 a[repr(i)] = i
333 if copymode < 0:
334 b[repr(i)] = i
335 if copymode > 0:
336 b = a.copy()
337 for i in range(size):
338 ka, va = ta = a.popitem()
339 self.assertEqual(va, int(ka))
340 kb, vb = tb = b.popitem()
341 self.assertEqual(vb, int(kb))
Florent Xiclunaa988e422010-03-02 16:06:24 +0000342 self.assertFalse(copymode < 0 and ta != tb)
343 self.assertFalse(a)
344 self.assertFalse(b)
Walter Dörwald59b23e82004-09-30 13:46:00 +0000345
346 d = {}
347 self.assertRaises(KeyError, d.popitem)
348
349 def test_pop(self):
350 # Tests for pop with specified key
351 d = {}
352 k, v = 'abc', 'def'
353 d[k] = v
354 self.assertRaises(KeyError, d.pop, 'ghi')
355
356 self.assertEqual(d.pop(k), v)
357 self.assertEqual(len(d), 0)
358
359 self.assertRaises(KeyError, d.pop, k)
360
Walter Dörwald59b23e82004-09-30 13:46:00 +0000361 self.assertEqual(d.pop(k, v), v)
362 d[k] = v
363 self.assertEqual(d.pop(k, 1), v)
364
365 self.assertRaises(TypeError, d.pop)
366
367 class Exc(Exception): pass
368
369 class BadHash(object):
370 fail = False
371 def __hash__(self):
372 if self.fail:
373 raise Exc()
374 else:
375 return 42
376
377 x = BadHash()
378 d[x] = 42
379 x.fail = True
380 self.assertRaises(Exc, d.pop, x)
381
Victor Stinner198b2912012-03-06 01:03:13 +0100382 def test_mutating_iteration(self):
Florent Xiclunaa988e422010-03-02 16:06:24 +0000383 # changing dict size during iteration
Walter Dörwald59b23e82004-09-30 13:46:00 +0000384 d = {}
385 d[1] = 1
Florent Xiclunaa988e422010-03-02 16:06:24 +0000386 with self.assertRaises(RuntimeError):
Walter Dörwald59b23e82004-09-30 13:46:00 +0000387 for i in d:
388 d[i+1] = 1
Walter Dörwald59b23e82004-09-30 13:46:00 +0000389
Victor Stinner198b2912012-03-06 01:03:13 +0100390 def test_mutating_lookup(self):
391 # changing dict during a lookup
392 class NastyKey:
393 mutate_dict = None
394
395 def __hash__(self):
396 # hash collision!
397 return 1
398
399 def __eq__(self, other):
400 if self.mutate_dict:
401 self.mutate_dict[self] = 1
402 return self == other
403
404 d = {}
405 d[NastyKey()] = 0
406 NastyKey.mutate_dict = d
407 with self.assertRaises(RuntimeError):
408 d[NastyKey()] = None
409
Walter Dörwald59b23e82004-09-30 13:46:00 +0000410 def test_repr(self):
411 d = {}
412 self.assertEqual(repr(d), '{}')
413 d[1] = 2
414 self.assertEqual(repr(d), '{1: 2}')
415 d = {}
416 d[1] = d
417 self.assertEqual(repr(d), '{1: {...}}')
418
419 class Exc(Exception): pass
420
421 class BadRepr(object):
422 def __repr__(self):
423 raise Exc()
424
425 d = {1: BadRepr()}
426 self.assertRaises(Exc, repr, d)
427
Guido van Rossum47b9ff62006-08-24 00:41:19 +0000428 def test_eq(self):
429 self.assertEqual({}, {})
Guido van Rossume2a383d2007-01-15 16:59:06 +0000430 self.assertEqual({1: 2}, {1: 2})
Walter Dörwald59b23e82004-09-30 13:46:00 +0000431
432 class Exc(Exception): pass
433
434 class BadCmp(object):
435 def __eq__(self, other):
436 raise Exc()
Guido van Rossum38938152006-08-21 23:36:26 +0000437 def __hash__(self):
Guido van Rossum47b9ff62006-08-24 00:41:19 +0000438 return 1
Walter Dörwald59b23e82004-09-30 13:46:00 +0000439
440 d1 = {BadCmp(): 1}
441 d2 = {1: 1}
Florent Xiclunaa988e422010-03-02 16:06:24 +0000442
443 with self.assertRaises(Exc):
Guido van Rossum47b9ff62006-08-24 00:41:19 +0000444 d1 == d2
Walter Dörwald59b23e82004-09-30 13:46:00 +0000445
Guido van Rossumaac530c2007-08-24 22:33:45 +0000446 def test_keys_contained(self):
Guido van Rossumcf2ce362007-08-24 23:49:54 +0000447 self.helper_keys_contained(lambda x: x.keys())
448 self.helper_keys_contained(lambda x: x.items())
449
450 def helper_keys_contained(self, fn):
Guido van Rossumaac530c2007-08-24 22:33:45 +0000451 # Test rich comparisons against dict key views, which should behave the
452 # same as sets.
Guido van Rossumcf2ce362007-08-24 23:49:54 +0000453 empty = fn(dict())
454 empty2 = fn(dict())
455 smaller = fn({1:1, 2:2})
456 larger = fn({1:1, 2:2, 3:3})
457 larger2 = fn({1:1, 2:2, 3:3})
458 larger3 = fn({4:1, 2:2, 3:3})
Guido van Rossumaac530c2007-08-24 22:33:45 +0000459
Guido van Rossumcf2ce362007-08-24 23:49:54 +0000460 self.assertTrue(smaller < larger)
461 self.assertTrue(smaller <= larger)
462 self.assertTrue(larger > smaller)
463 self.assertTrue(larger >= smaller)
Guido van Rossumaac530c2007-08-24 22:33:45 +0000464
Guido van Rossumcf2ce362007-08-24 23:49:54 +0000465 self.assertFalse(smaller >= larger)
466 self.assertFalse(smaller > larger)
467 self.assertFalse(larger <= smaller)
468 self.assertFalse(larger < smaller)
Guido van Rossumaac530c2007-08-24 22:33:45 +0000469
Guido van Rossumcf2ce362007-08-24 23:49:54 +0000470 self.assertFalse(smaller < larger3)
471 self.assertFalse(smaller <= larger3)
472 self.assertFalse(larger3 > smaller)
473 self.assertFalse(larger3 >= smaller)
Guido van Rossumaac530c2007-08-24 22:33:45 +0000474
475 # Inequality strictness
Guido van Rossumcf2ce362007-08-24 23:49:54 +0000476 self.assertTrue(larger2 >= larger)
477 self.assertTrue(larger2 <= larger)
478 self.assertFalse(larger2 > larger)
479 self.assertFalse(larger2 < larger)
Guido van Rossumaac530c2007-08-24 22:33:45 +0000480
Guido van Rossumcf2ce362007-08-24 23:49:54 +0000481 self.assertTrue(larger == larger2)
482 self.assertTrue(smaller != larger)
Guido van Rossumaac530c2007-08-24 22:33:45 +0000483
484 # There is an optimization on the zero-element case.
Guido van Rossumcf2ce362007-08-24 23:49:54 +0000485 self.assertTrue(empty == empty2)
486 self.assertFalse(empty != empty2)
487 self.assertFalse(empty == smaller)
488 self.assertTrue(empty != smaller)
Guido van Rossumaac530c2007-08-24 22:33:45 +0000489
490 # With the same size, an elementwise compare happens
Guido van Rossumcf2ce362007-08-24 23:49:54 +0000491 self.assertTrue(larger != larger3)
492 self.assertFalse(larger == larger3)
Guido van Rossumaac530c2007-08-24 22:33:45 +0000493
494 def test_errors_in_view_containment_check(self):
495 class C:
496 def __eq__(self, other):
497 raise RuntimeError
Florent Xiclunaa988e422010-03-02 16:06:24 +0000498
Guido van Rossumaac530c2007-08-24 22:33:45 +0000499 d1 = {1: C()}
500 d2 = {1: C()}
Florent Xiclunaa988e422010-03-02 16:06:24 +0000501 with self.assertRaises(RuntimeError):
502 d1.items() == d2.items()
503 with self.assertRaises(RuntimeError):
504 d1.items() != d2.items()
505 with self.assertRaises(RuntimeError):
506 d1.items() <= d2.items()
507 with self.assertRaises(RuntimeError):
508 d1.items() >= d2.items()
509
Guido van Rossumaac530c2007-08-24 22:33:45 +0000510 d3 = {1: C(), 2: C()}
Florent Xiclunaa988e422010-03-02 16:06:24 +0000511 with self.assertRaises(RuntimeError):
512 d2.items() < d3.items()
513 with self.assertRaises(RuntimeError):
514 d3.items() > d2.items()
Guido van Rossumaac530c2007-08-24 22:33:45 +0000515
Guido van Rossumbe534712007-08-24 23:43:52 +0000516 def test_dictview_set_operations_on_keys(self):
Guido van Rossum523259b2007-08-24 23:41:22 +0000517 k1 = {1:1, 2:2}.keys()
518 k2 = {1:1, 2:2, 3:3}.keys()
519 k3 = {4:4}.keys()
520
Florent Xiclunaa988e422010-03-02 16:06:24 +0000521 self.assertEqual(k1 - k2, set())
522 self.assertEqual(k1 - k3, {1,2})
523 self.assertEqual(k2 - k1, {3})
524 self.assertEqual(k3 - k1, {4})
525 self.assertEqual(k1 & k2, {1,2})
526 self.assertEqual(k1 & k3, set())
527 self.assertEqual(k1 | k2, {1,2,3})
528 self.assertEqual(k1 ^ k2, {3})
529 self.assertEqual(k1 ^ k3, {1,2,4})
Guido van Rossum523259b2007-08-24 23:41:22 +0000530
Guido van Rossumbe534712007-08-24 23:43:52 +0000531 def test_dictview_set_operations_on_items(self):
532 k1 = {1:1, 2:2}.items()
533 k2 = {1:1, 2:2, 3:3}.items()
534 k3 = {4:4}.items()
535
Florent Xiclunaa988e422010-03-02 16:06:24 +0000536 self.assertEqual(k1 - k2, set())
537 self.assertEqual(k1 - k3, {(1,1), (2,2)})
538 self.assertEqual(k2 - k1, {(3,3)})
539 self.assertEqual(k3 - k1, {(4,4)})
540 self.assertEqual(k1 & k2, {(1,1), (2,2)})
541 self.assertEqual(k1 & k3, set())
542 self.assertEqual(k1 | k2, {(1,1), (2,2), (3,3)})
543 self.assertEqual(k1 ^ k2, {(3,3)})
544 self.assertEqual(k1 ^ k3, {(1,1), (2,2), (4,4)})
Guido van Rossum523259b2007-08-24 23:41:22 +0000545
Guido van Rossum1d719962007-08-24 23:47:30 +0000546 def test_dictview_mixed_set_operations(self):
547 # Just a few for .keys()
Guido van Rossuma401bbe2007-08-24 23:51:55 +0000548 self.assertTrue({1:1}.keys() == {1})
549 self.assertTrue({1} == {1:1}.keys())
Florent Xiclunaa988e422010-03-02 16:06:24 +0000550 self.assertEqual({1:1}.keys() | {2}, {1, 2})
551 self.assertEqual({2} | {1:1}.keys(), {1, 2})
Guido van Rossum1d719962007-08-24 23:47:30 +0000552 # And a few for .items()
Guido van Rossuma401bbe2007-08-24 23:51:55 +0000553 self.assertTrue({1:1}.items() == {(1,1)})
554 self.assertTrue({(1,1)} == {1:1}.items())
Florent Xiclunaa988e422010-03-02 16:06:24 +0000555 self.assertEqual({1:1}.items() | {2}, {(1,1), 2})
556 self.assertEqual({2} | {1:1}.items(), {(1,1), 2})
Guido van Rossum1d719962007-08-24 23:47:30 +0000557
Guido van Rossum1968ad32006-02-25 22:38:04 +0000558 def test_missing(self):
559 # Make sure dict doesn't have a __missing__ method
Florent Xiclunaa988e422010-03-02 16:06:24 +0000560 self.assertFalse(hasattr(dict, "__missing__"))
561 self.assertFalse(hasattr({}, "__missing__"))
Guido van Rossum1968ad32006-02-25 22:38:04 +0000562 # Test several cases:
563 # (D) subclass defines __missing__ method returning a value
564 # (E) subclass defines __missing__ method raising RuntimeError
565 # (F) subclass sets __missing__ instance variable (no effect)
566 # (G) subclass doesn't define __missing__ at a all
567 class D(dict):
568 def __missing__(self, key):
569 return 42
570 d = D({1: 2, 3: 4})
571 self.assertEqual(d[1], 2)
572 self.assertEqual(d[3], 4)
Benjamin Peterson577473f2010-01-19 00:09:57 +0000573 self.assertNotIn(2, d)
Benjamin Peterson577473f2010-01-19 00:09:57 +0000574 self.assertNotIn(2, d.keys())
Guido van Rossum1968ad32006-02-25 22:38:04 +0000575 self.assertEqual(d[2], 42)
Florent Xiclunaa988e422010-03-02 16:06:24 +0000576
Guido van Rossum1968ad32006-02-25 22:38:04 +0000577 class E(dict):
578 def __missing__(self, key):
579 raise RuntimeError(key)
580 e = E()
Florent Xiclunaa988e422010-03-02 16:06:24 +0000581 with self.assertRaises(RuntimeError) as c:
Guido van Rossum1968ad32006-02-25 22:38:04 +0000582 e[42]
Florent Xiclunaa988e422010-03-02 16:06:24 +0000583 self.assertEqual(c.exception.args, (42,))
584
Guido van Rossum1968ad32006-02-25 22:38:04 +0000585 class F(dict):
586 def __init__(self):
587 # An instance variable __missing__ should have no effect
588 self.__missing__ = lambda key: None
589 f = F()
Florent Xiclunaa988e422010-03-02 16:06:24 +0000590 with self.assertRaises(KeyError) as c:
Guido van Rossum1968ad32006-02-25 22:38:04 +0000591 f[42]
Florent Xiclunaa988e422010-03-02 16:06:24 +0000592 self.assertEqual(c.exception.args, (42,))
593
Guido van Rossum1968ad32006-02-25 22:38:04 +0000594 class G(dict):
595 pass
596 g = G()
Florent Xiclunaa988e422010-03-02 16:06:24 +0000597 with self.assertRaises(KeyError) as c:
Guido van Rossum1968ad32006-02-25 22:38:04 +0000598 g[42]
Florent Xiclunaa988e422010-03-02 16:06:24 +0000599 self.assertEqual(c.exception.args, (42,))
Guido van Rossum1968ad32006-02-25 22:38:04 +0000600
Thomas Wouters89f507f2006-12-13 04:49:30 +0000601 def test_tuple_keyerror(self):
602 # SF #1576657
603 d = {}
Florent Xiclunaa988e422010-03-02 16:06:24 +0000604 with self.assertRaises(KeyError) as c:
Thomas Wouters89f507f2006-12-13 04:49:30 +0000605 d[(1,)]
Florent Xiclunaa988e422010-03-02 16:06:24 +0000606 self.assertEqual(c.exception.args, ((1,),))
Thomas Wouters89f507f2006-12-13 04:49:30 +0000607
Guido van Rossumd8faa362007-04-27 19:54:29 +0000608 def test_bad_key(self):
Mark Dickinsona56c4672009-01-27 18:17:45 +0000609 # Dictionary lookups should fail if __eq__() raises an exception.
Guido van Rossumd8faa362007-04-27 19:54:29 +0000610 class CustomException(Exception):
611 pass
612
613 class BadDictKey:
614 def __hash__(self):
615 return hash(self.__class__)
616
617 def __eq__(self, other):
618 if isinstance(other, self.__class__):
619 raise CustomException
620 return other
621
622 d = {}
623 x1 = BadDictKey()
624 x2 = BadDictKey()
625 d[x1] = 1
626 for stmt in ['d[x2] = 2',
627 'z = d[x2]',
628 'x2 in d',
629 'd.get(x2)',
630 'd.setdefault(x2, 42)',
631 'd.pop(x2)',
632 'd.update({x2: 2})']:
Florent Xiclunaa988e422010-03-02 16:06:24 +0000633 with self.assertRaises(CustomException):
Guido van Rossumd8faa362007-04-27 19:54:29 +0000634 exec(stmt, locals())
Guido van Rossumd8faa362007-04-27 19:54:29 +0000635
636 def test_resize1(self):
637 # Dict resizing bug, found by Jack Jansen in 2.2 CVS development.
638 # This version got an assert failure in debug build, infinite loop in
639 # release build. Unfortunately, provoking this kind of stuff requires
640 # a mix of inserts and deletes hitting exactly the right hash codes in
641 # exactly the right order, and I can't think of a randomized approach
642 # that would be *likely* to hit a failing case in reasonable time.
643
644 d = {}
645 for i in range(5):
646 d[i] = i
647 for i in range(5):
648 del d[i]
649 for i in range(5, 9): # i==8 was the problem
650 d[i] = i
651
652 def test_resize2(self):
653 # Another dict resizing bug (SF bug #1456209).
654 # This caused Segmentation faults or Illegal instructions.
655
656 class X(object):
657 def __hash__(self):
658 return 5
659 def __eq__(self, other):
660 if resizing:
661 d.clear()
662 return False
663 d = {}
664 resizing = False
665 d[X()] = 1
666 d[X()] = 2
667 d[X()] = 3
668 d[X()] = 4
669 d[X()] = 5
670 # now trigger a resize
671 resizing = True
672 d[9] = 6
673
Georg Brandl6aa2d1f2008-08-12 08:35:52 +0000674 def test_empty_presized_dict_in_freelist(self):
675 # Bug #3537: if an empty but presized dict with a size larger
676 # than 7 was in the freelist, it triggered an assertion failure
Florent Xiclunaa988e422010-03-02 16:06:24 +0000677 with self.assertRaises(ZeroDivisionError):
678 d = {'a': 1 // 0, 'b': None, 'c': None, 'd': None, 'e': None,
Georg Brandl6aa2d1f2008-08-12 08:35:52 +0000679 'f': None, 'g': None, 'h': None}
Georg Brandl6aa2d1f2008-08-12 08:35:52 +0000680 d = {}
681
Antoine Pitrou7ddda782009-01-01 15:35:33 +0000682 def test_container_iterator(self):
683 # Bug #3680: tp_traverse was not implemented for dictiter and
684 # dictview objects.
685 class C(object):
686 pass
687 views = (dict.items, dict.values, dict.keys)
688 for v in views:
689 obj = C()
690 ref = weakref.ref(obj)
691 container = {obj: 1}
692 obj.v = v(container)
693 obj.x = iter(obj.v)
694 del obj, container
695 gc.collect()
Florent Xiclunaa988e422010-03-02 16:06:24 +0000696 self.assertIs(ref(), None, "Cycle was not collected")
Georg Brandl6aa2d1f2008-08-12 08:35:52 +0000697
Antoine Pitrou3a652b12009-03-23 18:52:06 +0000698 def _not_tracked(self, t):
699 # Nested containers can take several collections to untrack
700 gc.collect()
701 gc.collect()
702 self.assertFalse(gc.is_tracked(t), t)
703
704 def _tracked(self, t):
705 self.assertTrue(gc.is_tracked(t), t)
706 gc.collect()
707 gc.collect()
708 self.assertTrue(gc.is_tracked(t), t)
709
Benjamin Peterson3e5cd1d2010-06-27 21:45:24 +0000710 @support.cpython_only
Antoine Pitrou3a652b12009-03-23 18:52:06 +0000711 def test_track_literals(self):
712 # Test GC-optimization of dict literals
713 x, y, z, w = 1.5, "a", (1, None), []
714
715 self._not_tracked({})
716 self._not_tracked({x:(), y:x, z:1})
717 self._not_tracked({1: "a", "b": 2})
718 self._not_tracked({1: 2, (None, True, False, ()): int})
719 self._not_tracked({1: object()})
720
721 # Dicts with mutable elements are always tracked, even if those
722 # elements are not tracked right now.
723 self._tracked({1: []})
724 self._tracked({1: ([],)})
725 self._tracked({1: {}})
726 self._tracked({1: set()})
727
Benjamin Peterson3e5cd1d2010-06-27 21:45:24 +0000728 @support.cpython_only
Antoine Pitrou3a652b12009-03-23 18:52:06 +0000729 def test_track_dynamic(self):
730 # Test GC-optimization of dynamically-created dicts
731 class MyObject(object):
732 pass
733 x, y, z, w, o = 1.5, "a", (1, object()), [], MyObject()
734
735 d = dict()
736 self._not_tracked(d)
737 d[1] = "a"
738 self._not_tracked(d)
739 d[y] = 2
740 self._not_tracked(d)
741 d[z] = 3
742 self._not_tracked(d)
743 self._not_tracked(d.copy())
744 d[4] = w
745 self._tracked(d)
746 self._tracked(d.copy())
747 d[4] = None
748 self._not_tracked(d)
749 self._not_tracked(d.copy())
750
751 # dd isn't tracked right now, but it may mutate and therefore d
752 # which contains it must be tracked.
753 d = dict()
754 dd = dict()
755 d[1] = dd
756 self._not_tracked(dd)
757 self._tracked(d)
758 dd[1] = d
759 self._tracked(dd)
760
761 d = dict.fromkeys([x, y, z])
762 self._not_tracked(d)
763 dd = dict()
764 dd.update(d)
765 self._not_tracked(dd)
766 d = dict.fromkeys([x, y, z, o])
767 self._tracked(d)
768 dd = dict()
769 dd.update(d)
770 self._tracked(dd)
771
772 d = dict(x=x, y=y, z=z)
773 self._not_tracked(d)
774 d = dict(x=x, y=y, z=z, w=w)
775 self._tracked(d)
776 d = dict()
777 d.update(x=x, y=y, z=z)
778 self._not_tracked(d)
779 d.update(w=w)
780 self._tracked(d)
781
782 d = dict([(x, y), (z, 1)])
783 self._not_tracked(d)
784 d = dict([(x, y), (z, w)])
785 self._tracked(d)
786 d = dict()
787 d.update([(x, y), (z, 1)])
788 self._not_tracked(d)
789 d.update([(x, y), (z, w)])
790 self._tracked(d)
791
Benjamin Peterson3e5cd1d2010-06-27 21:45:24 +0000792 @support.cpython_only
Antoine Pitrou3a652b12009-03-23 18:52:06 +0000793 def test_track_subtypes(self):
794 # Dict subtypes are always tracked
795 class MyDict(dict):
796 pass
797 self._tracked(MyDict())
798
Guido van Rossum1968ad32006-02-25 22:38:04 +0000799
Neal Norwitzc3e54b82006-03-24 07:38:37 +0000800from test import mapping_tests
Raymond Hettinger49c522b2004-09-30 15:07:29 +0000801
802class GeneralMappingTests(mapping_tests.BasicTestMappingProtocol):
803 type2test = dict
804
805class Dict(dict):
806 pass
807
808class SubclassMappingTests(mapping_tests.BasicTestMappingProtocol):
809 type2test = Dict
810
Walter Dörwald59b23e82004-09-30 13:46:00 +0000811def test_main():
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000812 support.run_unittest(
Walter Dörwald59b23e82004-09-30 13:46:00 +0000813 DictTest,
Raymond Hettinger49c522b2004-09-30 15:07:29 +0000814 GeneralMappingTests,
815 SubclassMappingTests,
Walter Dörwald59b23e82004-09-30 13:46:00 +0000816 )
817
818if __name__ == "__main__":
819 test_main()