blob: 732b327c3ea4cfba489987f65238cd0ef29ab961 [file] [log] [blame]
Guido van Rossum4acc25b2000-02-02 15:10:15 +00001"""A more or less complete user-defined wrapper around dictionary objects."""
Guido van Rossumae3b3a31993-11-30 13:43:54 +00002
Benjamin Petersonbde67df2010-06-22 18:09:02 +00003class UserDict:
Serhiy Storchakacab45662015-09-29 23:33:03 +03004 def __init__(*args, **kwargs):
5 if not args:
6 raise TypeError("descriptor '__init__' of 'UserDict' object "
7 "needs an argument")
8 self = args[0]
9 args = args[1:]
10 if len(args) > 1:
11 raise TypeError('expected at most 1 arguments, got %d' % len(args))
12 if args:
13 dict = args[0]
14 elif 'dict' in kwargs:
15 dict = kwargs.pop('dict')
16 import warnings
17 warnings.warn("Passing 'dict' as keyword argument is "
18 "deprecated", PendingDeprecationWarning,
19 stacklevel=2)
20 else:
21 dict = None
Raymond Hettingere4827eb2002-11-27 08:29:11 +000022 self.data = {}
Raymond Hettinger54405452002-11-22 00:07:40 +000023 if dict is not None:
Raymond Hettingere4827eb2002-11-27 08:29:11 +000024 self.update(dict)
25 if len(kwargs):
26 self.update(kwargs)
Guido van Rossumb94cd961997-06-03 14:10:01 +000027 def __repr__(self): return repr(self.data)
28 def __cmp__(self, dict):
Guido van Rossum1697b9c1999-03-26 15:31:12 +000029 if isinstance(dict, UserDict):
Guido van Rossum45e2fbc1998-03-26 21:13:24 +000030 return cmp(self.data, dict.data)
Guido van Rossum1697b9c1999-03-26 15:31:12 +000031 else:
32 return cmp(self.data, dict)
Benjamin Peterson11fab6c2010-01-28 02:18:25 +000033 __hash__ = None # Avoid Py3k warning
Guido van Rossumb94cd961997-06-03 14:10:01 +000034 def __len__(self): return len(self.data)
Guido van Rossum1968ad32006-02-25 22:38:04 +000035 def __getitem__(self, key):
36 if key in self.data:
37 return self.data[key]
38 if hasattr(self.__class__, "__missing__"):
39 return self.__class__.__missing__(self, key)
40 raise KeyError(key)
Guido van Rossumb94cd961997-06-03 14:10:01 +000041 def __setitem__(self, key, item): self.data[key] = item
42 def __delitem__(self, key): del self.data[key]
Guido van Rossum1697b9c1999-03-26 15:31:12 +000043 def clear(self): self.data.clear()
Guido van Rossumb94cd961997-06-03 14:10:01 +000044 def copy(self):
Guido van Rossum1697b9c1999-03-26 15:31:12 +000045 if self.__class__ is UserDict:
Raymond Hettinger90145602003-12-21 22:19:08 +000046 return UserDict(self.data.copy())
Guido van Rossum45e2fbc1998-03-26 21:13:24 +000047 import copy
Fred Drake3ce5af72001-11-05 17:40:48 +000048 data = self.data
49 try:
50 self.data = {}
51 c = copy.copy(self)
52 finally:
53 self.data = data
54 c.update(self)
55 return c
Guido van Rossumb94cd961997-06-03 14:10:01 +000056 def keys(self): return self.data.keys()
57 def items(self): return self.data.items()
Fred Drakebedebbd2001-05-03 04:54:41 +000058 def iteritems(self): return self.data.iteritems()
59 def iterkeys(self): return self.data.iterkeys()
60 def itervalues(self): return self.data.itervalues()
Guido van Rossumb94cd961997-06-03 14:10:01 +000061 def values(self): return self.data.values()
Raymond Hettinger31ce5cb2008-02-21 19:24:53 +000062 def has_key(self, key): return key in self.data
Serhiy Storchakacab45662015-09-29 23:33:03 +030063 def update(*args, **kwargs):
64 if not args:
65 raise TypeError("descriptor 'update' of 'UserDict' object "
66 "needs an argument")
67 self = args[0]
68 args = args[1:]
69 if len(args) > 1:
70 raise TypeError('expected at most 1 arguments, got %d' % len(args))
71 if args:
72 dict = args[0]
73 elif 'dict' in kwargs:
74 dict = kwargs.pop('dict')
75 import warnings
76 warnings.warn("Passing 'dict' as keyword argument is deprecated",
77 PendingDeprecationWarning, stacklevel=2)
78 else:
79 dict = None
Raymond Hettinger31017ae2004-03-04 08:25:44 +000080 if dict is None:
81 pass
82 elif isinstance(dict, UserDict):
Guido van Rossum1697b9c1999-03-26 15:31:12 +000083 self.data.update(dict.data)
Raymond Hettinger31017ae2004-03-04 08:25:44 +000084 elif isinstance(dict, type({})) or not hasattr(dict, 'items'):
Guido van Rossum1697b9c1999-03-26 15:31:12 +000085 self.data.update(dict)
Guido van Rossum45e2fbc1998-03-26 21:13:24 +000086 else:
Guido van Rossum1697b9c1999-03-26 15:31:12 +000087 for k, v in dict.items():
Martin v. Löwisf3b30742001-06-18 01:09:41 +000088 self[k] = v
Raymond Hettinger31017ae2004-03-04 08:25:44 +000089 if len(kwargs):
90 self.data.update(kwargs)
Barry Warsawfc3e61c1997-10-06 17:50:04 +000091 def get(self, key, failobj=None):
Guido van Rossum8bc09652008-02-21 18:18:37 +000092 if key not in self:
Martin v. Löwisf3b30742001-06-18 01:09:41 +000093 return failobj
94 return self[key]
Guido van Rossum164452c2000-08-08 16:12:54 +000095 def setdefault(self, key, failobj=None):
Raymond Hettinger31ce5cb2008-02-21 19:24:53 +000096 if key not in self:
Martin v. Löwisf3b30742001-06-18 01:09:41 +000097 self[key] = failobj
98 return self[key]
Raymond Hettingera3e1e4c2003-03-06 23:54:28 +000099 def pop(self, key, *args):
100 return self.data.pop(key, *args)
Guido van Rossum1072c382000-12-12 22:06:00 +0000101 def popitem(self):
102 return self.data.popitem()
Tim Petersa3f98d62001-04-21 09:13:15 +0000103 def __contains__(self, key):
104 return key in self.data
Guido van Rossum75b64e62005-01-16 00:16:11 +0000105 @classmethod
Raymond Hettingere4827eb2002-11-27 08:29:11 +0000106 def fromkeys(cls, iterable, value=None):
107 d = cls()
108 for key in iterable:
109 d[key] = value
110 return d
Guido van Rossum2050b652001-08-07 17:40:42 +0000111
112class IterableUserDict(UserDict):
Tim Petersa3f98d62001-04-21 09:13:15 +0000113 def __iter__(self):
114 return iter(self.data)
Raymond Hettinger79947162002-11-15 06:46:14 +0000115
Georg Brandl3bed4ae2008-06-07 16:16:12 +0000116import _abcoll
117_abcoll.MutableMapping.register(IterableUserDict)
118
119
Raymond Hettinger79947162002-11-15 06:46:14 +0000120class DictMixin:
Raymond Hettinger8ddc1762002-11-18 04:34:10 +0000121 # Mixin defining all dictionary methods for classes that already have
122 # a minimum dictionary interface including getitem, setitem, delitem,
123 # and keys. Without knowledge of the subclass constructor, the mixin
124 # does not define __init__() or copy(). In addition to the four base
Barry Warsaw1a177042003-01-31 03:30:09 +0000125 # methods, progressively more efficiency comes with defining
Raymond Hettinger8ddc1762002-11-18 04:34:10 +0000126 # __contains__(), __iter__(), and iteritems().
Raymond Hettinger79947162002-11-15 06:46:14 +0000127
Raymond Hettinger8ddc1762002-11-18 04:34:10 +0000128 # second level definitions support higher levels
129 def __iter__(self):
130 for k in self.keys():
131 yield k
Raymond Hettinger79947162002-11-15 06:46:14 +0000132 def has_key(self, key):
133 try:
Georg Brandl84fedf72010-02-06 22:59:15 +0000134 self[key]
Raymond Hettinger79947162002-11-15 06:46:14 +0000135 except KeyError:
136 return False
137 return True
Raymond Hettinger51f3f1b2003-01-22 01:39:06 +0000138 def __contains__(self, key):
139 return self.has_key(key)
Raymond Hettinger79947162002-11-15 06:46:14 +0000140
Raymond Hettinger8ddc1762002-11-18 04:34:10 +0000141 # third level takes advantage of second level definitions
Raymond Hettinger79947162002-11-15 06:46:14 +0000142 def iteritems(self):
143 for k in self:
144 yield (k, self[k])
Raymond Hettinger51f3f1b2003-01-22 01:39:06 +0000145 def iterkeys(self):
146 return self.__iter__()
Raymond Hettinger79947162002-11-15 06:46:14 +0000147
Raymond Hettinger8ddc1762002-11-18 04:34:10 +0000148 # fourth level uses definitions from lower levels
Raymond Hettinger79947162002-11-15 06:46:14 +0000149 def itervalues(self):
150 for _, v in self.iteritems():
151 yield v
152 def values(self):
Raymond Hettinger8ddc1762002-11-18 04:34:10 +0000153 return [v for _, v in self.iteritems()]
Raymond Hettinger79947162002-11-15 06:46:14 +0000154 def items(self):
155 return list(self.iteritems())
156 def clear(self):
157 for key in self.keys():
158 del self[key]
Walter Dörwaldc1399092004-05-27 09:41:04 +0000159 def setdefault(self, key, default=None):
Raymond Hettinger8ddc1762002-11-18 04:34:10 +0000160 try:
161 return self[key]
162 except KeyError:
Raymond Hettinger79947162002-11-15 06:46:14 +0000163 self[key] = default
Raymond Hettinger8ddc1762002-11-18 04:34:10 +0000164 return default
Raymond Hettingera3e1e4c2003-03-06 23:54:28 +0000165 def pop(self, key, *args):
166 if len(args) > 1:
167 raise TypeError, "pop expected at most 2 arguments, got "\
168 + repr(1 + len(args))
169 try:
170 value = self[key]
171 except KeyError:
172 if args:
173 return args[0]
174 raise
Raymond Hettinger79947162002-11-15 06:46:14 +0000175 del self[key]
176 return value
177 def popitem(self):
178 try:
179 k, v = self.iteritems().next()
180 except StopIteration:
Raymond Hettinger8ddc1762002-11-18 04:34:10 +0000181 raise KeyError, 'container is empty'
Raymond Hettinger79947162002-11-15 06:46:14 +0000182 del self[k]
183 return (k, v)
Raymond Hettinger31017ae2004-03-04 08:25:44 +0000184 def update(self, other=None, **kwargs):
Raymond Hettinger8ddc1762002-11-18 04:34:10 +0000185 # Make progressively weaker assumptions about "other"
Raymond Hettinger31017ae2004-03-04 08:25:44 +0000186 if other is None:
187 pass
188 elif hasattr(other, 'iteritems'): # iteritems saves memory and lookups
Raymond Hettinger8ddc1762002-11-18 04:34:10 +0000189 for k, v in other.iteritems():
190 self[k] = v
Raymond Hettinger31017ae2004-03-04 08:25:44 +0000191 elif hasattr(other, 'keys'):
Raymond Hettinger8ddc1762002-11-18 04:34:10 +0000192 for k in other.keys():
193 self[k] = other[k]
Raymond Hettinger31017ae2004-03-04 08:25:44 +0000194 else:
195 for k, v in other:
196 self[k] = v
197 if kwargs:
198 self.update(kwargs)
Raymond Hettinger79947162002-11-15 06:46:14 +0000199 def get(self, key, default=None):
Raymond Hettinger8ddc1762002-11-18 04:34:10 +0000200 try:
Raymond Hettinger79947162002-11-15 06:46:14 +0000201 return self[key]
Raymond Hettinger8ddc1762002-11-18 04:34:10 +0000202 except KeyError:
203 return default
Raymond Hettinger79947162002-11-15 06:46:14 +0000204 def __repr__(self):
Raymond Hettinger8ddc1762002-11-18 04:34:10 +0000205 return repr(dict(self.iteritems()))
206 def __cmp__(self, other):
Raymond Hettingerf477c882003-12-04 20:04:09 +0000207 if other is None:
208 return 1
Raymond Hettinger8ddc1762002-11-18 04:34:10 +0000209 if isinstance(other, DictMixin):
210 other = dict(other.iteritems())
211 return cmp(dict(self.iteritems()), other)
212 def __len__(self):
213 return len(self.keys())