contextvars — Context Variables¶
Este módulo proporciona APIs para gestionar, almacenar y acceder a estados en el contexto local. La clase ContextVar se utiliza para declarar y trabajar con Variables de Contexto (Context Variables). La función copy_context() y la clase Context deberían ser utilizadas para gestionar el contexto actual en frameworks asíncronos.
Context managers that have state should use Context Variables
instead of threading.local() to prevent their state from
bleeding to other code unexpectedly, when used in concurrent code.
Ver PEP 567 para más detalles.
Added in version 3.7.
variables de contexto¶
- class contextvars.ContextVar(name[, *, default])¶
- Esta clase se utiliza para declarar una nueva Variable de Contexto, por ejemplo: - var: ContextVar[int] = ContextVar('var', default=42) - El parámetro obligatorio name se utiliza para introspección y depuración. - El parámetro opcional de sólo palabra clave default es utilizado por - ContextVar.get(), cuando en el contexto actual no se encuentra ningún valor para la variable.- Importante: las Variables de Contexto deberían ser creadas en lo más alto a nivel de módulo y nunca en clausura. Los objetos - Contextmantienen referencias a variables de contexto, lo cual no permitiría que estas variables de contexto sean limpiadas por el recolector de basura.- name¶
- El nombre de la variable. Propiedad de sólo lectura. - Added in version 3.7.1. 
 - get([default])¶
- Retorna un valor para la variable de contexto en el contexto actual. - Si la variable no tiene ningún valor en el contexto actual, el método: - retornará el valor del argumento default del método, si alguno fue dado; o 
- retornará el valor por defecto de la variable de contexto, si ésta fue creada con alguno; o 
- lanzará - LookupError.
 
 - set(value)¶
- Establece un nuevo valor para la variable de contexto en el contexto actual. - El argumento obligatorio value es el nuevo valor de la variable de contexto. - Retorna un objeto - Tokenque puede utilizarse para restaurar la variable a su valor anterior, utilizando el método- ContextVar.reset().
 - reset(token)¶
- Restablece la variable de contexto al valor que tenía antes de llamar al método - ContextVar.set(), que creó el token utilizado.- Por ejemplo: - var = ContextVar('var') token = var.set('new value') # code that uses 'var'; var.get() returns 'new value'. var.reset(token) # After the reset call the var has no value again, so # var.get() would raise a LookupError. 
 
- class contextvars.Token¶
- Los objetos token son retornados por el método - ContextVar.set(). Se le pueden dar al método- ContextVar.reset()para restablecer el valor de la variable al que estuviese dado antes del set correspondiente.- var¶
- Propiedad de sólo lectura. Apunta al objeto - ContextVarque creó el token.
 - old_value¶
- Propiedad de sólo lectura. Es el valor que la variable tenía antes de llamar al método - ContextVar.set()que creó el token. Apunta a- Token.MISSINGsi la variable no estaba establecida antes de la llamada.
 - MISSING¶
- Marcador utilizado por - Token.old_value.
 
