1 """Utilities for with-statement contexts. See PEP 343."""
5 __all__
= ["contextmanager", "nested", "closing"]
7 class GeneratorContextManager(object):
8 """Helper for @contextmanager decorator."""
10 def __init__(self
, gen
):
13 def __context__(self
):
18 return self
.gen
.next()
20 raise RuntimeError("generator didn't yield")
22 def __exit__(self
, type, value
, traceback
):
29 raise RuntimeError("generator didn't stop")
32 self
.gen
.throw(type, value
, traceback
)
38 def contextmanager(func
):
39 """@contextmanager decorator.
44 def some_generator(<arguments>):
53 with some_generator(<arguments>) as <variable>:
66 def helper(*args
, **kwds
):
67 return GeneratorContextManager(func(*args
, **kwds
))
69 helper
.__name
__ = func
.__name
__
70 helper
.__doc
__ = func
.__doc
__
77 def nested(*contexts
):
78 """Support multiple context managers in a single with-statement.
82 with nested(A, B, C) as (X, Y, Z):
85 is equivalent to this:
95 exc
= (None, None, None)
98 for context
in contexts
:
99 mgr
= context
.__context
__()
101 enter
= mgr
.__enter
__
112 exc
= (None, None, None)
115 if exc
!= (None, None, None):
121 """Context manager to automatically close something at the end of a block.
125 with closing(<module>.open(<arguments>)) as f:
128 is equivalent to this:
130 f = <module>.open(<arguments>)