subprocess.Popen呼び出しの出力を文字列に格納します


300

私はPythonでシステムコールを作成し、Pythonプログラムで操作できる文字列に出力を保存しようとしています。

#!/usr/bin/python
import subprocess
p2 = subprocess.Popen("ntpq -p")

私はここにいくつかの提案を含むいくつかのことを試しました:

subprocess.call()の出力の取得

しかし運がありません。


3
このような具体的な質問については、実行した実際のコードと実際のトレースバックまたは予期しない動作を投稿することを常にお勧めします。たとえば、出力を取得するために何を行おうとしたのかわからないので、実際にそれを開始できなかったのではないかと思われます。のファイルが見つからないというエラーが発生したはず"ntpq -p"です。あなたが尋ねているよりも問題。
マイクグラハム、

回答:


467

Python 2.7またはPython 3の場合

Popenオブジェクトを直接作成する代わりに、subprocess.check_output()関数を使用してコマンドの出力を文字列に保存できます。

from subprocess import check_output
out = check_output(["ntpq", "-p"])

Python 2.4-2.6の場合

communicateメソッドを使用します。

import subprocess
p = subprocess.Popen(["ntpq", "-p"], stdout=subprocess.PIPE)
out, err = p.communicate()

out あなたが欲しいものです。

他の回答についての重要な注意

コマンドをどのように渡したかに注意してください。この"ntpq -p"例は別の問題を提起します。以来Popenシェルを起動していない、あなたは、コマンドとオプション]のリストを使用します["ntpq", "-p"]


3
この場合、pythonはこのシステムコールが完了するまで待機しますか?または、明示的にwait / waitpid関数を呼び出す必要がありますか?
なしタイプ

7
@NoneType Popen.communicate。プロセスが終了するまで戻りません。
マイクグラハム

10
あなたは、エラーストリームの追加標準エラー出力を取得したい場合:p = subprocess.Popen(["ntpq", "-p"], stdout=subprocess.PIPE, stderr=subprocess.PIPE)
Timofey

82
注意して、ではなくオブジェクトをsubprocess.check_output()返します。結果を印刷するだけの場合は、違いはありません。ただし、結果などのメソッドを使用する場合は、まずオブジェクトをデコードする必要があります(たとえば)。bytesstrmyString.split("\n")bytessubprocess.check_output(myParams).decode("utf-8")
TanguyP 2015

17
universal_newlines=Trueパラメータとして追加すると、Python 3で文字列オブジェクトを取得するのに役立ちました。universal_newlinesがTrueの場合、デフォルトのエンコーディングでテキストモードで開かれます。それ以外の場合は、バイナリストリームとして開かれます。
ジョナサンコマー

38

これは私にとってstdoutをリダイレクトするために機能しました(stderrも同様に処理できます):

from subprocess import Popen, PIPE
pipe = Popen(path, stdout=PIPE)
text = pipe.communicate()[0]

うまくいかない場合は、発生している問題を正確に明記してください。


3
これは奇妙なオブジェクトを生成します。文字列に変換すると、のように空白がエスケープされます\n
トマーシュZato -復活モニカ

1
これはサブプロセスが正しく実行されたかどうかをチェックしないことに注意してください。pipe.returncode == 0最後の行の後で確認することもできます。
thakis 18/06/18

これが機能する理由Popenは、とのタプルを返すためです。アクセスするstdoutstderr[0]を取得するだけですstdout。あなたも行うことができtext, err = pipe.communicate()、その後、textあなたは何を期待しています
ジョナ


22

subprocess.Popen:http : //docs.python.org/2/library/subprocess.html#subprocess.Popen

import subprocess

command = "ntpq -p"  # the shell command
process = subprocess.Popen(command, stdout=subprocess.PIPE, stderr=None, shell=True)

#Launch the shell command:
output = process.communicate()

print output[0]

PopenコンストラクターでshellTrueの場合、コマンドをシーケンスではなく文字列として渡す必要があります。それ以外の場合は、コマンドをリストに分割するだけです。

command = ["ntpq", "-p"]  # the shell command
process = subprocess.Popen(command, stdout=subprocess.PIPE, stderr=None)

あなたはpopenの初期化に、また標準誤差を読み取るために必要がある場合は、設定することができます標準エラー出力をするsubprocess.PIPEかにsubprocess.STDOUT

import subprocess

command = "ntpq -p"  # the shell command
process = subprocess.Popen(command, stdout=subprocess.PIPE, stderr=subprocess.PIPE, shell=True)

#Launch the shell command:
output, error = process.communicate()

これは私が探していたものです
kRazzy R

14

Python 2.7以降の場合、慣用的な答えは subprocess.check_output()

サブプロセスを呼び出すときの引数の処理にも注意してください。少し混乱する可能性があります。...

argsが単一のコマンドで、独自のargsがない場合(またはユーザーがshell=True設定した場合)は、文字列にすることができます。それ以外の場合はリストでなければなりません。

たとえば... lsコマンドを呼び出すには、これで問題ありません。

from subprocess import check_call
check_call('ls')

これもそうです:

from subprocess import check_call
check_call(['ls',])

