summaryrefslogtreecommitdiffstats
path: root/Lib
diff options
context:
space:
mode:
authorRaymond Hettinger <python@rcn.com>2009-04-01 18:50:56 (GMT)
committerRaymond Hettinger <python@rcn.com>2009-04-01 18:50:56 (GMT)
commit66c4a6b51cea40215e8f61e1abe2e3d89c4aeb1e (patch)
tree9514206018d91419cad25cdfdc9f8f4084270935 /Lib
parent449b7d95d46f0b2e8a1fd73642d391ac3a29518e (diff)
downloadcpython-66c4a6b51cea40215e8f61e1abe2e3d89c4aeb1e.zip
cpython-66c4a6b51cea40215e8f61e1abe2e3d89c4aeb1e.tar.gz
cpython-66c4a6b51cea40215e8f61e1abe2e3d89c4aeb1e.tar.bz2
Issue #5647: MutableSet.__iand__() no longer mutates self during iteration.
Diffstat (limited to 'Lib')
-rw-r--r--Lib/_abcoll.py7
-rw-r--r--Lib/test/test_collections.py25
2 files changed, 28 insertions, 4 deletions
diff --git a/Lib/_abcoll.py b/Lib/_abcoll.py
index 40cc23e..990ff00 100644
--- a/Lib/_abcoll.py
+++ b/Lib/_abcoll.py
@@ -286,10 +286,9 @@ class MutableSet(Set):
self.add(value)
return self
- def __iand__(self, c):
- for value in self:
- if value not in c:
- self.discard(value)
+ def __iand__(self, it):
+ for value in (self - it):
+ self.discard(value)
return self
def __ixor__(self, it):
diff --git a/Lib/test/test_collections.py b/Lib/test/test_collections.py
index 9e984ba..8ffa75b 100644
--- a/Lib/test/test_collections.py
+++ b/Lib/test/test_collections.py
@@ -327,6 +327,25 @@ class TestOneTrickPonyABCs(ABCTestCase):
B.register(C)
self.failUnless(issubclass(C, B))
+class WithSet(MutableSet):
+
+ def __init__(self, it=()):
+ self.data = set(it)
+
+ def __len__(self):
+ return len(self.data)
+
+ def __iter__(self):
+ return iter(self.data)
+
+ def __contains__(self, item):
+ return item in self.data
+
+ def add(self, item):
+ self.data.add(item)
+
+ def discard(self, item):
+ self.data.discard(item)
class TestCollectionABCs(ABCTestCase):
@@ -363,6 +382,12 @@ class TestCollectionABCs(ABCTestCase):
self.validate_abstract_methods(MutableSet, '__contains__', '__iter__', '__len__',
'add', 'discard')
+ def test_issue_5647(self):
+ # MutableSet.__iand__ mutated the set during iteration
+ s = WithSet('abcd')
+ s &= WithSet('cdef') # This used to fail
+ self.assertEqual(set(s), set('cd'))
+
def test_issue_4920(self):
# MutableSet.pop() method did not work
class MySet(collections.MutableSet):