summaryrefslogtreecommitdiffstats
path: root/Lib/contextlib.py
diff options
context:
space:
mode:
authorNick Coghlan <ncoghlan@gmail.com>2013-10-01 13:28:00 (GMT)
committerNick Coghlan <ncoghlan@gmail.com>2013-10-01 13:28:00 (GMT)
commite6f4631f0839a004c1d272c48811cdbed3e5ac9d (patch)
tree9cb22ea0168de7b6b812c5f1ce4c6e9fbd7360e4 /Lib/contextlib.py
parentc13516b0a0fd3142b29e1f354c2080cd52948213 (diff)
parent1a33b2f35b9195b440b492afa87dcf83ba2ecca4 (diff)
downloadcpython-e6f4631f0839a004c1d272c48811cdbed3e5ac9d.zip
cpython-e6f4631f0839a004c1d272c48811cdbed3e5ac9d.tar.gz
cpython-e6f4631f0839a004c1d272c48811cdbed3e5ac9d.tar.bz2
Merge #19092 from 3.3
Diffstat (limited to 'Lib/contextlib.py')
-rw-r--r--Lib/contextlib.py18
1 files changed, 15 insertions, 3 deletions
diff --git a/Lib/contextlib.py b/Lib/contextlib.py
index 3222ac8..aaab095 100644
--- a/Lib/contextlib.py
+++ b/Lib/contextlib.py
@@ -237,6 +237,8 @@ class ExitStack(object):
return self
def __exit__(self, *exc_details):
+ received_exc = exc_details[0] is not None
+
# We manipulate the exception state so it behaves as though
# we were actually nesting multiple with statements
frame_exc = sys.exc_info()[1]
@@ -251,17 +253,27 @@ class ExitStack(object):
# Callbacks are invoked in LIFO order to match the behaviour of
# nested context managers
suppressed_exc = False
+ pending_raise = False
while self._exit_callbacks:
cb = self._exit_callbacks.pop()
try:
if cb(*exc_details):
suppressed_exc = True
+ pending_raise = False
exc_details = (None, None, None)
except:
new_exc_details = sys.exc_info()
# simulate the stack of exceptions by setting the context
_fix_exception_context(new_exc_details[1], exc_details[1])
- if not self._exit_callbacks:
- raise
+ pending_raise = True
exc_details = new_exc_details
- return suppressed_exc
+ if pending_raise:
+ try:
+ # bare "raise exc_details[1]" replaces our carefully
+ # set-up context
+ fixed_ctx = exc_details[1].__context__
+ raise exc_details[1]
+ except BaseException:
+ exc_details[1].__context__ = fixed_ctx
+ raise
+ return received_exc and suppressed_exc