summaryrefslogtreecommitdiffstats
path: root/Lib
diff options
context:
space:
mode:
authorPhillip J. Eby <pje@telecommunity.com>2006-04-03 20:05:05 (GMT)
committerPhillip J. Eby <pje@telecommunity.com>2006-04-03 20:05:05 (GMT)
commit9444bd51c46a34410e92560e8f96ab75bd9eb332 (patch)
treef439b7e2658f49e350f1c5e4fc33862265b0368d /Lib
parent9161a0d8da0ea3d8159ceed0d0bdb9e85e52e448 (diff)
downloadcpython-9444bd51c46a34410e92560e8f96ab75bd9eb332.zip
cpython-9444bd51c46a34410e92560e8f96ab75bd9eb332.tar.gz
cpython-9444bd51c46a34410e92560e8f96ab75bd9eb332.tar.bz2
Fix SF#1462485: StopIteration raised in body of 'with' statement suppressed
Diffstat (limited to 'Lib')
-rw-r--r--Lib/contextlib.py4
-rw-r--r--Lib/test/test_with.py56
2 files changed, 59 insertions, 1 deletions
diff --git a/Lib/contextlib.py b/Lib/contextlib.py
index 9c00ef0..c26e27e 100644
--- a/Lib/contextlib.py
+++ b/Lib/contextlib.py
@@ -32,7 +32,9 @@ class GeneratorContextManager(object):
self.gen.throw(type, value, traceback)
raise RuntimeError("generator didn't stop after throw()")
except StopIteration:
- return True
+ # Supress the exception unless it's the same exception the
+ # was passed to throw().
+ return sys.exc_info()[1] is not value
except:
# only re-raise if it's *not* the exception that was
# passed to throw(), because __exit__() must not raise
diff --git a/Lib/test/test_with.py b/Lib/test/test_with.py
index 4854436..48e00f4 100644
--- a/Lib/test/test_with.py
+++ b/Lib/test/test_with.py
@@ -494,6 +494,62 @@ class ExceptionalTestCase(unittest.TestCase, ContextmanagerAssertionMixin):
self.assertAfterWithGeneratorInvariantsWithError(self.foo)
self.assertAfterWithGeneratorInvariantsNoError(self.bar)
+ def testRaisedStopIteration1(self):
+ @contextmanager
+ def cm():
+ yield
+
+ def shouldThrow():
+ with cm():
+ raise StopIteration("from with")
+
+ self.assertRaises(StopIteration, shouldThrow)
+
+ def testRaisedStopIteration2(self):
+ class cm (object):
+ def __context__(self):
+ return self
+
+ def __enter__(self):
+ pass
+
+ def __exit__(self, type, value, traceback):
+ pass
+
+ def shouldThrow():
+ with cm():
+ raise StopIteration("from with")
+
+ self.assertRaises(StopIteration, shouldThrow)
+
+ def testRaisedGeneratorExit1(self):
+ @contextmanager
+ def cm():
+ yield
+
+ def shouldThrow():
+ with cm():
+ raise GeneratorExit("from with")
+
+ self.assertRaises(GeneratorExit, shouldThrow)
+
+ def testRaisedGeneratorExit2(self):
+ class cm (object):
+ def __context__(self):
+ return self
+
+ def __enter__(self):
+ pass
+
+ def __exit__(self, type, value, traceback):
+ pass
+
+ def shouldThrow():
+ with cm():
+ raise GeneratorExit("from with")
+
+ self.assertRaises(GeneratorExit, shouldThrow)
+
class NonLocalFlowControlTestCase(unittest.TestCase):