Como enviar resultados de processos filhos para arquivos e terminal simultaneamente em Python
Ao empregar subprocess.call(), é possível especificar descritores de arquivo como outf e errf para redirecionar stdout e stderr para arquivos específicos. No entanto, esses resultados não serão exibidos simultaneamente no terminal.
Solução usando Popen e Threading:
Para superar isso, podemos aproveitar Popen diretamente e utilizar o argumento stdout=PIPE para ler o stdout do processo filho. Veja como:
import subprocess
from threading import Thread
def tee(infile, *files):
# Forward output from `infile` to `files` in a separate thread
def fanout(infile, *files):
for line in iter(infile.readline, ""):
for f in files:
f.write(line)
t = Thread(target=fanout, args=(infile,) files)
t.daemon = True
t.start()
return t
def teed_call(cmd_args, **kwargs):
# Override `stdout` and `stderr` arguments with PIPE to capture standard outputs
stdout, stderr = [kwargs.pop(s, None) for s in ["stdout", "stderr"]]
p = subprocess.Popen(
cmd_args,
stdout=subprocess.PIPE if stdout is not None else None,
stderr=subprocess.PIPE if stderr is not None else None,
**kwargs
)
# Create threads to simultaneously write to files and terminal
threads = []
if stdout is not None:
threads.append(tee(p.stdout, stdout, sys.stdout))
if stderr is not None:
threads.append(tee(p.stderr, stderr, sys.stderr))
# Join the threads to ensure IO completion before proceeding
for t in threads:
t.join()
return p.wait()
Usando esta função, podemos executar processos filhos e gravar sua saída nos arquivos e no terminal ao mesmo tempo:
outf, errf = open("out.txt", "wb"), open("err.txt", "wb")
teed_call(["cat", __file__], stdout=None, stderr=errf)
teed_call(["echo", "abc"], stdout=outf, stderr=errf, bufsize=0)
teed_call(["gcc", "a b"], close_fds=True, stdout=outf, stderr=errf)
Isenção de responsabilidade: Todos os recursos fornecidos são parcialmente provenientes da Internet. Se houver qualquer violação de seus direitos autorais ou outros direitos e interesses, explique os motivos detalhados e forneça prova de direitos autorais ou direitos e interesses e envie-a para o e-mail: [email protected]. Nós cuidaremos disso para você o mais rápido possível.
Copyright© 2022 湘ICP备2022001581号-3