ただし、シェルコマンドに引数を渡したい場合は、次のことはできません。

from subprocess import check_call
check_call('ls -al')

代わりに、リストとして渡す必要があります。

from subprocess import check_call
check_call(['ls', '-al'])

このshlex.split()関数は、サブプロセスを作成する前に文字列をシェルのような構文に分割するのに役立つ場合があります...このように:

from subprocess import check_call
import shlex
check_call(shlex.split('ls -al'))

5年後、この質問はまだ多くの愛を得ています。2.7以上のアップデートをありがとう、コーリー!
Mark

11

これは私にとって完璧に機能します:

import subprocess
try:
    #prints results and merges stdout and std
    result = subprocess.check_output("echo %USERNAME%", stderr=subprocess.STDOUT, shell=True)
    print result
    #causes error and merges stdout and stderr
    result = subprocess.check_output("copy testfds", stderr=subprocess.STDOUT, shell=True)
except subprocess.CalledProcessError, ex: # error code <> 0 
    print "--------error------"
    print ex.cmd
    print ex.message
    print ex.returncode
    print ex.output # contains stdout and stderr together 

9

これは私にぴったりでした。タプルの戻りコード、stdoutおよびstderrを取得します。

from subprocess import Popen, PIPE

def console(cmd):
    p = Popen(cmd, shell=True, stdout=PIPE)
    out, err = p.communicate()
    return (p.returncode, out, err)

例えば:

result = console('ls -l')
print 'returncode: %s' % result[0]
print 'output: %s' % result[1]
print 'error: %s' % result[2]

6

受け入れられた答えは依然として良好であり、新しい機能についてのほんの一部のコメントです。Python 3.6以降ではcheck_output、エンコーディングをで直接処理できますドキュメントを参照してください。これは今すぐ文字列オブジェクトを返します:

import subprocess 
out = subprocess.check_output(["ls", "-l"], encoding="utf-8")

Python 3.7では、パラメーターcapture_outputがsubprocess.run()に追加されました。これにより、Popen / PIPE処理の一部が実行されます。pythonのドキュメントを参照してください。

import subprocess 
p2 = subprocess.run(["ls", "-l"], capture_output=True, encoding="utf-8")
p2.stdout

4
 import os   
 list = os.popen('pwd').read()

この場合、リストには1つの要素しかありません。


7
os.popensubprocessモジュールのために廃止予定です。
マイクグラハム

3
これは、Pythonの2.2.Xシリーズを使用している古いボックスの管理者にとって非常に役に立ちました。
Neil McF、2011

4

私はここに他の答えに基づいて小さな関数を書きました:

def pexec(*args):
    return subprocess.Popen(args, stdout=subprocess.PIPE).communicate()[0].rstrip()

使用法:

changeset = pexec('hg','id','--id')
branch = pexec('hg','id','--branch')
revnum = pexec('hg','id','--num')
print('%s : %s (%s)' % (revnum, changeset, branch))

4

Python 3.7では、に新しいキーワード引数capture_outputが導入されましたsubprocess.run。短くてシンプルなものを有効にする:

import subprocess

p = subprocess.run("echo 'hello world!'", capture_output=True, shell=True, encoding="utf8")
assert p.stdout == 'hello world!\n'

1
import subprocess
output = str(subprocess.Popen("ntpq -p",shell = True,stdout = subprocess.PIPE, 
stderr = subprocess.STDOUT).communicate()[0])

これは1行のソリューションです


0

以下は、プロセスのstdoutとstderrを単一の変数にキャプチャします。Python 2および3と互換性があります。

from subprocess import check_output, CalledProcessError, STDOUT

command = ["ls", "-l"]
try:
    output = check_output(command, stderr=STDOUT).decode()
    success = True 
except CalledProcessError as e:
    output = e.output.decode()
    success = False

コマンドが配列ではなく文字列である場合は、これに次の接頭辞を付けます。

import shlex
command = shlex.split(command)

0

Python 3.5の場合、私は以前の回答に基づいて関数を設定しました。ログが削除されている可能性があります。

import shlex
from subprocess import check_output, CalledProcessError, STDOUT


def cmdline(command):
    log("cmdline:{}".format(command))
    cmdArr = shlex.split(command)
    try:
        output = check_output(cmdArr,  stderr=STDOUT).decode()
        log("Success:{}".format(output))
    except (CalledProcessError) as e:
        output = e.output.decode()
        log("Fail:{}".format(output))
    except (Exception) as e:
        output = str(e);
        log("Fail:{}".format(e))
    return str(output)


def log(msg):
    msg = str(msg)
    d_date = datetime.datetime.now()
    now = str(d_date.strftime("%Y-%m-%d %H:%M:%S"))
    print(now + " " + msg)
    if ("LOG_FILE" in globals()):
        with open(LOG_FILE, "a") as myfile:
            myfile.write(now + " " + msg + "\n")

0

使用ckeck_output方法subprocess

import subprocess
address = 192.168.x.x
res = subprocess.check_output(['ping', address, '-c', '3'])

最後に文字列を解析します

for line in res.splitlines():

それが役に立てば幸いです、幸せなコーディング

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.