blob: 8c724638e71ad2da03140f57428699f97578c935 [file] [log] [blame]
Guido van Rossum3eccc481999-03-26 15:32:05 +00001# Check every path through every method of UserDict
2
Walter Dörwald0a6d0ff2004-05-31 16:29:04 +00003import unittest
4from test import test_support, mapping_tests
Walter Dörwalde28be592003-01-19 23:26:59 +00005import UserDict
Guido van Rossum3eccc481999-03-26 15:32:05 +00006
7d0 = {}
8d1 = {"one": 1}
9d2 = {"one": 1, "two": 2}
Raymond Hettingere4827eb2002-11-27 08:29:11 +000010d3 = {"one": 1, "two": 3, "three": 5}
11d4 = {"one": None, "two": None}
12d5 = {"one": 1, "two": 1}
Guido van Rossum3eccc481999-03-26 15:32:05 +000013
Walter Dörwald0a6d0ff2004-05-31 16:29:04 +000014class UserDictTest(mapping_tests.TestHashMappingProtocol):
Walter Dörwald118f9312004-06-02 18:42:25 +000015 type2test = UserDict.IterableUserDict
Raymond Hettinger2c2d3222003-03-09 07:05:43 +000016
Walter Dörwalde28be592003-01-19 23:26:59 +000017 def test_all(self):
18 # Test constructors
19 u = UserDict.UserDict()
20 u0 = UserDict.UserDict(d0)
21 u1 = UserDict.UserDict(d1)
22 u2 = UserDict.IterableUserDict(d2)
Guido van Rossum3eccc481999-03-26 15:32:05 +000023
Walter Dörwalde28be592003-01-19 23:26:59 +000024 uu = UserDict.UserDict(u)
25 uu0 = UserDict.UserDict(u0)
26 uu1 = UserDict.UserDict(u1)
27 uu2 = UserDict.UserDict(u2)
Guido van Rossum3eccc481999-03-26 15:32:05 +000028
Walter Dörwalde28be592003-01-19 23:26:59 +000029 # keyword arg constructor
30 self.assertEqual(UserDict.UserDict(one=1, two=2), d2)
31 # item sequence constructor
32 self.assertEqual(UserDict.UserDict([('one',1), ('two',2)]), d2)
33 self.assertEqual(UserDict.UserDict(dict=[('one',1), ('two',2)]), d2)
34 # both together
35 self.assertEqual(UserDict.UserDict([('one',1), ('two',2)], two=3, three=5), d3)
Guido van Rossum3eccc481999-03-26 15:32:05 +000036
Walter Dörwalde28be592003-01-19 23:26:59 +000037 # alternate constructor
38 self.assertEqual(UserDict.UserDict.fromkeys('one two'.split()), d4)
39 self.assertEqual(UserDict.UserDict().fromkeys('one two'.split()), d4)
40 self.assertEqual(UserDict.UserDict.fromkeys('one two'.split(), 1), d5)
41 self.assertEqual(UserDict.UserDict().fromkeys('one two'.split(), 1), d5)
42 self.assert_(u1.fromkeys('one two'.split()) is not u1)
43 self.assert_(isinstance(u1.fromkeys('one two'.split()), UserDict.UserDict))
44 self.assert_(isinstance(u2.fromkeys('one two'.split()), UserDict.IterableUserDict))
Raymond Hettingere4827eb2002-11-27 08:29:11 +000045
Walter Dörwalde28be592003-01-19 23:26:59 +000046 # Test __repr__
47 self.assertEqual(str(u0), str(d0))
48 self.assertEqual(repr(u1), repr(d1))
Brett Cannon0b70cca2006-08-25 02:59:59 +000049 self.assertEqual(repr(u2), repr(d2))
Raymond Hettinger54405452002-11-22 00:07:40 +000050
Walter Dörwalde28be592003-01-19 23:26:59 +000051 # Test __cmp__ and __len__
52 all = [d0, d1, d2, u, u0, u1, u2, uu, uu0, uu1, uu2]
53 for a in all:
54 for b in all:
Guido van Rossum47b9ff62006-08-24 00:41:19 +000055 self.assertEqual(a == b, len(a) == len(b))
Guido van Rossum3eccc481999-03-26 15:32:05 +000056
Walter Dörwalde28be592003-01-19 23:26:59 +000057 # Test __getitem__
58 self.assertEqual(u2["one"], 1)
59 self.assertRaises(KeyError, u1.__getitem__, "two")
Guido van Rossum3eccc481999-03-26 15:32:05 +000060
Walter Dörwalde28be592003-01-19 23:26:59 +000061 # Test __setitem__
62 u3 = UserDict.UserDict(u2)
63 u3["two"] = 2
64 u3["three"] = 3
Guido van Rossum3eccc481999-03-26 15:32:05 +000065
Walter Dörwalde28be592003-01-19 23:26:59 +000066 # Test __delitem__
67 del u3["three"]
68 self.assertRaises(KeyError, u3.__delitem__, "three")
Guido van Rossum3eccc481999-03-26 15:32:05 +000069
Walter Dörwalde28be592003-01-19 23:26:59 +000070 # Test clear
71 u3.clear()
72 self.assertEqual(u3, {})
Guido van Rossum3eccc481999-03-26 15:32:05 +000073
Walter Dörwalde28be592003-01-19 23:26:59 +000074 # Test copy()
75 u2a = u2.copy()
76 self.assertEqual(u2a, u2)
77 u2b = UserDict.UserDict(x=42, y=23)
78 u2c = u2b.copy() # making a copy of a UserDict is special cased
79 self.assertEqual(u2b, u2c)
Guido van Rossum3eccc481999-03-26 15:32:05 +000080
Walter Dörwalde28be592003-01-19 23:26:59 +000081 class MyUserDict(UserDict.UserDict):
Guido van Rossumbe19ed72007-02-09 05:37:30 +000082 def display(self): print(self)
Guido van Rossum3eccc481999-03-26 15:32:05 +000083
Walter Dörwalde28be592003-01-19 23:26:59 +000084 m2 = MyUserDict(u2)
85 m2a = m2.copy()
86 self.assertEqual(m2a, m2)
Guido van Rossum3eccc481999-03-26 15:32:05 +000087
Walter Dörwalde28be592003-01-19 23:26:59 +000088 # SF bug #476616 -- copy() of UserDict subclass shared data
89 m2['foo'] = 'bar'
90 self.assertNotEqual(m2a, m2)
Guido van Rossum3eccc481999-03-26 15:32:05 +000091
Walter Dörwalde28be592003-01-19 23:26:59 +000092 # Test keys, items, values
93 self.assertEqual(u2.keys(), d2.keys())
94 self.assertEqual(u2.items(), d2.items())
Guido van Rossumd81206d2007-02-15 03:49:08 +000095 self.assertEqual(list(u2.values()), list(d2.values()))
Guido van Rossum3eccc481999-03-26 15:32:05 +000096
Guido van Rossume2b70bc2006-08-18 22:13:04 +000097 # Test "in".
Walter Dörwalde28be592003-01-19 23:26:59 +000098 for i in u2.keys():
Walter Dörwalde28be592003-01-19 23:26:59 +000099 self.assert_(i in u2)
Walter Dörwalde28be592003-01-19 23:26:59 +0000100 self.assertEqual(i in u1, i in d1)
Walter Dörwalde28be592003-01-19 23:26:59 +0000101 self.assertEqual(i in u0, i in d0)
Guido van Rossum3eccc481999-03-26 15:32:05 +0000102
Walter Dörwalde28be592003-01-19 23:26:59 +0000103 # Test update
104 t = UserDict.UserDict()
105 t.update(u2)
106 self.assertEqual(t, u2)
107 class Items:
108 def items(self):
109 return (("x", 42), ("y", 23))
110 t = UserDict.UserDict()
111 t.update(Items())
112 self.assertEqual(t, {"x": 42, "y": 23})
Guido van Rossum3eccc481999-03-26 15:32:05 +0000113
Walter Dörwalde28be592003-01-19 23:26:59 +0000114 # Test get
115 for i in u2.keys():
116 self.assertEqual(u2.get(i), u2[i])
117 self.assertEqual(u1.get(i), d1.get(i))
118 self.assertEqual(u0.get(i), d0.get(i))
Guido van Rossum3eccc481999-03-26 15:32:05 +0000119
Walter Dörwalde28be592003-01-19 23:26:59 +0000120 # Test "in" iteration.
121 for i in xrange(20):
122 u2[i] = str(i)
123 ikeys = []
124 for k in u2:
125 ikeys.append(k)
Walter Dörwalde28be592003-01-19 23:26:59 +0000126 keys = u2.keys()
Raymond Hettingera690a992003-11-16 16:17:49 +0000127 self.assertEqual(set(ikeys), set(keys))
Guido van Rossum3eccc481999-03-26 15:32:05 +0000128
Walter Dörwalde28be592003-01-19 23:26:59 +0000129 # Test setdefault
130 t = UserDict.UserDict()
131 self.assertEqual(t.setdefault("x", 42), 42)
Guido van Rossume2b70bc2006-08-18 22:13:04 +0000132 self.assert_("x" in t)
Walter Dörwalde28be592003-01-19 23:26:59 +0000133 self.assertEqual(t.setdefault("x", 23), 42)
Guido van Rossum3eccc481999-03-26 15:32:05 +0000134
Walter Dörwalde28be592003-01-19 23:26:59 +0000135 # Test pop
136 t = UserDict.UserDict(x=42)
137 self.assertEqual(t.pop("x"), 42)
138 self.assertRaises(KeyError, t.pop, "x")
Raymond Hettingera3e1e4c2003-03-06 23:54:28 +0000139 self.assertEqual(t.pop("x", 1), 1)
140 t["x"] = 42
141 self.assertEqual(t.pop("x", 1), 42)
Guido van Rossum3eccc481999-03-26 15:32:05 +0000142
Walter Dörwalde28be592003-01-19 23:26:59 +0000143 # Test popitem
144 t = UserDict.UserDict(x=42)
145 self.assertEqual(t.popitem(), ("x", 42))
146 self.assertRaises(KeyError, t.popitem)
Raymond Hettinger903bf902002-11-15 08:39:40 +0000147
Guido van Rossum1968ad32006-02-25 22:38:04 +0000148 def test_missing(self):
149 # Make sure UserDict doesn't have a __missing__ method
150 self.assertEqual(hasattr(UserDict, "__missing__"), False)
151 # Test several cases:
152 # (D) subclass defines __missing__ method returning a value
153 # (E) subclass defines __missing__ method raising RuntimeError
154 # (F) subclass sets __missing__ instance variable (no effect)
155 # (G) subclass doesn't define __missing__ at a all
156 class D(UserDict.UserDict):
157 def __missing__(self, key):
158 return 42
159 d = D({1: 2, 3: 4})
160 self.assertEqual(d[1], 2)
161 self.assertEqual(d[3], 4)
162 self.assert_(2 not in d)
163 self.assert_(2 not in d.keys())
164 self.assertEqual(d[2], 42)
165 class E(UserDict.UserDict):
166 def __missing__(self, key):
167 raise RuntimeError(key)
168 e = E()
169 try:
170 e[42]
Guido van Rossumb940e112007-01-10 16:19:56 +0000171 except RuntimeError as err:
Guido van Rossum1968ad32006-02-25 22:38:04 +0000172 self.assertEqual(err.args, (42,))
173 else:
Guido van Rossumd8faa362007-04-27 19:54:29 +0000174 self.fail("e[42] didn't raise RuntimeError")
Guido van Rossum1968ad32006-02-25 22:38:04 +0000175 class F(UserDict.UserDict):
176 def __init__(self):
177 # An instance variable __missing__ should have no effect
178 self.__missing__ = lambda key: None
179 UserDict.UserDict.__init__(self)
180 f = F()
181 try:
182 f[42]
Guido van Rossumb940e112007-01-10 16:19:56 +0000183 except KeyError as err:
Guido van Rossum1968ad32006-02-25 22:38:04 +0000184 self.assertEqual(err.args, (42,))
185 else:
Guido van Rossumd8faa362007-04-27 19:54:29 +0000186 self.fail("f[42] didn't raise KeyError")
Guido van Rossum1968ad32006-02-25 22:38:04 +0000187 class G(UserDict.UserDict):
188 pass
189 g = G()
190 try:
191 g[42]
Guido van Rossumb940e112007-01-10 16:19:56 +0000192 except KeyError as err:
Guido van Rossum1968ad32006-02-25 22:38:04 +0000193 self.assertEqual(err.args, (42,))
194 else:
Guido van Rossumd8faa362007-04-27 19:54:29 +0000195 self.fail("g[42] didn't raise KeyError")
Guido van Rossum1968ad32006-02-25 22:38:04 +0000196
Raymond Hettinger903bf902002-11-15 08:39:40 +0000197##########################
198# Test Dict Mixin
199
Walter Dörwalde28be592003-01-19 23:26:59 +0000200class SeqDict(UserDict.DictMixin):
Raymond Hettinger903bf902002-11-15 08:39:40 +0000201 """Dictionary lookalike implemented with lists.
202
203 Used to test and demonstrate DictMixin
204 """
Walter Dörwald0a6d0ff2004-05-31 16:29:04 +0000205 def __init__(self, other=None, **kwargs):
Raymond Hettinger903bf902002-11-15 08:39:40 +0000206 self.keylist = []
207 self.valuelist = []
Walter Dörwald0a6d0ff2004-05-31 16:29:04 +0000208 if other is not None:
209 for (key, value) in other:
210 self[key] = value
Guido van Rossumcc2b0162007-02-11 06:12:03 +0000211 for (key, value) in kwargs.items():
Walter Dörwald0a6d0ff2004-05-31 16:29:04 +0000212 self[key] = value
Raymond Hettinger903bf902002-11-15 08:39:40 +0000213 def __getitem__(self, key):
214 try:
215 i = self.keylist.index(key)
216 except ValueError:
217 raise KeyError
218 return self.valuelist[i]
219 def __setitem__(self, key, value):
Raymond Hettinger0efa17c2002-12-11 07:16:06 +0000220 try:
221 i = self.keylist.index(key)
222 self.valuelist[i] = value
223 except ValueError:
224 self.keylist.append(key)
225 self.valuelist.append(value)
Raymond Hettinger903bf902002-11-15 08:39:40 +0000226 def __delitem__(self, key):
227 try:
228 i = self.keylist.index(key)
229 except ValueError:
230 raise KeyError
231 self.keylist.pop(i)
232 self.valuelist.pop(i)
233 def keys(self):
234 return list(self.keylist)
Walter Dörwald0a6d0ff2004-05-31 16:29:04 +0000235 def copy(self):
236 d = self.__class__()
Guido van Rossumcc2b0162007-02-11 06:12:03 +0000237 for key, value in self.items():
Walter Dörwald0a6d0ff2004-05-31 16:29:04 +0000238 d[key] = value
239 return d
Guido van Rossum5a8a0372005-01-16 00:25:31 +0000240 @classmethod
Walter Dörwald0a6d0ff2004-05-31 16:29:04 +0000241 def fromkeys(cls, keys, value=None):
242 d = cls()
243 for key in keys:
244 d[key] = value
245 return d
Raymond Hettinger903bf902002-11-15 08:39:40 +0000246
Walter Dörwald0a6d0ff2004-05-31 16:29:04 +0000247class UserDictMixinTest(mapping_tests.TestMappingProtocol):
Walter Dörwald118f9312004-06-02 18:42:25 +0000248 type2test = SeqDict
Raymond Hettinger2c2d3222003-03-09 07:05:43 +0000249
Walter Dörwalde28be592003-01-19 23:26:59 +0000250 def test_all(self):
251 ## Setup test and verify working of the test class
Raymond Hettinger903bf902002-11-15 08:39:40 +0000252
Walter Dörwalde28be592003-01-19 23:26:59 +0000253 # check init
254 s = SeqDict()
Raymond Hettinger903bf902002-11-15 08:39:40 +0000255
Walter Dörwalde28be592003-01-19 23:26:59 +0000256 # exercise setitem
257 s[10] = 'ten'
258 s[20] = 'twenty'
259 s[30] = 'thirty'
Raymond Hettinger903bf902002-11-15 08:39:40 +0000260
Walter Dörwalde28be592003-01-19 23:26:59 +0000261 # exercise delitem
262 del s[20]
263 # check getitem and setitem
264 self.assertEqual(s[10], 'ten')
265 # check keys() and delitem
266 self.assertEqual(s.keys(), [10, 30])
Raymond Hettinger903bf902002-11-15 08:39:40 +0000267
Walter Dörwalde28be592003-01-19 23:26:59 +0000268 ## Now, test the DictMixin methods one by one
Raymond Hettinger903bf902002-11-15 08:39:40 +0000269
Walter Dörwalde28be592003-01-19 23:26:59 +0000270 # __contains__
271 self.assert_(10 in s)
272 self.assert_(20 not in s)
Raymond Hettinger903bf902002-11-15 08:39:40 +0000273
Walter Dörwalde28be592003-01-19 23:26:59 +0000274 # __iter__
275 self.assertEqual([k for k in s], [10, 30])
Raymond Hettinger903bf902002-11-15 08:39:40 +0000276
Walter Dörwalde28be592003-01-19 23:26:59 +0000277 # __len__
278 self.assertEqual(len(s), 2)
Raymond Hettinger903bf902002-11-15 08:39:40 +0000279
Walter Dörwalde28be592003-01-19 23:26:59 +0000280 # iteritems
Guido van Rossumcc2b0162007-02-11 06:12:03 +0000281 self.assertEqual(list(s.items()), [(10,'ten'), (30, 'thirty')])
Raymond Hettinger903bf902002-11-15 08:39:40 +0000282
Walter Dörwalde28be592003-01-19 23:26:59 +0000283 # iterkeys
Guido van Rossumcc2b0162007-02-11 06:12:03 +0000284 self.assertEqual(list(s.keys()), [10, 30])
Raymond Hettinger903bf902002-11-15 08:39:40 +0000285
Walter Dörwalde28be592003-01-19 23:26:59 +0000286 # itervalues
Guido van Rossumcc2b0162007-02-11 06:12:03 +0000287 self.assertEqual(list(s.values()), ['ten', 'thirty'])
Raymond Hettinger903bf902002-11-15 08:39:40 +0000288
Walter Dörwalde28be592003-01-19 23:26:59 +0000289 # values
290 self.assertEqual(s.values(), ['ten', 'thirty'])
Raymond Hettinger903bf902002-11-15 08:39:40 +0000291
Walter Dörwalde28be592003-01-19 23:26:59 +0000292 # items
293 self.assertEqual(s.items(), [(10,'ten'), (30, 'thirty')])
Raymond Hettinger903bf902002-11-15 08:39:40 +0000294
Walter Dörwalde28be592003-01-19 23:26:59 +0000295 # get
296 self.assertEqual(s.get(10), 'ten')
297 self.assertEqual(s.get(15,'fifteen'), 'fifteen')
298 self.assertEqual(s.get(15), None)
Raymond Hettinger903bf902002-11-15 08:39:40 +0000299
Walter Dörwalde28be592003-01-19 23:26:59 +0000300 # setdefault
301 self.assertEqual(s.setdefault(40, 'forty'), 'forty')
302 self.assertEqual(s.setdefault(10, 'null'), 'ten')
303 del s[40]
Raymond Hettinger903bf902002-11-15 08:39:40 +0000304
Walter Dörwalde28be592003-01-19 23:26:59 +0000305 # pop
306 self.assertEqual(s.pop(10), 'ten')
307 self.assert_(10 not in s)
308 s[10] = 'ten'
Raymond Hettingera3e1e4c2003-03-06 23:54:28 +0000309 self.assertEqual(s.pop("x", 1), 1)
310 s["x"] = 42
311 self.assertEqual(s.pop("x", 1), 42)
Raymond Hettinger903bf902002-11-15 08:39:40 +0000312
Walter Dörwalde28be592003-01-19 23:26:59 +0000313 # popitem
314 k, v = s.popitem()
315 self.assert_(k not in s)
316 s[k] = v
Raymond Hettinger903bf902002-11-15 08:39:40 +0000317
Walter Dörwalde28be592003-01-19 23:26:59 +0000318 # clear
319 s.clear()
320 self.assertEqual(len(s), 0)
321
322 # empty popitem
323 self.assertRaises(KeyError, s.popitem)
324
325 # update
326 s.update({10: 'ten', 20:'twenty'})
327 self.assertEqual(s[10], 'ten')
328 self.assertEqual(s[20], 'twenty')
329
330 # cmp
331 self.assertEqual(s, {10: 'ten', 20:'twenty'})
332 t = SeqDict()
333 t[20] = 'twenty'
334 t[10] = 'ten'
335 self.assertEqual(s, t)
336
337def test_main():
Walter Dörwald0a6d0ff2004-05-31 16:29:04 +0000338 test_support.run_unittest(
Walter Dörwald21d3a322003-05-01 17:45:56 +0000339 UserDictTest,
340 UserDictMixinTest
341 )
Walter Dörwalde28be592003-01-19 23:26:59 +0000342
343if __name__ == "__main__":
344 test_main()