Commit 77452fc1 authored by Nick Coghlan's avatar Nick Coghlan

Close #14969: Improve the handling of exception chaining in contextlib.ExitStack

parent c4b78a3e
...@@ -225,6 +225,17 @@ class ExitStack(object): ...@@ -225,6 +225,17 @@ class ExitStack(object):
return self return self
def __exit__(self, *exc_details): def __exit__(self, *exc_details):
# We manipulate the exception state so it behaves as though
# we were actually nesting multiple with statements
frame_exc = sys.exc_info()[1]
def _fix_exception_context(new_exc, old_exc):
while 1:
exc_context = new_exc.__context__
if exc_context in (None, frame_exc):
break
new_exc = exc_context
new_exc.__context__ = old_exc
# Callbacks are invoked in LIFO order to match the behaviour of # Callbacks are invoked in LIFO order to match the behaviour of
# nested context managers # nested context managers
suppressed_exc = False suppressed_exc = False
...@@ -236,9 +247,8 @@ class ExitStack(object): ...@@ -236,9 +247,8 @@ class ExitStack(object):
exc_details = (None, None, None) exc_details = (None, None, None)
except: except:
new_exc_details = sys.exc_info() new_exc_details = sys.exc_info()
if exc_details != (None, None, None):
# simulate the stack of exceptions by setting the context # simulate the stack of exceptions by setting the context
new_exc_details[1].__context__ = exc_details[1] _fix_exception_context(new_exc_details[1], exc_details[1])
if not self._exit_callbacks: if not self._exit_callbacks:
raise raise
exc_details = new_exc_details exc_details = new_exc_details
......
...@@ -505,6 +505,18 @@ class TestExitStack(unittest.TestCase): ...@@ -505,6 +505,18 @@ class TestExitStack(unittest.TestCase):
def __exit__(self, *exc_details): def __exit__(self, *exc_details):
raise self.exc raise self.exc
class RaiseExcWithContext:
def __init__(self, outer, inner):
self.outer = outer
self.inner = inner
def __enter__(self):
return self
def __exit__(self, *exc_details):
try:
raise self.inner
except:
raise self.outer
class SuppressExc: class SuppressExc:
def __enter__(self): def __enter__(self):
return self return self
...@@ -514,8 +526,7 @@ class TestExitStack(unittest.TestCase): ...@@ -514,8 +526,7 @@ class TestExitStack(unittest.TestCase):
try: try:
with RaiseExc(IndexError): with RaiseExc(IndexError):
with RaiseExc(KeyError): with RaiseExcWithContext(KeyError, AttributeError):
with RaiseExc(AttributeError):
with SuppressExc(): with SuppressExc():
with RaiseExc(ValueError): with RaiseExc(ValueError):
1 / 0 1 / 0
...@@ -553,12 +564,8 @@ class TestExitStack(unittest.TestCase): ...@@ -553,12 +564,8 @@ class TestExitStack(unittest.TestCase):
except IndexError as exc: except IndexError as exc:
self.assertIsInstance(exc.__context__, KeyError) self.assertIsInstance(exc.__context__, KeyError)
self.assertIsInstance(exc.__context__.__context__, AttributeError) self.assertIsInstance(exc.__context__.__context__, AttributeError)
# Inner exceptions were suppressed, but the with statement # Inner exceptions were suppressed
# cleanup code adds the one from the body back in as the self.assertIsNone(exc.__context__.__context__.__context__)
# context of the exception raised by the outer callbacks
# See http://bugs.python.org/issue14969
suite_exc = exc.__context__.__context__.__context__
self.assertIsInstance(suite_exc, ZeroDivisionError)
else: else:
self.fail("Expected IndexError, but no exception was raised") self.fail("Expected IndexError, but no exception was raised")
# Check the inner exceptions # Check the inner exceptions
......
...@@ -10,6 +10,8 @@ What's New in Python 3.3.0 Beta 1? ...@@ -10,6 +10,8 @@ What's New in Python 3.3.0 Beta 1?
Library Library
------- -------
- Issue #14969: Better handling of exception chaining in contextlib.ExitStack
- Issue #14962: Update text coloring in IDLE shell window after changing - Issue #14962: Update text coloring in IDLE shell window after changing
options. Patch by Roger Serwy. options. Patch by Roger Serwy.
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment