Uh oh!
There was an error while loading.Please reload this page.
- Notifications
You must be signed in to change notification settings - Fork32k
Description
Bug report
Bug description:
Overview:
ContextDecorator can not be safely used on functions that make recursive calls, or may be used with multithreading, even if those context managers support use in sequential with statements. The documentationsuggests otherwise, and could be clarified. Additionally, this functionality could be added via a class method.
Detailed description
Thedocumentation for ContextDecorator states:
This change is just syntactic sugar for any construct of the following form:
def f(): with cm(): # Do stuff
ContextDecorator lets you instead write:
@cm()def f(): # Do stuff
However, ContextDecorator is closer in functionality to the following:
cm = cm()def f(): with cm: # Do stuff
The documentation does contain the following warning. However, the wording could be more clear, especially given the syntactic sugar example.
Note As the decorated function must be able to be called multiple times, the underlying context manager must support use in multiplewith statements. If this is not the case, then the original construct with the explicit with statement inside the function should be used.
Repro Code
This code demonstrates the issue:
import contextlibimport timeclass timed(contextlib.ContextDecorator): def __enter__(self): self._start_time = time.monotonic() def __exit__(self, *exc): print(f"Execution took {time.monotonic() - self._start_time:.0f} seconds")@timed()def my_func(recurse_once = False): time.sleep(1) if recurse_once: my_func()my_func(recurse_once = True)
Expected output:
Execution took 1 seconds
Execution took 2 seconds
Actual output:
Execution took 1 seconds
Execution took 1 seconds
Potential fixes
Implementing one or more of these fixes could alleviate the issue.
1. Clarify the syntactic sugar section to show that all function invocations share a single instance of CM.
The "syntactic sugar" section could be changed to read:
ContextDecorator lets you instead write:
@cm()def f(): # Do stuff
Which is equivalent to:
cm = cm()def f(): with cm: # Do stuff
2. Clarify the warning note
The warning note could be changed to make it more clear that separate functions share state in the context manager.
Note The underlying context manager is instantiated once when the function definition is evaluated: this instance is shared between all calls to the function. As the decorated function must be able to be called multiple times, the underlying context manager must support use in multiplewith statements. If this is not the case, then the original construct with the explicit with statement inside the function should be used.
3. Provide a method for wrapping functions with stateful context managers
For example:
def wrap_with_context(cm_factory, *cm_args, **cm_kwargs): """Wraps the decorated function with a context created by cm_factory.""" def wrap(func): @functools.wraps(func) def inner(*func_args, **func_kwargs): with cm_factory(*cm_args, **cm_kwargs): return func(*func_args, **func_kwargs) return inner return wrap
The previous example now works as expected:
@wrap_with_context(timed)def my_func2(recurse_once = False): time.sleep(1) if recurse_once: my_func2()my_func2(recurse_once = True)
Actual output:
Execution took 1 seconds
Execution took 2 seconds
4. Add class method to ContextDecorator that acts as a decorator and a factory.
class ContextDecorator:... @classmethod def wrap(cls, *cls_args, **cls_kwargs): def enclose(func): @functools.wraps(func) def inner(*args, **kwds): with cls(*cls_args, **cls_kwargs): return func(*args, **kwds) return inner return enclose
This allows usage that is similar to existing, but instantiates a separate CM for each function invocation:
@timed.wrap()def my_func2(recurse_once = False): time.sleep(1) if recurse_once: my_func2()
CPython versions tested on:
3.13
Operating systems tested on:
macOS