| 
									
										
										
										
											2000-02-02 15:10:15 +00:00
										 |  |  | """A more or less complete user-defined wrapper around dictionary objects.""" | 
					
						
							| 
									
										
										
										
											1993-11-30 13:43:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class UserDict: | 
					
						
							| 
									
										
										
										
											2002-11-22 00:07:40 +00:00
										 |  |  |     def __init__(self, dict=None, **kwargs): | 
					
						
							| 
									
										
										
										
											2002-11-27 08:29:11 +00:00
										 |  |  |         self.data = {} | 
					
						
							| 
									
										
										
										
											2002-11-22 00:07:40 +00:00
										 |  |  |         if dict is not None: | 
					
						
							| 
									
										
										
										
											2002-11-27 08:29:11 +00:00
										 |  |  |             if not hasattr(dict,'keys'): | 
					
						
							|  |  |  |                 dict = type({})(dict)   # make mapping from a sequence | 
					
						
							|  |  |  |             self.update(dict) | 
					
						
							|  |  |  |         if len(kwargs): | 
					
						
							|  |  |  |             self.update(kwargs) | 
					
						
							| 
									
										
										
										
											1997-06-03 14:10:01 +00:00
										 |  |  |     def __repr__(self): return repr(self.data) | 
					
						
							|  |  |  |     def __cmp__(self, dict): | 
					
						
							| 
									
										
										
										
											1999-03-26 15:31:12 +00:00
										 |  |  |         if isinstance(dict, UserDict): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             return cmp(self.data, dict.data) | 
					
						
							| 
									
										
										
										
											1999-03-26 15:31:12 +00:00
										 |  |  |         else: | 
					
						
							|  |  |  |             return cmp(self.data, dict) | 
					
						
							| 
									
										
										
										
											1997-06-03 14:10:01 +00:00
										 |  |  |     def __len__(self): return len(self.data) | 
					
						
							|  |  |  |     def __getitem__(self, key): return self.data[key] | 
					
						
							|  |  |  |     def __setitem__(self, key, item): self.data[key] = item | 
					
						
							|  |  |  |     def __delitem__(self, key): del self.data[key] | 
					
						
							| 
									
										
										
										
											1999-03-26 15:31:12 +00:00
										 |  |  |     def clear(self): self.data.clear() | 
					
						
							| 
									
										
										
										
											1997-06-03 14:10:01 +00:00
										 |  |  |     def copy(self): | 
					
						
							| 
									
										
										
										
											1999-03-26 15:31:12 +00:00
										 |  |  |         if self.__class__ is UserDict: | 
					
						
							|  |  |  |             return UserDict(self.data) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         import copy | 
					
						
							| 
									
										
										
										
											2001-11-05 17:40:48 +00:00
										 |  |  |         data = self.data | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             self.data = {} | 
					
						
							|  |  |  |             c = copy.copy(self) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             self.data = data | 
					
						
							|  |  |  |         c.update(self) | 
					
						
							|  |  |  |         return c | 
					
						
							| 
									
										
										
										
											1997-06-03 14:10:01 +00:00
										 |  |  |     def keys(self): return self.data.keys() | 
					
						
							|  |  |  |     def items(self): return self.data.items() | 
					
						
							| 
									
										
										
										
											2001-05-03 04:54:41 +00:00
										 |  |  |     def iteritems(self): return self.data.iteritems() | 
					
						
							|  |  |  |     def iterkeys(self): return self.data.iterkeys() | 
					
						
							|  |  |  |     def itervalues(self): return self.data.itervalues() | 
					
						
							| 
									
										
										
										
											1997-06-03 14:10:01 +00:00
										 |  |  |     def values(self): return self.data.values() | 
					
						
							|  |  |  |     def has_key(self, key): return self.data.has_key(key) | 
					
						
							| 
									
										
										
										
											1999-03-26 15:31:12 +00:00
										 |  |  |     def update(self, dict): | 
					
						
							|  |  |  |         if isinstance(dict, UserDict): | 
					
						
							|  |  |  |             self.data.update(dict.data) | 
					
						
							|  |  |  |         elif isinstance(dict, type(self.data)): | 
					
						
							|  |  |  |             self.data.update(dict) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											1999-03-26 15:31:12 +00:00
										 |  |  |             for k, v in dict.items(): | 
					
						
							| 
									
										
										
										
											2001-06-18 01:09:41 +00:00
										 |  |  |                 self[k] = v | 
					
						
							| 
									
										
										
										
											1997-10-06 17:50:04 +00:00
										 |  |  |     def get(self, key, failobj=None): | 
					
						
							| 
									
										
										
										
											2001-06-18 01:09:41 +00:00
										 |  |  |         if not self.has_key(key): | 
					
						
							|  |  |  |             return failobj | 
					
						
							|  |  |  |         return self[key] | 
					
						
							| 
									
										
										
										
											2000-08-08 16:12:54 +00:00
										 |  |  |     def setdefault(self, key, failobj=None): | 
					
						
							| 
									
										
										
										
											2001-06-18 01:09:41 +00:00
										 |  |  |         if not self.has_key(key): | 
					
						
							|  |  |  |             self[key] = failobj | 
					
						
							|  |  |  |         return self[key] | 
					
						
							| 
									
										
										
										
											2002-04-13 14:03:38 +00:00
										 |  |  |     def pop(self, key): | 
					
						
							|  |  |  |         return self.data.pop(key) | 
					
						
							| 
									
										
										
										
											2000-12-12 22:06:00 +00:00
										 |  |  |     def popitem(self): | 
					
						
							|  |  |  |         return self.data.popitem() | 
					
						
							| 
									
										
										
										
											2001-04-21 09:13:15 +00:00
										 |  |  |     def __contains__(self, key): | 
					
						
							|  |  |  |         return key in self.data | 
					
						
							| 
									
										
										
										
											2002-11-27 08:29:11 +00:00
										 |  |  |     def fromkeys(cls, iterable, value=None): | 
					
						
							|  |  |  |         d = cls() | 
					
						
							|  |  |  |         for key in iterable: | 
					
						
							|  |  |  |             d[key] = value | 
					
						
							|  |  |  |         return d | 
					
						
							|  |  |  |     fromkeys = classmethod(fromkeys) | 
					
						
							| 
									
										
										
										
											2001-08-07 17:40:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class IterableUserDict(UserDict): | 
					
						
							| 
									
										
										
										
											2001-04-21 09:13:15 +00:00
										 |  |  |     def __iter__(self): | 
					
						
							|  |  |  |         return iter(self.data) | 
					
						
							| 
									
										
										
										
											2002-11-15 06:46:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class DictMixin: | 
					
						
							| 
									
										
										
										
											2002-11-18 04:34:10 +00:00
										 |  |  |     # Mixin defining all dictionary methods for classes that already have | 
					
						
							|  |  |  |     # a minimum dictionary interface including getitem, setitem, delitem, | 
					
						
							|  |  |  |     # and keys. Without knowledge of the subclass constructor, the mixin | 
					
						
							|  |  |  |     # does not define __init__() or copy().  In addition to the four base | 
					
						
							|  |  |  |     # methods, progessively more efficiency comes with defining | 
					
						
							|  |  |  |     # __contains__(), __iter__(), and iteritems(). | 
					
						
							| 
									
										
										
										
											2002-11-15 06:46:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-18 04:34:10 +00:00
										 |  |  |     # second level definitions support higher levels | 
					
						
							|  |  |  |     def __iter__(self): | 
					
						
							|  |  |  |         for k in self.keys(): | 
					
						
							|  |  |  |             yield k | 
					
						
							| 
									
										
										
										
											2002-11-15 06:46:14 +00:00
										 |  |  |     def has_key(self, key): | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             value = self[key] | 
					
						
							|  |  |  |         except KeyError: | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  |         return True | 
					
						
							| 
									
										
										
										
											2003-01-22 01:39:06 +00:00
										 |  |  |     def __contains__(self, key): | 
					
						
							|  |  |  |         return self.has_key(key) | 
					
						
							| 
									
										
										
										
											2002-11-15 06:46:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-18 04:34:10 +00:00
										 |  |  |     # third level takes advantage of second level definitions | 
					
						
							| 
									
										
										
										
											2002-11-15 06:46:14 +00:00
										 |  |  |     def iteritems(self): | 
					
						
							|  |  |  |         for k in self: | 
					
						
							|  |  |  |             yield (k, self[k]) | 
					
						
							| 
									
										
										
										
											2003-01-22 01:39:06 +00:00
										 |  |  |     def iterkeys(self): | 
					
						
							|  |  |  |         return self.__iter__() | 
					
						
							| 
									
										
										
										
											2002-11-15 06:46:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-18 04:34:10 +00:00
										 |  |  |     # fourth level uses definitions from lower levels | 
					
						
							| 
									
										
										
										
											2002-11-15 06:46:14 +00:00
										 |  |  |     def itervalues(self): | 
					
						
							|  |  |  |         for _, v in self.iteritems(): | 
					
						
							|  |  |  |             yield v | 
					
						
							|  |  |  |     def values(self): | 
					
						
							| 
									
										
										
										
											2002-11-18 04:34:10 +00:00
										 |  |  |         return [v for _, v in self.iteritems()] | 
					
						
							| 
									
										
										
										
											2002-11-15 06:46:14 +00:00
										 |  |  |     def items(self): | 
					
						
							|  |  |  |         return list(self.iteritems()) | 
					
						
							|  |  |  |     def clear(self): | 
					
						
							|  |  |  |         for key in self.keys(): | 
					
						
							|  |  |  |             del self[key] | 
					
						
							|  |  |  |     def setdefault(self, key, default): | 
					
						
							| 
									
										
										
										
											2002-11-18 04:34:10 +00:00
										 |  |  |         try: | 
					
						
							|  |  |  |             return self[key] | 
					
						
							|  |  |  |         except KeyError: | 
					
						
							| 
									
										
										
										
											2002-11-15 06:46:14 +00:00
										 |  |  |             self[key] = default | 
					
						
							| 
									
										
										
										
											2002-11-18 04:34:10 +00:00
										 |  |  |         return default | 
					
						
							| 
									
										
										
										
											2002-11-15 06:46:14 +00:00
										 |  |  |     def pop(self, key): | 
					
						
							|  |  |  |         value = self[key] | 
					
						
							|  |  |  |         del self[key] | 
					
						
							|  |  |  |         return value | 
					
						
							|  |  |  |     def popitem(self): | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             k, v = self.iteritems().next() | 
					
						
							|  |  |  |         except StopIteration: | 
					
						
							| 
									
										
										
										
											2002-11-18 04:34:10 +00:00
										 |  |  |             raise KeyError, 'container is empty' | 
					
						
							| 
									
										
										
										
											2002-11-15 06:46:14 +00:00
										 |  |  |         del self[k] | 
					
						
							|  |  |  |         return (k, v) | 
					
						
							|  |  |  |     def update(self, other): | 
					
						
							| 
									
										
										
										
											2002-11-18 04:34:10 +00:00
										 |  |  |         # Make progressively weaker assumptions about "other" | 
					
						
							|  |  |  |         if hasattr(other, 'iteritems'):  # iteritems saves memory and lookups | 
					
						
							|  |  |  |             for k, v in other.iteritems(): | 
					
						
							|  |  |  |                 self[k] = v | 
					
						
							|  |  |  |         elif hasattr(other, '__iter__'): # iter saves memory | 
					
						
							|  |  |  |             for k in other: | 
					
						
							|  |  |  |                 self[k] = other[k] | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             for k in other.keys(): | 
					
						
							|  |  |  |                 self[k] = other[k] | 
					
						
							| 
									
										
										
										
											2002-11-15 06:46:14 +00:00
										 |  |  |     def get(self, key, default=None): | 
					
						
							| 
									
										
										
										
											2002-11-18 04:34:10 +00:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2002-11-15 06:46:14 +00:00
										 |  |  |             return self[key] | 
					
						
							| 
									
										
										
										
											2002-11-18 04:34:10 +00:00
										 |  |  |         except KeyError: | 
					
						
							|  |  |  |             return default | 
					
						
							| 
									
										
										
										
											2002-11-15 06:46:14 +00:00
										 |  |  |     def __repr__(self): | 
					
						
							| 
									
										
										
										
											2002-11-18 04:34:10 +00:00
										 |  |  |         return repr(dict(self.iteritems())) | 
					
						
							|  |  |  |     def __cmp__(self, other): | 
					
						
							|  |  |  |         if isinstance(other, DictMixin): | 
					
						
							|  |  |  |             other = dict(other.iteritems()) | 
					
						
							|  |  |  |         return cmp(dict(self.iteritems()), other) | 
					
						
							|  |  |  |     def __len__(self): | 
					
						
							|  |  |  |         return len(self.keys()) |