| 
									
										
										
										
											2007-03-01 06:16:43 +00:00
										 |  |  | __all__ = ['deque', 'defaultdict', 'NamedTuple'] | 
					
						
							| 
									
										
										
										
											2007-02-28 18:37:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | from _collections import deque, defaultdict | 
					
						
							| 
									
										
										
										
											2007-03-01 06:16:43 +00:00
										 |  |  | from operator import itemgetter as _itemgetter | 
					
						
							|  |  |  | import sys as _sys | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def NamedTuple(typename, s): | 
					
						
							|  |  |  |     """Returns a new subclass of tuple with named fields.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     >>> Point = NamedTuple('Point', 'x y') | 
					
						
							| 
									
										
										
										
											2007-09-17 00:55:00 +00:00
										 |  |  |     >>> Point.__doc__                   # docstring for the new class | 
					
						
							| 
									
										
										
										
											2007-03-01 06:16:43 +00:00
										 |  |  |     'Point(x, y)' | 
					
						
							| 
									
										
										
										
											2007-09-17 00:55:00 +00:00
										 |  |  |     >>> p = Point(11, y=22)             # instantiate with positional args or keywords | 
					
						
							|  |  |  |     >>> p[0] + p[1]                     # works just like the tuple (11, 22) | 
					
						
							| 
									
										
										
										
											2007-03-01 06:16:43 +00:00
										 |  |  |     33 | 
					
						
							| 
									
										
										
										
											2007-09-17 00:55:00 +00:00
										 |  |  |     >>> x, y = p                        # unpacks just like a tuple | 
					
						
							| 
									
										
										
										
											2007-03-01 06:16:43 +00:00
										 |  |  |     >>> x, y | 
					
						
							|  |  |  |     (11, 22) | 
					
						
							| 
									
										
										
										
											2007-09-17 00:55:00 +00:00
										 |  |  |     >>> p.x + p.y                       # fields also accessable by name | 
					
						
							| 
									
										
										
										
											2007-03-01 06:16:43 +00:00
										 |  |  |     33 | 
					
						
							| 
									
										
										
										
											2007-09-17 00:55:00 +00:00
										 |  |  |     >>> p                               # readable __repr__ with name=value style | 
					
						
							| 
									
										
										
										
											2007-03-01 06:16:43 +00:00
										 |  |  |     Point(x=11, y=22) | 
					
						
							| 
									
										
										
										
											2007-09-17 00:55:00 +00:00
										 |  |  |     >>> p.__replace__('x', 100)         # __replace__() is like str.replace() but targets a named field | 
					
						
							|  |  |  |     Point(x=100, y=22) | 
					
						
							|  |  |  |     >>> d = dict(zip(p.__fields__, p))  # use __fields__ to make a dictionary | 
					
						
							|  |  |  |     >>> d['x'] | 
					
						
							|  |  |  |     11 | 
					
						
							| 
									
										
										
										
											2007-03-01 06:16:43 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-09-17 00:55:00 +00:00
										 |  |  |     field_names = tuple(s.replace(',', ' ').split())   # names separated by spaces and/or commas | 
					
						
							|  |  |  |     if not ''.join((typename,) + field_names).replace('_', '').isalnum(): | 
					
						
							| 
									
										
										
										
											2007-05-21 08:13:35 +00:00
										 |  |  |         raise ValueError('Type names and field names can only contain alphanumeric characters and underscores') | 
					
						
							| 
									
										
										
										
											2007-05-19 01:11:16 +00:00
										 |  |  |     argtxt = ', '.join(field_names) | 
					
						
							|  |  |  |     reprtxt = ', '.join('%s=%%r' % name for name in field_names) | 
					
						
							|  |  |  |     template = '''class %(typename)s(tuple):
 | 
					
						
							|  |  |  |         '%(typename)s(%(argtxt)s)' | 
					
						
							|  |  |  |         __slots__ = () | 
					
						
							| 
									
										
										
										
											2007-09-17 00:55:00 +00:00
										 |  |  |         __fields__ = %(field_names)r | 
					
						
							| 
									
										
										
										
											2007-05-19 01:11:16 +00:00
										 |  |  |         def __new__(cls, %(argtxt)s): | 
					
						
							|  |  |  |             return tuple.__new__(cls, (%(argtxt)s,)) | 
					
						
							|  |  |  |         def __repr__(self): | 
					
						
							|  |  |  |             return '%(typename)s(%(reprtxt)s)' %% self | 
					
						
							| 
									
										
										
										
											2007-09-17 00:55:00 +00:00
										 |  |  |         def __replace__(self, field, value): | 
					
						
							|  |  |  |             'Return a new %(typename)s object replacing one field with a new value' | 
					
						
							|  |  |  |             return %(typename)s(**dict(zip(%(field_names)r, self) + [(field, value)])) | 
					
						
							| 
									
										
										
										
											2007-05-19 01:11:16 +00:00
										 |  |  |     ''' % locals()
 | 
					
						
							|  |  |  |     for i, name in enumerate(field_names): | 
					
						
							| 
									
										
										
										
											2007-05-19 01:50:11 +00:00
										 |  |  |         template += '\n        %s = property(itemgetter(%d))\n' % (name, i) | 
					
						
							| 
									
										
										
										
											2007-05-19 01:11:16 +00:00
										 |  |  |     m = dict(itemgetter=_itemgetter) | 
					
						
							|  |  |  |     exec template in m | 
					
						
							|  |  |  |     result = m[typename] | 
					
						
							|  |  |  |     if hasattr(_sys, '_getframe'): | 
					
						
							|  |  |  |         result.__module__ = _sys._getframe(1).f_globals['__name__'] | 
					
						
							|  |  |  |     return result | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-03-01 06:16:43 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == '__main__': | 
					
						
							| 
									
										
										
										
											2007-09-17 00:55:00 +00:00
										 |  |  |     # verify that instances can be pickled | 
					
						
							| 
									
										
										
										
											2007-03-01 06:16:43 +00:00
										 |  |  |     from cPickle import loads, dumps | 
					
						
							| 
									
										
										
										
											2007-09-17 00:55:00 +00:00
										 |  |  |     Point = NamedTuple('Point', 'x, y') | 
					
						
							| 
									
										
										
										
											2007-03-01 06:16:43 +00:00
										 |  |  |     p = Point(x=10, y=20) | 
					
						
							|  |  |  |     assert p == loads(dumps(p)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     import doctest | 
					
						
							|  |  |  |     TestResults = NamedTuple('TestResults', 'failed attempted') | 
					
						
							|  |  |  |     print TestResults(*doctest.testmod()) |