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