| 
									
										
										
										
											2008-02-05 00:20:01 +00:00
										 |  |  | # Access WeakSet through the weakref module. | 
					
						
							|  |  |  | # This code is separated-out because it is needed | 
					
						
							|  |  |  | # by abc.py to load everything else at startup. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | from _weakref import ref | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | __all__ = ['WeakSet'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class WeakSet: | 
					
						
							|  |  |  |     def __init__(self, data=None): | 
					
						
							|  |  |  |         self.data = set() | 
					
						
							|  |  |  |         def _remove(item, selfref=ref(self)): | 
					
						
							|  |  |  |             self = selfref() | 
					
						
							|  |  |  |             if self is not None: | 
					
						
							|  |  |  |                 self.data.discard(item) | 
					
						
							|  |  |  |         self._remove = _remove | 
					
						
							|  |  |  |         if data is not None: | 
					
						
							|  |  |  |             self.update(data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __iter__(self): | 
					
						
							|  |  |  |         for itemref in self.data: | 
					
						
							|  |  |  |             item = itemref() | 
					
						
							|  |  |  |             if item is not None: | 
					
						
							|  |  |  |                 yield item | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-18 16:27:29 +00:00
										 |  |  |     def __len__(self): | 
					
						
							|  |  |  |         return sum(x() is not None for x in self.data) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-05 00:20:01 +00:00
										 |  |  |     def __contains__(self, item): | 
					
						
							|  |  |  |         return ref(item) in self.data | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __reduce__(self): | 
					
						
							|  |  |  |         return (self.__class__, (list(self),), | 
					
						
							|  |  |  |                 getattr(self, '__dict__', None)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def add(self, item): | 
					
						
							|  |  |  |         self.data.add(ref(item, self._remove)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def clear(self): | 
					
						
							|  |  |  |         self.data.clear() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def copy(self): | 
					
						
							|  |  |  |         return self.__class__(self) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def pop(self): | 
					
						
							|  |  |  |         while True: | 
					
						
							| 
									
										
										
										
											2008-05-18 07:46:13 +00:00
										 |  |  |             try: | 
					
						
							|  |  |  |                 itemref = self.data.pop() | 
					
						
							|  |  |  |             except KeyError: | 
					
						
							|  |  |  |                 raise KeyError('pop from empty WeakSet') | 
					
						
							| 
									
										
										
										
											2008-02-05 00:20:01 +00:00
										 |  |  |             item = itemref() | 
					
						
							|  |  |  |             if item is not None: | 
					
						
							|  |  |  |                 return item | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def remove(self, item): | 
					
						
							|  |  |  |         self.data.remove(ref(item)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def discard(self, item): | 
					
						
							|  |  |  |         self.data.discard(ref(item)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def update(self, other): | 
					
						
							|  |  |  |         if isinstance(other, self.__class__): | 
					
						
							|  |  |  |             self.data.update(other.data) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             for element in other: | 
					
						
							|  |  |  |                 self.add(element) | 
					
						
							| 
									
										
										
										
											2008-05-18 16:27:29 +00:00
										 |  |  |     def __ior__(self, other): | 
					
						
							|  |  |  |         self.update(other) | 
					
						
							|  |  |  |         return self | 
					
						
							| 
									
										
										
										
											2008-02-05 00:20:01 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Helper functions for simple delegating methods. | 
					
						
							|  |  |  |     def _apply(self, other, method): | 
					
						
							|  |  |  |         if not isinstance(other, self.__class__): | 
					
						
							|  |  |  |             other = self.__class__(other) | 
					
						
							|  |  |  |         newdata = method(other.data) | 
					
						
							|  |  |  |         newset = self.__class__() | 
					
						
							|  |  |  |         newset.data = newdata | 
					
						
							|  |  |  |         return newset | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def difference(self, other): | 
					
						
							|  |  |  |         return self._apply(other, self.data.difference) | 
					
						
							|  |  |  |     __sub__ = difference | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def difference_update(self, other): | 
					
						
							| 
									
										
										
										
											2008-05-18 16:27:29 +00:00
										 |  |  |         if self is other: | 
					
						
							|  |  |  |             self.data.clear() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.data.difference_update(ref(item) for item in other) | 
					
						
							|  |  |  |     def __isub__(self, other): | 
					
						
							|  |  |  |         if self is other: | 
					
						
							|  |  |  |             self.data.clear() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.data.difference_update(ref(item) for item in other) | 
					
						
							|  |  |  |         return self | 
					
						
							| 
									
										
										
										
											2008-02-05 00:20:01 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def intersection(self, other): | 
					
						
							|  |  |  |         return self._apply(other, self.data.intersection) | 
					
						
							|  |  |  |     __and__ = intersection | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def intersection_update(self, other): | 
					
						
							| 
									
										
										
										
											2008-05-18 16:27:29 +00:00
										 |  |  |         self.data.intersection_update(ref(item) for item in other) | 
					
						
							|  |  |  |     def __iand__(self, other): | 
					
						
							|  |  |  |         self.data.intersection_update(ref(item) for item in other) | 
					
						
							|  |  |  |         return self | 
					
						
							| 
									
										
										
										
											2008-02-05 00:20:01 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def issubset(self, other): | 
					
						
							|  |  |  |         return self.data.issubset(ref(item) for item in other) | 
					
						
							|  |  |  |     __lt__ = issubset | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-18 16:27:29 +00:00
										 |  |  |     def __le__(self, other): | 
					
						
							|  |  |  |         return self.data <= set(ref(item) for item in other) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-05 00:20:01 +00:00
										 |  |  |     def issuperset(self, other): | 
					
						
							|  |  |  |         return self.data.issuperset(ref(item) for item in other) | 
					
						
							|  |  |  |     __gt__ = issuperset | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-18 16:27:29 +00:00
										 |  |  |     def __ge__(self, other): | 
					
						
							|  |  |  |         return self.data >= set(ref(item) for item in other) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __eq__(self, other): | 
					
						
							|  |  |  |         return self.data == set(ref(item) for item in other) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-05 00:20:01 +00:00
										 |  |  |     def symmetric_difference(self, other): | 
					
						
							|  |  |  |         return self._apply(other, self.data.symmetric_difference) | 
					
						
							|  |  |  |     __xor__ = symmetric_difference | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def symmetric_difference_update(self, other): | 
					
						
							| 
									
										
										
										
											2008-05-18 16:27:29 +00:00
										 |  |  |         if self is other: | 
					
						
							|  |  |  |             self.data.clear() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.data.symmetric_difference_update(ref(item) for item in other) | 
					
						
							|  |  |  |     def __ixor__(self, other): | 
					
						
							|  |  |  |         if self is other: | 
					
						
							|  |  |  |             self.data.clear() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.data.symmetric_difference_update(ref(item) for item in other) | 
					
						
							|  |  |  |         return self | 
					
						
							| 
									
										
										
										
											2008-02-05 00:20:01 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def union(self, other): | 
					
						
							| 
									
										
										
										
											2008-05-18 07:46:13 +00:00
										 |  |  |         return self._apply(other, self.data.union) | 
					
						
							| 
									
										
										
										
											2008-02-05 00:20:01 +00:00
										 |  |  |     __or__ = union | 
					
						
							| 
									
										
										
										
											2008-05-18 16:27:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def isdisjoint(self, other): | 
					
						
							|  |  |  |         return len(self.intersection(other)) == 0 |