blob: 91508d8201e68c0876cab6c82262687d89339de0 [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
3class UserDict:
Raymond Hettinger54405452002-11-22 00:07:40 +00004 def __init__(self, dict=None, **kwargs):
Raymond Hettingere4827eb2002-11-27 08:29:11 +00005 self.data = {}
Raymond Hettinger54405452002-11-22 00:07:40 +00006 if dict is not None:
Raymond Hettingere4827eb2002-11-27 08:29:11 +00007 self.update(dict)
8 if len(kwargs):
9 self.update(kwargs)
Guido van Rossumb94cd961997-06-03 14:10:01 +000010 def __repr__(self): return repr(self.data)
Guido van Rossum47b9ff62006-08-24 00:41:19 +000011 def __eq__(self, dict):
Guido van Rossum1697b9c1999-03-26 15:31:12 +000012 if isinstance(dict, UserDict):
Guido van Rossum47b9ff62006-08-24 00:41:19 +000013 return self.data == dict.data
Guido van Rossum1697b9c1999-03-26 15:31:12 +000014 else:
Guido van Rossum47b9ff62006-08-24 00:41:19 +000015 return self.data == dict
16 def __ne__(self, dict):
17 if isinstance(dict, UserDict):
18 return self.data != dict.data
19 else:
20 return self.data != dict
Guido van Rossumb94cd961997-06-03 14:10:01 +000021 def __len__(self): return len(self.data)
Guido van Rossum1968ad32006-02-25 22:38:04 +000022 def __getitem__(self, key):
23 if key in self.data:
24 return self.data[key]
25 if hasattr(self.__class__, "__missing__"):
26 return self.__class__.__missing__(self, key)
27 raise KeyError(key)
Guido van Rossumb94cd961997-06-03 14:10:01 +000028 def __setitem__(self, key, item): self.data[key] = item
29 def __delitem__(self, key): del self.data[key]
Guido van Rossum1697b9c1999-03-26 15:31:12 +000030 def clear(self): self.data.clear()
Guido van Rossumb94cd961997-06-03 14:10:01 +000031 def copy(self):
Guido van Rossum1697b9c1999-03-26 15:31:12 +000032 if self.__class__ is UserDict:
Raymond Hettinger90145602003-12-21 22:19:08 +000033 return UserDict(self.data.copy())
Guido van Rossum45e2fbc1998-03-26 21:13:24 +000034 import copy
Fred Drake3ce5af72001-11-05 17:40:48 +000035 data = self.data
36 try:
37 self.data = {}
38 c = copy.copy(self)
39 finally:
40 self.data = data
41 c.update(self)
42 return c
Guido van Rossumb94cd961997-06-03 14:10:01 +000043 def keys(self): return self.data.keys()
44 def items(self): return self.data.items()
45 def values(self): return self.data.values()
Raymond Hettinger31017ae2004-03-04 08:25:44 +000046 def update(self, dict=None, **kwargs):
47 if dict is None:
48 pass
49 elif isinstance(dict, UserDict):
Guido van Rossum1697b9c1999-03-26 15:31:12 +000050 self.data.update(dict.data)
Raymond Hettinger31017ae2004-03-04 08:25:44 +000051 elif isinstance(dict, type({})) or not hasattr(dict, 'items'):
Guido van Rossum1697b9c1999-03-26 15:31:12 +000052 self.data.update(dict)
Guido van Rossum45e2fbc1998-03-26 21:13:24 +000053 else:
Guido van Rossum1697b9c1999-03-26 15:31:12 +000054 for k, v in dict.items():
Martin v. Löwisf3b30742001-06-18 01:09:41 +000055 self[k] = v
Raymond Hettinger31017ae2004-03-04 08:25:44 +000056 if len(kwargs):
57 self.data.update(kwargs)
Barry Warsawfc3e61c1997-10-06 17:50:04 +000058 def get(self, key, failobj=None):
Guido van Rossume2b70bc2006-08-18 22:13:04 +000059 if key not in self:
Martin v. Löwisf3b30742001-06-18 01:09:41 +000060 return failobj
61 return self[key]
Guido van Rossum164452c2000-08-08 16:12:54 +000062 def setdefault(self, key, failobj=None):
Guido van Rossume2b70bc2006-08-18 22:13:04 +000063 if key not in self:
Martin v. Löwisf3b30742001-06-18 01:09:41 +000064 self[key] = failobj
65 return self[key]
Raymond Hettingera3e1e4c2003-03-06 23:54:28 +000066 def pop(self, key, *args):
67 return self.data.pop(key, *args)
Guido van Rossum1072c382000-12-12 22:06:00 +000068 def popitem(self):
69 return self.data.popitem()
Tim Petersa3f98d62001-04-21 09:13:15 +000070 def __contains__(self, key):
71 return key in self.data
Guido van Rossum75b64e62005-01-16 00:16:11 +000072 @classmethod
Raymond Hettingere4827eb2002-11-27 08:29:11 +000073 def fromkeys(cls, iterable, value=None):
74 d = cls()
75 for key in iterable:
76 d[key] = value
77 return d
Guido van Rossum2050b652001-08-07 17:40:42 +000078
79class IterableUserDict(UserDict):
Tim Petersa3f98d62001-04-21 09:13:15 +000080 def __iter__(self):
81 return iter(self.data)
Raymond Hettinger79947162002-11-15 06:46:14 +000082
83class DictMixin:
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +000084 # Mixin defining all dictionary methods for classes that already have
85 # a minimum dictionary interface including getitem, setitem, delitem,
86 # and keys. Without knowledge of the subclass constructor, the mixin
87 # does not define __init__() or copy(). In addition to the four base
Barry Warsaw1a177042003-01-31 03:30:09 +000088 # methods, progressively more efficiency comes with defining
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +000089 # __contains__(), __iter__(), and iteritems().
Raymond Hettinger79947162002-11-15 06:46:14 +000090
Guido van Rossumd81206d2007-02-15 03:49:08 +000091 # XXX It would make more sense to expect __iter__ to be primitive.
92
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +000093 # second level definitions support higher levels
94 def __iter__(self):
95 for k in self.keys():
96 yield k
Guido van Rossume2b70bc2006-08-18 22:13:04 +000097 def __contains__(self, key):
Raymond Hettinger79947162002-11-15 06:46:14 +000098 try:
99 value = self[key]
100 except KeyError:
101 return False
102 return True
Raymond Hettinger79947162002-11-15 06:46:14 +0000103
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000104 # third level takes advantage of second level definitions
Guido van Rossumd81206d2007-02-15 03:49:08 +0000105 def iterkeys(self):
106 return self.__iter__()
Raymond Hettinger79947162002-11-15 06:46:14 +0000107 def iteritems(self):
108 for k in self:
109 yield (k, self[k])
Raymond Hettinger79947162002-11-15 06:46:14 +0000110
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000111 # fourth level uses definitions from lower levels
Raymond Hettinger79947162002-11-15 06:46:14 +0000112 def itervalues(self):
Guido van Rossumd81206d2007-02-15 03:49:08 +0000113 for _, v in self.iteritems():
Raymond Hettinger79947162002-11-15 06:46:14 +0000114 yield v
115 def values(self):
Guido van Rossumd81206d2007-02-15 03:49:08 +0000116 return [v for _, v in self.iteritems()]
Raymond Hettinger79947162002-11-15 06:46:14 +0000117 def items(self):
Guido van Rossumd81206d2007-02-15 03:49:08 +0000118 return list(self.iteritems())
Raymond Hettinger79947162002-11-15 06:46:14 +0000119 def clear(self):
Guido van Rossum75d26cc2007-02-15 04:01:01 +0000120 for key in list(self.iterkeys()):
Raymond Hettinger79947162002-11-15 06:46:14 +0000121 del self[key]
Walter Dörwaldc1399092004-05-27 09:41:04 +0000122 def setdefault(self, key, default=None):
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000123 try:
124 return self[key]
125 except KeyError:
Raymond Hettinger79947162002-11-15 06:46:14 +0000126 self[key] = default
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000127 return default
Raymond Hettingera3e1e4c2003-03-06 23:54:28 +0000128 def pop(self, key, *args):
129 if len(args) > 1:
130 raise TypeError, "pop expected at most 2 arguments, got "\
131 + repr(1 + len(args))
132 try:
133 value = self[key]
134 except KeyError:
135 if args:
136 return args[0]
137 raise
Raymond Hettinger79947162002-11-15 06:46:14 +0000138 del self[key]
139 return value
140 def popitem(self):
141 try:
Guido van Rossumd81206d2007-02-15 03:49:08 +0000142 k, v = self.iteritems().next()
Raymond Hettinger79947162002-11-15 06:46:14 +0000143 except StopIteration:
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000144 raise KeyError, 'container is empty'
Raymond Hettinger79947162002-11-15 06:46:14 +0000145 del self[k]
146 return (k, v)
Raymond Hettinger31017ae2004-03-04 08:25:44 +0000147 def update(self, other=None, **kwargs):
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000148 # Make progressively weaker assumptions about "other"
Raymond Hettinger31017ae2004-03-04 08:25:44 +0000149 if other is None:
150 pass
151 elif hasattr(other, 'iteritems'): # iteritems saves memory and lookups
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000152 for k, v in other.iteritems():
153 self[k] = v
Guido van Rossumcc2b0162007-02-11 06:12:03 +0000154 elif hasattr(other, 'items'): # items may also save memory and lookups
155 for k, v in other.items():
156 self[k] = v
Raymond Hettinger31017ae2004-03-04 08:25:44 +0000157 elif hasattr(other, 'keys'):
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000158 for k in other.keys():
159 self[k] = other[k]
Raymond Hettinger31017ae2004-03-04 08:25:44 +0000160 else:
161 for k, v in other:
162 self[k] = v
163 if kwargs:
164 self.update(kwargs)
Raymond Hettinger79947162002-11-15 06:46:14 +0000165 def get(self, key, default=None):
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000166 try:
Raymond Hettinger79947162002-11-15 06:46:14 +0000167 return self[key]
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000168 except KeyError:
169 return default
Raymond Hettinger79947162002-11-15 06:46:14 +0000170 def __repr__(self):
Guido van Rossumd81206d2007-02-15 03:49:08 +0000171 return repr(dict(self.iteritems()))
Guido van Rossum47b9ff62006-08-24 00:41:19 +0000172 def __eq__(self, other):
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000173 if isinstance(other, DictMixin):
Guido van Rossumd81206d2007-02-15 03:49:08 +0000174 other = dict(other.iteritems())
175 return dict(self.iteritems()) == other
Guido van Rossum47b9ff62006-08-24 00:41:19 +0000176 def __ne__(self, other):
177 if isinstance(other, DictMixin):
Guido van Rossumd81206d2007-02-15 03:49:08 +0000178 other = dict(other.iteritems())
179 return dict(self.iteritems()) != other
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000180 def __len__(self):
181 return len(self.keys())