Gestión de contexto manual¶
- contextvars.copy_context()¶
- Retorna una copia del objeto - Contextactual.- El siguiente código obtiene una copia del contexto actual e imprime todas las variables y sus valores establecidos en el contexto: - ctx: Context = copy_context() print(list(ctx.items())) - The function has an O(1) complexity, i.e. works equally fast for contexts with a few context variables and for contexts that have a lot of them. 
- class contextvars.Context¶
- Mapeo de - ContextVarscon sus valores.- Context()crea un contexto vacío sin valores. Para obtener una copia del contexto actual, se puede utilizar la función- copy_context().- Each thread has its own effective stack of - Contextobjects. The current context is the- Contextobject at the top of the current thread’s stack. All- Contextobjects in the stacks are considered to be entered.- Entering a context, which can be done by calling its - run()method, makes the context the current context by pushing it onto the top of the current thread’s context stack.- Exiting from the current context, which can be done by returning from the callback passed to the - run()method, restores the current context to what it was before the context was entered by popping the context off the top of the context stack.- Since each thread has its own context stack, - ContextVarobjects behave in a similar fashion to- threading.local()when values are assigned in different threads.- Attempting to enter an already entered context, including contexts entered in other threads, raises a - RuntimeError.- After exiting a context, it can later be re-entered (from any thread). - Any changes to - ContextVarvalues via the- ContextVar.set()method are recorded in the current context. The- ContextVar.get()method returns the value associated with the current context. Exiting a context effectively reverts any changes made to context variables while the context was entered (if needed, the values can be restored by re-entering the context).- Context implementa la interfaz - collections.abc.Mapping.- run(callable, *args, **kwargs)¶
- Enters the Context, executes - callable(*args, **kwargs), then exits the Context. Returns callable’s return value, or propagates an exception if one occurred.- Example: - import contextvars var = contextvars.ContextVar('var') var.set('spam') print(var.get()) # 'spam' ctx = contextvars.copy_context() def main(): # 'var' was set to 'spam' before # calling 'copy_context()' and 'ctx.run(main)', so: print(var.get()) # 'spam' print(ctx[var]) # 'spam' var.set('ham') # Now, after setting 'var' to 'ham': print(var.get()) # 'ham' print(ctx[var]) # 'ham' # Any changes that the 'main' function makes to 'var' # will be contained in 'ctx'. ctx.run(main) # The 'main()' function was run in the 'ctx' context, # so changes to 'var' are contained in it: print(ctx[var]) # 'ham' # However, outside of 'ctx', 'var' is still set to 'spam': print(var.get()) # 'spam' 
 - copy()¶
- Retorna una copia superficial (shallow copy) del objeto de contexto. 
 - var in context
- Retorna - Truesi context tiene un valor establecido para var; de lo contrario, retorna- False.
 - context[var]
- Retorna el valor de la variable - ContextVarvar. Si la variable no está establecida en el contexto actual, se lanzará- KeyError.
 - get(var[, default])¶
- Retorna el valor de var, si var tiene el valor en el objeto de contexto; de lo contrario, retorna default. Si default no es dado, retorna - None.
 - iter(context)
- Retorna un iterador de las variables almacenadas en el objeto de contexto. 
 - len(proxy)
- Retorna el número de variables establecidas en el objeto de contexto. 
 - keys()¶
- Retorna un listado de todas las variables en el objeto de contexto. 
 - values()¶
- Retorna un listado de los valores de todas las variables en el objeto de contexto. 
 - items()¶
- Retorna un listado de dos tuplas que contienen todas las variables y sus variables en el contexto actual. 
 
Soporte asyncio¶
Las variables de contexto están soportadas de forma nativa en asyncio y se pueden utilizar sin ninguna configuración adicional. Por ejemplo, el siguiente código crea un servidor simple de respuesta, que utiliza una variable de contexto que hace que la dirección del cliente remoto esté disponible en la Task que gestiona al cliente:
import asyncio
import contextvars
client_addr_var = contextvars.ContextVar('client_addr')
def render_goodbye():
    # The address of the currently handled client can be accessed
    # without passing it explicitly to this function.
    client_addr = client_addr_var.get()
    return f'Good bye, client @ {client_addr}\r\n'.encode()
async def handle_request(reader, writer):
    addr = writer.transport.get_extra_info('socket').getpeername()
    client_addr_var.set(addr)
    # In any code that we call is now possible to get
    # client's address by calling 'client_addr_var.get()'.
    while True:
        line = await reader.readline()
        print(line)
        if not line.strip():
            break
    writer.write(b'HTTP/1.1 200 OK\r\n')  # status line
    writer.write(b'\r\n')  # headers
    writer.write(render_goodbye())  # body
    writer.close()
async def main():
    srv = await asyncio.start_server(
        handle_request, '127.0.0.1', 8081)
    async with srv:
        await srv.serve_forever()
asyncio.run(main())
# To test it you can use telnet or curl:
#     telnet 127.0.0.1 8081
#     curl 127.0.0.1:8081