diff options
author | Phillip J. Eby <pje@telecommunity.com> | 2006-04-03 20:05:05 (GMT) |
---|---|---|
committer | Phillip J. Eby <pje@telecommunity.com> | 2006-04-03 20:05:05 (GMT) |
commit | 9444bd51c46a34410e92560e8f96ab75bd9eb332 (patch) | |
tree | f439b7e2658f49e350f1c5e4fc33862265b0368d /Lib | |
parent | 9161a0d8da0ea3d8159ceed0d0bdb9e85e52e448 (diff) | |
download | cpython-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.py | 4 | ||||
-rw-r--r-- | Lib/test/test_with.py | 56 |
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): |