Paramikoを使用してSSHリターンコードを取得するにはどうすればよいですか?


97
client = paramiko.SSHClient()
stdin, stdout, stderr = client.exec_command(command)

コマンドの戻りコードを取得する方法はありますか?

すべてのstdout / stderrを解析して、コマンドが正常に終了したかどうかを知るのは困難です。

回答:


51

SSHClientは、Paramikoのより低レベルの機能の単純なラッパークラスです。APIドキュメントは、一覧表示されますrecv_exit_status() Channelクラスのメソッドを。

非常に簡単なデモスクリプト:

$ cat sshtest.py
import paramiko
import getpass

pw = getpass.getpass()

client = paramiko.SSHClient()
client.set_missing_host_key_policy(paramiko.WarningPolicy())
client.connect('127.0.0.1', password=pw)

while True:
    cmd = raw_input("Command to run: ")
    if cmd == "":
        break
    chan = client.get_transport().open_session()
    print "running '%s'" % cmd
    chan.exec_command(cmd)
    print "exit status: %s" % chan.recv_exit_status()

client.close()

$ python sshtest.py
Password: 
Command to run: true
running 'true'
exit status: 0
Command to run: false
running 'false'
exit status: 1
Command to run: 
$

ご回答ありがとうございます!プロジェクトで使用しました。どうもありがとうございました。これまでのところ、Pythonでのプログラミングでシステムコマンドsshをラップする代わりにparamikoを使用できます。
2010

SSHClientのソースを調べてまとめただけなので、最適ではない可能性があります。ところで:これがあなたの望むことをするなら、あなたは私の答えを「受け入れる」かもしれません。
JanC、

Link Dead Please Update
bhathiya-perera 2015

3
chan.recv_exit_status()-ハングアップして戻ってこないときの素敵なもの。素晴らしいアドバイス。
aaa90210 2015

9
これは悪い解決策です。以下の@apdastous 'の回答を参照してください。
Patrick

282

「より低いレベル」のチャネルクラスを直接呼び出すことを含まないはるかに簡単な例(つまり、コマンドを使用しないclient.get_transport().open_session()):

import paramiko

client = paramiko.SSHClient()
client.set_missing_host_key_policy(paramiko.AutoAddPolicy())
client.connect('blahblah.com')

stdin, stdout, stderr = client.exec_command("uptime")
print stdout.channel.recv_exit_status()    # status is 0

stdin, stdout, stderr = client.exec_command("oauwhduawhd")
print stdout.channel.recv_exit_status()    # status is 127

5
この例の良い点は、EXITだけでなく(質問のように)キャプチャすることですが、まだSTDOUTおよびSTDERRを取得できることを示しています。数年前、私はParamikoの貧弱なサンプルコードベース(無礼ではない)に惑わされ、EXITを取得するために、低レベルのtransport()呼び出しに頼りました。transport()は「1つを選択」するように強制しているようです(これは真実ではないかもしれませんが、例やチュートリアルドキュメントがないために私はそれを信じました)
Scott Prive

5

JanCに感謝し、サンプルにいくつかの変更を加えて、Python3でテストしました。

import paramiko
import getpass

pw = getpass.getpass()

client = paramiko.SSHClient()
client.set_missing_host_key_policy(paramiko.WarningPolicy())
#client.set_missing_host_key_policy(paramiko.AutoAddPolicy())

def start():
    try :
        client.connect('127.0.0.1', port=22, username='ubuntu', password=pw)
        return True
    except Exception as e:
        #client.close()
        print(e)
        return False

while start():
    key = True
    cmd = input("Command to run: ")
    if cmd == "":
        break
    chan = client.get_transport().open_session()
    print("running '%s'" % cmd)
    chan.exec_command(cmd)
    while key:
        if chan.recv_ready():
            print("recv:\n%s" % chan.recv(4096).decode('ascii'))
        if chan.recv_stderr_ready():
            print("error:\n%s" % chan.recv_stderr(4096).decode('ascii'))
        if chan.exit_status_ready():
            print("exit status: %s" % chan.recv_exit_status())
            key = False
            client.close()
client.close()

chan.recv_stderr_ready()でエラーメッセージが表示されますか?chan.recv_stderr(4096).decode( 'ascii')はメッセージテキストを返しますか?
kten

0

私の場合、出力バッファリングが問題でした。バッファリングのため、アプリケーションからの出力はノンブロッキングの方法で出力されません。バッファリングなしで出力を印刷する方法については、出力バッファリングを無効にするの答えを見つけることができます。簡単に言うと、次のように-uオプションを指定してpythonを実行します。

> python -u script.py

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