blob: 7295f41b7864e3fa4f961b769b323af94b9d4bb1 [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()
Guido van Rossume2b70bc2006-08-18 22:13:04 +000024 self.assert_('a' in d)
25 self.assert_('b' in d)
Walter Dörwald59b23e82004-09-30 13:46:00 +000026
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
Walter Dörwald59b23e82004-09-30 13:46:00 +000046 def test_contains(self):
47 d = {}
48 self.assert_(not ('a' in d))
49 self.assert_('a' not in d)
50 d = {'a': 1, 'b': 2}
51 self.assert_('a' in d)
52 self.assert_('b' in d)
53 self.assert_('c' not in d)
54
55 self.assertRaises(TypeError, d.__contains__)
56
57 def test_len(self):
58 d = {}
59 self.assertEqual(len(d), 0)
60 d = {'a': 1, 'b': 2}
61 self.assertEqual(len(d), 2)
62
63 def test_getitem(self):
64 d = {'a': 1, 'b': 2}
65 self.assertEqual(d['a'], 1)
66 self.assertEqual(d['b'], 2)
67 d['c'] = 3
68 d['a'] = 4
69 self.assertEqual(d['c'], 3)
70 self.assertEqual(d['a'], 4)
71 del d['b']
72 self.assertEqual(d, {'a': 4, 'c': 3})
73
74 self.assertRaises(TypeError, d.__getitem__)
75
76 class BadEq(object):
77 def __eq__(self, other):
78 raise Exc()
Guido van Rossum38938152006-08-21 23:36:26 +000079 def __hash__(self):
80 return 24
Walter Dörwald59b23e82004-09-30 13:46:00 +000081
82 d = {}
83 d[BadEq()] = 42
84 self.assertRaises(KeyError, d.__getitem__, 23)
85
86 class Exc(Exception): pass
87
88 class BadHash(object):
89 fail = False
90 def __hash__(self):
91 if self.fail:
92 raise Exc()
93 else:
94 return 42
95
96 x = BadHash()
97 d[x] = 42
98 x.fail = True
99 self.assertRaises(Exc, d.__getitem__, x)
100
101 def test_clear(self):
102 d = {1:1, 2:2, 3:3}
103 d.clear()
104 self.assertEqual(d, {})
105
106 self.assertRaises(TypeError, d.clear, None)
107
108 def test_update(self):
109 d = {}
110 d.update({1:100})
111 d.update({2:20})
112 d.update({1:1, 2:2, 3:3})
113 self.assertEqual(d, {1:1, 2:2, 3:3})
114
115 d.update()
116 self.assertEqual(d, {1:1, 2:2, 3:3})
117
118 self.assertRaises((TypeError, AttributeError), d.update, None)
119
120 class SimpleUserDict:
121 def __init__(self):
122 self.d = {1:1, 2:2, 3:3}
123 def keys(self):
124 return self.d.keys()
125 def __getitem__(self, i):
126 return self.d[i]
127 d.clear()
128 d.update(SimpleUserDict())
129 self.assertEqual(d, {1:1, 2:2, 3:3})
130
131 class Exc(Exception): pass
132
133 d.clear()
134 class FailingUserDict:
135 def keys(self):
136 raise Exc
137 self.assertRaises(Exc, d.update, FailingUserDict())
138
139 class FailingUserDict:
140 def keys(self):
141 class BogonIter:
142 def __init__(self):
143 self.i = 1
144 def __iter__(self):
145 return self
146 def next(self):
147 if self.i:
148 self.i = 0
149 return 'a'
150 raise Exc
151 return BogonIter()
152 def __getitem__(self, key):
153 return key
154 self.assertRaises(Exc, d.update, FailingUserDict())
155
156 class FailingUserDict:
157 def keys(self):
158 class BogonIter:
159 def __init__(self):
160 self.i = ord('a')
161 def __iter__(self):
162 return self
163 def next(self):
164 if self.i <= ord('z'):
165 rtn = chr(self.i)
166 self.i += 1
167 return rtn
168 raise StopIteration
169 return BogonIter()
170 def __getitem__(self, key):
171 raise Exc
172 self.assertRaises(Exc, d.update, FailingUserDict())
173
174 class badseq(object):
175 def __iter__(self):
176 return self
177 def next(self):
178 raise Exc()
179
180 self.assertRaises(Exc, {}.update, badseq())
181
182 self.assertRaises(ValueError, {}.update, [(1, 2, 3)])
183
184 def test_fromkeys(self):
185 self.assertEqual(dict.fromkeys('abc'), {'a':None, 'b':None, 'c':None})
186 d = {}
187 self.assert_(not(d.fromkeys('abc') is d))
188 self.assertEqual(d.fromkeys('abc'), {'a':None, 'b':None, 'c':None})
189 self.assertEqual(d.fromkeys((4,5),0), {4:0, 5:0})
190 self.assertEqual(d.fromkeys([]), {})
191 def g():
192 yield 1
193 self.assertEqual(d.fromkeys(g()), {1:None})
194 self.assertRaises(TypeError, {}.fromkeys, 3)
195 class dictlike(dict): pass
196 self.assertEqual(dictlike.fromkeys('a'), {'a':None})
197 self.assertEqual(dictlike().fromkeys('a'), {'a':None})
198 self.assert_(type(dictlike.fromkeys('a')) is dictlike)
199 self.assert_(type(dictlike().fromkeys('a')) is dictlike)
200 class mydict(dict):
201 def __new__(cls):
202 return UserDict.UserDict()
203 ud = mydict.fromkeys('ab')
204 self.assertEqual(ud, {'a':None, 'b':None})
205 self.assert_(isinstance(ud, UserDict.UserDict))
206 self.assertRaises(TypeError, dict.fromkeys)
207
208 class Exc(Exception): pass
209
210 class baddict1(dict):
211 def __init__(self):
212 raise Exc()
213
214 self.assertRaises(Exc, baddict1.fromkeys, [1])
215
216 class BadSeq(object):
217 def __iter__(self):
218 return self
219 def next(self):
220 raise Exc()
221
222 self.assertRaises(Exc, dict.fromkeys, BadSeq())
223
224 class baddict2(dict):
225 def __setitem__(self, key, value):
226 raise Exc()
227
228 self.assertRaises(Exc, baddict2.fromkeys, [1])
229
230 def test_copy(self):
231 d = {1:1, 2:2, 3:3}
232 self.assertEqual(d.copy(), {1:1, 2:2, 3:3})
233 self.assertEqual({}.copy(), {})
234 self.assertRaises(TypeError, d.copy, None)
235
236 def test_get(self):
237 d = {}
238 self.assert_(d.get('c') is None)
239 self.assertEqual(d.get('c', 3), 3)
240 d = {'a' : 1, 'b' : 2}
241 self.assert_(d.get('c') is None)
242 self.assertEqual(d.get('c', 3), 3)
243 self.assertEqual(d.get('a'), 1)
244 self.assertEqual(d.get('a', 3), 1)
245 self.assertRaises(TypeError, d.get)
246 self.assertRaises(TypeError, d.get, None, None, None)
247
248 def test_setdefault(self):
249 # dict.setdefault()
250 d = {}
251 self.assert_(d.setdefault('key0') is None)
252 d.setdefault('key0', [])
253 self.assert_(d.setdefault('key0') is None)
254 d.setdefault('key', []).append(3)
255 self.assertEqual(d['key'][0], 3)
256 d.setdefault('key', []).append(4)
257 self.assertEqual(len(d['key']), 2)
258 self.assertRaises(TypeError, d.setdefault)
259
260 class Exc(Exception): pass
261
262 class BadHash(object):
263 fail = False
264 def __hash__(self):
265 if self.fail:
266 raise Exc()
267 else:
268 return 42
269
270 x = BadHash()
271 d[x] = 42
272 x.fail = True
273 self.assertRaises(Exc, d.setdefault, x, [])
274
275 def test_popitem(self):
276 # dict.popitem()
277 for copymode in -1, +1:
278 # -1: b has same structure as a
279 # +1: b is a.copy()
280 for log2size in range(12):
281 size = 2**log2size
282 a = {}
283 b = {}
284 for i in range(size):
285 a[repr(i)] = i
286 if copymode < 0:
287 b[repr(i)] = i
288 if copymode > 0:
289 b = a.copy()
290 for i in range(size):
291 ka, va = ta = a.popitem()
292 self.assertEqual(va, int(ka))
293 kb, vb = tb = b.popitem()
294 self.assertEqual(vb, int(kb))
295 self.assert_(not(copymode < 0 and ta != tb))
296 self.assert_(not a)
297 self.assert_(not b)
298
299 d = {}
300 self.assertRaises(KeyError, d.popitem)
301
302 def test_pop(self):
303 # Tests for pop with specified key
304 d = {}
305 k, v = 'abc', 'def'
306 d[k] = v
307 self.assertRaises(KeyError, d.pop, 'ghi')
308
309 self.assertEqual(d.pop(k), v)
310 self.assertEqual(len(d), 0)
311
312 self.assertRaises(KeyError, d.pop, k)
313
314 # verify longs/ints get same value when key > 32 bits (for 64-bit archs)
315 # see SF bug #689659
316 x = 4503599627370496L
317 y = 4503599627370496
318 h = {x: 'anything', y: 'something else'}
319 self.assertEqual(h[x], h[y])
320
321 self.assertEqual(d.pop(k, v), v)
322 d[k] = v
323 self.assertEqual(d.pop(k, 1), v)
324
325 self.assertRaises(TypeError, d.pop)
326
327 class Exc(Exception): pass
328
329 class BadHash(object):
330 fail = False
331 def __hash__(self):
332 if self.fail:
333 raise Exc()
334 else:
335 return 42
336
337 x = BadHash()
338 d[x] = 42
339 x.fail = True
340 self.assertRaises(Exc, d.pop, x)
341
342 def test_mutatingiteration(self):
343 d = {}
344 d[1] = 1
345 try:
346 for i in d:
347 d[i+1] = 1
348 except RuntimeError:
349 pass
350 else:
351 self.fail("changing dict size during iteration doesn't raise Error")
352
353 def test_repr(self):
354 d = {}
355 self.assertEqual(repr(d), '{}')
356 d[1] = 2
357 self.assertEqual(repr(d), '{1: 2}')
358 d = {}
359 d[1] = d
360 self.assertEqual(repr(d), '{1: {...}}')
361
362 class Exc(Exception): pass
363
364 class BadRepr(object):
365 def __repr__(self):
366 raise Exc()
367
368 d = {1: BadRepr()}
369 self.assertRaises(Exc, repr, d)
370
371 def test_le(self):
372 self.assert_(not ({} < {}))
373 self.assert_(not ({1: 2} < {1L: 2L}))
374
375 class Exc(Exception): pass
376
377 class BadCmp(object):
378 def __eq__(self, other):
379 raise Exc()
Guido van Rossum38938152006-08-21 23:36:26 +0000380 def __hash__(self):
381 return 42
Walter Dörwald59b23e82004-09-30 13:46:00 +0000382
383 d1 = {BadCmp(): 1}
384 d2 = {1: 1}
385 try:
386 d1 < d2
387 except Exc:
388 pass
389 else:
390 self.fail("< didn't raise Exc")
391
Guido van Rossum1968ad32006-02-25 22:38:04 +0000392 def test_missing(self):
393 # Make sure dict doesn't have a __missing__ method
394 self.assertEqual(hasattr(dict, "__missing__"), False)
395 self.assertEqual(hasattr({}, "__missing__"), False)
396 # Test several cases:
397 # (D) subclass defines __missing__ method returning a value
398 # (E) subclass defines __missing__ method raising RuntimeError
399 # (F) subclass sets __missing__ instance variable (no effect)
400 # (G) subclass doesn't define __missing__ at a all
401 class D(dict):
402 def __missing__(self, key):
403 return 42
404 d = D({1: 2, 3: 4})
405 self.assertEqual(d[1], 2)
406 self.assertEqual(d[3], 4)
407 self.assert_(2 not in d)
408 self.assert_(2 not in d.keys())
409 self.assertEqual(d[2], 42)
410 class E(dict):
411 def __missing__(self, key):
412 raise RuntimeError(key)
413 e = E()
414 try:
415 e[42]
416 except RuntimeError, err:
417 self.assertEqual(err.args, (42,))
418 else:
419 self.fail_("e[42] didn't raise RuntimeError")
420 class F(dict):
421 def __init__(self):
422 # An instance variable __missing__ should have no effect
423 self.__missing__ = lambda key: None
424 f = F()
425 try:
426 f[42]
427 except KeyError, err:
428 self.assertEqual(err.args, (42,))
429 else:
430 self.fail_("f[42] didn't raise KeyError")
431 class G(dict):
432 pass
433 g = G()
434 try:
435 g[42]
436 except KeyError, err:
437 self.assertEqual(err.args, (42,))
438 else:
439 self.fail_("g[42] didn't raise KeyError")
440
441
Neal Norwitzc3e54b82006-03-24 07:38:37 +0000442from test import mapping_tests
Raymond Hettinger49c522b2004-09-30 15:07:29 +0000443
444class GeneralMappingTests(mapping_tests.BasicTestMappingProtocol):
445 type2test = dict
446
447class Dict(dict):
448 pass
449
450class SubclassMappingTests(mapping_tests.BasicTestMappingProtocol):
451 type2test = Dict
452
Walter Dörwald59b23e82004-09-30 13:46:00 +0000453def test_main():
454 test_support.run_unittest(
455 DictTest,
Raymond Hettinger49c522b2004-09-30 15:07:29 +0000456 GeneralMappingTests,
457 SubclassMappingTests,
Walter Dörwald59b23e82004-09-30 13:46:00 +0000458 )
459
460if __name__ == "__main__":
461 test_main()