Sub-procesos

Código fuente: Lib/asyncio/subprocess.py, Lib/asyncio/base_subprocess.py


Esta sección describe APIs de alto nivel de async/await de asyncio para crear y gestionar sub-procesos.

Aquí podemos encontrar un ejemplo de cómo asyncio puede ejecutar un comando de shell y obtener su resultado:

import asyncio

async def run(cmd):
    proc = await asyncio.create_subprocess_shell(
        cmd,
        stdout=asyncio.subprocess.PIPE,
        stderr=asyncio.subprocess.PIPE)

    stdout, stderr = await proc.communicate()

    print(f'[{cmd!r} exited with {proc.returncode}]')
    if stdout:
        print(f'[stdout]\n{stdout.decode()}')
    if stderr:
        print(f'[stderr]\n{stderr.decode()}')

asyncio.run(run('ls /zzz'))

mostrará en pantalla:

['ls /zzz' exited with 1]
[stderr]
ls: /zzz: No such file or directory

Ya que todos las funciones de sub-procesos de asyncio son asíncronas y asyncio proporciona herramientas para trabajar con tales funciones, es fácil ejecutar y monitorear múltiples subprocesos en paralelo. De hecho es trivial modificar los ejemplos de arriba para ejecutar varios comandos simultáneamente:

async def main():
    await asyncio.gather(
        run('ls /zzz'),
        run('sleep 1; echo "hello"'))

asyncio.run(main())

Véase también la subsección Examples.

Creando sub-procesos

coroutine asyncio.create_subprocess_exec(program, *args, stdin=None, stdout=None, stderr=None, limit=None, **kwds)

Crea un sub-proceso.

El argumento limit establece el límite del buffer para los envoltorios StreamReader para Process.stdout y Process.stderr (si se pasa subprocess.PIPE a los argumentos stdout y stderr).

Retorna una instancia de Process.

Véase la documentación de loop.subprocess_exec() para los otros parámetros.

Distinto en la versión 3.10: Se eliminó el parámetro loop.

coroutine asyncio.create_subprocess_shell(cmd, stdin=None, stdout=None, stderr=None, limit=None, **kwds)

Ejecuta el comando de shell cmd.

El argumento limit establece el límite del buffer para los envoltorios StreamReader para Process.stdout y Process.stderr (si se pasa subprocess.PIPE a los argumentos stdout y stderr).

Retorna una instancia de Process.

Véase la documentación de loop.subprocess_shell() para los otros parámetros.

Importante

Es la responsabilidad de la aplicación asegurarse que todos los espacios en blanco y caracteres especiales estén citados apropiadamente para evitar vulnerabilidades de inyección de instrucciones. La función shlex.quote() puede ser usada para escapar caracteres en blanco y caracteres especiales de shell en cadenas de caracteres a ser usadas para construir comandos de shell.

Distinto en la versión 3.10: Se eliminó el parámetro loop.

Nota

Los subprocesos están disponibles para Windows si se utiliza un ProactorEventLoop. Consulte Soporte de subprocesos en Windows para obtener más detalles.

Ver también

asyncio también tiene las siguientes APIs de bajo nivel para trabajar con sub-procesos: loop.subprocess_exec(), loop.subprocess_shell(), loop.connect_read_pipe(), loop.connect_write_pipe(), así como también los Sub-procesos de Transportes y los Sub-procesos de Protocolos.

Constantes

asyncio.subprocess.PIPE

Se le puede pasar a los parámetros stding, stdout o stderr.

Si se le pasa PIPE al argumento stdin, el atributo Process.stdin apuntará a la instancia StreamWriter.

Si se pasa PIPE a los argumentos stdout o stderr, los atributos Process.stdout y Process.stderr apuntarán a las instancias StreamReader.

asyncio.subprocess.STDOUT

Un valor especial que puede ser usado como el argumento de stderr e indica que los errores estándar deben ser redireccionados en la salida estándar.

asyncio.subprocess.DEVNULL

Un valor especial que puede ser usado como el argumento de stdin, stdout, stderr para procesar funciones de creación. Indica que el archivo especial os.devnull será usado para la correspondiente transmisión del sub-proceso.

Interactuando con Subprocesos

Las dos funciones create_subprocess_exec() y create_subprocess_shell() retornan instancias de la clase Process. Process es un envoltorio de alto nivel que permite comunicarse con los subprocesos y estar atento a sus términos.

class asyncio.subprocess.Process

Un objeto que envuelve procesos del SO creados por las funciones create_subprocess_exec() y create_subprocess_shell().

