it-swarm.it

esecuzione di una riga di comando contenente Pipes e visualizzazione dei risultati su STDOUT

Come si chiamerebbe un comando Shell da Python che contiene una pipe e cattura l'output? 

Supponiamo che il comando fosse qualcosa del tipo:

cat file.log | tail -1

L'equivalente Perl di quello che sto cercando di fare sarebbe qualcosa di simile:

my $string = `cat file.log | tail -1`;
33
spudATX

Utilizzare un sottoprocesso.PIPE, come spiegato nella sezione dei documenti di sottoprocesso "Sostituzione della pipeline di Shell" :

import subprocess
p1 = subprocess.Popen(["cat", "file.log"], stdout=subprocess.PIPE)
p2 = subprocess.Popen(["tail", "-1"], stdin=p1.stdout, stdout=subprocess.PIPE)
p1.stdout.close()  # Allow p1 to receive a SIGPIPE if p2 exits.
output,err = p2.communicate()

Oppure, usando il modulo sh , il piping diventa composizione di funzioni :

import sh
output = sh.tail(sh.cat('file.log'), '-1')
44
unutbu
import subprocess
task = subprocess.Popen("cat file.log | tail -1", Shell=True, stdout=subprocess.PIPE)
data = task.stdout.read()
assert task.wait() == 0

Si noti che questo non cattura lo stderr. E se vuoi catturare anche stderr, dovrai usare task.communicate(); chiamando task.stdout.read() e quindi task.stderr.read() può deadlock se il buffer per stderr si riempie. Se li vuoi combinati, dovresti essere in grado di usare 2>&1 come parte del comando Shell.

Ma dato il tuo caso esatto,

task = subprocess.Popen(['tail', '-1', 'file.log'], stdout=subprocess.PIPE)
data = task.stdout.read()
assert task.wait() == 0

evita assolutamente la necessità del tubo.

7
retracile

Questo:

import subprocess
p = subprocess.Popen("cat file.log | tail -1", Shell=False, stdout=subprocess.PIPE, stderr=subprocess.PIPE, stdin=subprocess.PIPE)
#Try Shell=True if the above doesn't work with Shell=False
p_stdout = p.stdout.read()
p_stderr = p.stderr.read()
print p_stdout

O questo dovrebbe funzionare: 

import os
result = os.system("cat file.log | tail -1")
1
chown

Un altro modo simile a Popen sarebbe:

command=r"""cat file.log | tail -1 """
output=subprocess.check_output(command, Shell=True)
1
XAVI

Questo è un fork da @chown con alcuni miglioramenti: 

  • un alias per import subprocess, facilita l'impostazione dei parametri
  • se si desidera solo l'output, non è necessario impostare stderr o stdin quando si chiama Popen
  • per una migliore formattazione, si consiglia di decodificare l'output
  • Shell=True è necessario, per chiamare un interprete per la riga di comando 

#!/usr/bin/python3

import subprocess as sp

p = sp.Popen("cat app.log | grep guido", Shell=True, stdout=sp.PIPE)

output = p.stdout.read()
print(output.decode('utf-8'))

$ cat app.log 
2017-10-14 22:34:12, User Removed [albert.wesker]
2017-10-26 18:14:02, User Removed [alexei.ivanovich] 
2017-10-28 12:14:56, User Created [ivan.leon]
2017-11-14 09:22:07, User Created [guido.rossum]

$ python3 subproc.py 
2017-11-14 09:22:07, User Created [guido.rossum]
0
ivanleoncz