2011-06-15 21:57:02 +00:00
|
|
|
import contextlib
|
|
|
|
import sys
|
2011-06-16 17:37:11 +00:00
|
|
|
import logging
|
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
2011-06-15 21:57:02 +00:00
|
|
|
|
|
|
|
@contextlib.contextmanager
|
|
|
|
def nested(*managers):
|
|
|
|
"""
|
|
|
|
Like contextlib.nested but takes callables returning context
|
|
|
|
managers, to avoid the major reason why contextlib.nested was
|
|
|
|
deprecated.
|
2011-06-16 17:37:11 +00:00
|
|
|
|
|
|
|
This version also logs any exceptions early, much like run_tasks,
|
|
|
|
to ease debugging. TODO combine nested and run_tasks.
|
2011-06-15 21:57:02 +00:00
|
|
|
"""
|
|
|
|
exits = []
|
|
|
|
vars = []
|
|
|
|
exc = (None, None, None)
|
|
|
|
try:
|
|
|
|
for mgr_fn in managers:
|
|
|
|
mgr = mgr_fn()
|
|
|
|
exit = mgr.__exit__
|
|
|
|
enter = mgr.__enter__
|
|
|
|
vars.append(enter())
|
|
|
|
exits.append(exit)
|
|
|
|
yield vars
|
2013-08-30 15:58:10 +00:00
|
|
|
except Exception:
|
2011-06-16 17:37:11 +00:00
|
|
|
log.exception('Saw exception from nested tasks')
|
2011-06-15 21:57:02 +00:00
|
|
|
exc = sys.exc_info()
|
|
|
|
finally:
|
|
|
|
while exits:
|
|
|
|
exit = exits.pop()
|
|
|
|
try:
|
|
|
|
if exit(*exc):
|
|
|
|
exc = (None, None, None)
|
2013-08-30 15:58:10 +00:00
|
|
|
except Exception:
|
2011-06-15 21:57:02 +00:00
|
|
|
exc = sys.exc_info()
|
|
|
|
if exc != (None, None, None):
|
|
|
|
# Don't rely on sys.exc_info() still containing
|
|
|
|
# the right information. Another exception may
|
|
|
|
# have been raised and caught by an exit method
|
|
|
|
raise exc[0], exc[1], exc[2]
|