Esta clase está diseñada para tener un API similar a la clase subprocess.Popen, pero hay algunas diferencias notables:

  • a diferencia de Popen, las instancias de Process no tiene un equivalente del método poll();

  • los métodos communicate() y wait() no tienen un parámetro timeout: use la función wait_for();

  • el método Process.wait() es asíncrono, mientras que el método subprocess.Popen.wait() es implementado como un bucle de espera activa;

  • el parámetro universal_newlines no es compatible.

Esta clase no es segura para subprocesos (not thread safe).

Véase también la sección Subprocesos e Hilos.

coroutine wait()

Espera a que el proceso hijo termine.

Define y retorna el atributo returncode.

Nota

Este método puede bloquearse cuando usa stdout=PIPE o stderr=PIPE y el proceso hijo genera tanta salida que se bloquea esperando a que la tubería de búfer del SO acepte más datos. Use el método communicate() cuando use tuberías para evitar esta condición.

coroutine communicate(input=None)

Interactuar con el proceso:

  1. envía datos a stdin (si input no es None);

  2. cierra stdin;

  3. lee datos desde stdout y stderr, hasta que el llegue al EOF;

  4. espera a que el proceso termine.

El argumento opcional input son los datos (objetos bytes) que serán enviados a los procesos hijos.

Retorna una tupla (stdout_data, stderr_data).

Si se lanza la excepción BrokenPipeError o ConnectionResetError cuando se escribe input en stdin, la excepción es ignorada. Esta condición ocurre cuando el proceso finaliza antes de que todos los datos sean escritos en stdin.

Si se desea enviar datos al stdin del proceso, el proceso necesita ser creado con stdin=PIPE. De manera similar, para obtener cualquier cosa excepto None en la tupla del resultado, el proceso tiene que ser creado con los argumentos stdout=PIPE y/o stderr=PIPE.

Tenga en cuenta que los datos leídos son almacenados en memoria, por lo que no utilice este método si el tamaño de los datos es más grande o ilimitado.

Distinto en la versión 3.12: stdin se cierra cuando input=None también.

send_signal(signal)

Envíe la señal signal al proceso hijo.

Nota

On Windows, SIGTERM is an alias for terminate(). CTRL_C_EVENT and CTRL_BREAK_EVENT can be sent to processes started with a creationflags parameter which includes CREATE_NEW_PROCESS_GROUP.

terminate()

Para al proceso hijo.

On POSIX systems this method sends SIGTERM to the child process.

On Windows the Win32 API function TerminateProcess() is called to stop the child process.

kill()

Mata el proceso hijo.

En sistemas POSIX, este método envía SIGKILL al proceso hijo.

En Windows este método es un alias para terminate().

stdin

Flujo de entrada estándar (StreamWriter) o None si el proceso fue creado con stdin=None.

stdout

Flujo de salida estándar (SreamReader) o None si el proceso fue creado con stdout=None.

stderr

Flujo de salida estándar (StreamReader) o None si el proceso fue creado con stderr=None.

Advertencia

Utilice el método communicate() en vez de process.stdin.write(), await process.stdout.read() o await process.stderr.read. Esto evita bloqueos debido a que los flujos pausan la lectura o escritura y bloqueando al proceso hijo.

pid

Número de identificación del Proceso (PID por sus siglas en inglés).

Tenga en cuenta que para procesos creados por la función create­_subprocess_shell(), este atributo es el PID del shell generado.

returncode

Código de retorno del proceso cuando finaliza.

Un valor None indica que el proceso todavía no ha finalizado.

Un valor negativo -N indica que el hijo ha finalizado por la señal N (sólo para POSIX).

Subprocesos y Hilos

Por defecto el bucle de eventos de asyncio estándar permite correr subprocesos desde hilos diferentes.

En Windows, sólo ProactorEventLoop proporciona subprocesos (por defecto), SelectorEventLoop no es compatible con subprocesos.

Tenga en cuenta que implementaciones alternativas del bucle de eventos pueden tener limitaciones propias; por favor consulte su documentación.

Ver también

La sección Concurrencia y multihilos en asyncio.

Ejemplos

Un ejemplo usando la clase Process para controlar un subproceso y la clase StreamReader para leer de su salida estándar.

El subproceso es creado por la función create_subprocess_exec():

import asyncio
import sys

async def get_date():
    code = 'import datetime; print(datetime.datetime.now())'

    # Create the subprocess; redirect the standard output
    # into a pipe.
    proc = await asyncio.create_subprocess_exec(
        sys.executable, '-c', code,
        stdout=asyncio.subprocess.PIPE)

    # Read one line of output.
    data = await proc.stdout.readline()
    line = data.decode('ascii').rstrip()

    # Wait for the subprocess exit.
    await proc.wait()
    return line

date = asyncio.run(get_date())
print(f"Current date: {date}")

Véase también los ejemplos de prueba escritos usando APIs de bajo nivel.