パイプと組み合わせて外部コマンドから出力を取得する方法
質問
このようなコマンドがあります。
wmctrl -lp | awk '/gedit/ { print $1 }'
そして、Pythonスクリプト内で出力したいので、このコードを試しました
>>> import subprocess
>>> proc = subprocess.Popen(["wmctrl -lp", "|","awk '/gedit/ {print $1}"], shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
>>> proc.stdout.readline()
'0x0160001b -1 6504 beer-laptop x-nautilus-desktop\n'
>>> proc.stdout.readline()
'0x0352f117 0 6963 beer-laptop How to get output from external command combine with Pipe - Stack Overflow - Chromium\n'
>>> proc.stdout.readline()
'0x01400003 -1 6503 beer-laptop Bottom Expanded Edge Panel\n'
>>>
wmctrl -lp
が実行されただけで、私のコードは間違っているようです。 | awk '{print $ 1}'
は省略されます
私の期待する出力は 0x03800081
$ wmctrl -lp | awk '/gedit/ {print $1}'
0x03800081
サポートしてください。
解決
shell = True
では、配列の代わりに単一のコマンドラインを使用する必要があります。そうでない場合、追加の引数はシェル引数として解釈されます。 subprocess
ドキュメントから:
Unixでは、shell = Trueの場合:argsが文字列の場合、シェルを介して実行するコマンド文字列を指定します。 argsがシーケンスの場合、最初の項目はコマンド文字列を指定し、追加の項目は追加のシェル引数として扱われます。
だからあなたの呼び出しは次のようになります:
subprocess.Popen("wmctrl -lp | sed /gedit/ '{print $1}'", shell=True, ...
不均衡な一重引用符も含まれている可能性があると思います。
他のヒント
プログラムにシーケンスを渡すため、パイプは argument への wmcrtrl
であるとみなします。たとえば、
wmctrl -lp "|"
したがって、実際のパイプ操作は失われます。
単一の文字列にすることで、正しい結果が得られるはずです:
>>> import subprocess as s
>>> proc = s.Popen("echo hello | grep e", shell=True, stdout=s.PIPE, stderr=s.PIPE)
>>> proc.stdout.readline()
'hello\n'
>>> proc.stdout.readline()
''
いくつかの調査の後、私は非常にうまく機能する次のコードを持っています。基本的には、stdoutとstderrの両方をリアルタイムで出力します。それが必要な他の誰かに役立つことを願っています。
stdout_result = 1
stderr_result = 1
def stdout_thread(pipe):
global stdout_result
while True:
out = pipe.stdout.read(1)
stdout_result = pipe.poll()
if out == '' and stdout_result is not None:
break
if out != '':
sys.stdout.write(out)
sys.stdout.flush()
def stderr_thread(pipe):
global stderr_result
while True:
err = pipe.stderr.read(1)
stderr_result = pipe.poll()
if err == '' and stderr_result is not None:
break
if err != '':
sys.stdout.write(err)
sys.stdout.flush()
def exec_command(command, cwd=None):
if cwd is not None:
print '[' + ' '.join(command) + '] in ' + cwd
else:
print '[' + ' '.join(command) + ']'
p = subprocess.Popen(
command, stdout=subprocess.PIPE, stderr=subprocess.PIPE, cwd=cwd
)
out_thread = threading.Thread(name='stdout_thread', target=stdout_thread, args=(p,))
err_thread = threading.Thread(name='stderr_thread', target=stderr_thread, args=(p,))
err_thread.start()
out_thread.start()
out_thread.join()
err_thread.join()
return stdout_result + stderr_result
必要に応じて、出力またはエラーを文字列に収集して返すのは簡単だと思います。
所属していません StackOverflow