Jak uzyskać kod zwrotny SSH używając Paramiko?

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

Czy Jest jakiś sposób na uzyskanie kodu zwrotnego polecenia?

Trudno jest przetworzyć wszystkie stdout/stderr i dowiedzieć się, czy polecenie zakończyło się pomyślnie, czy nie.

Author: Charlie, 2010-08-25

4 answers

SSHClient jest prostą klasą owijającą wokół bardziej niższego poziomu funkcjonalności w Paramiko. Dokumentacja API zawiera listę metody recv_exit_status () w klasie kanału.

Bardzo prosty skrypt demonstracyjny:

$ 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: 
$
 43
Author: JanC,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2016-06-01 13:22:36

Znacznie łatwiejszy przykład, który nie wymaga bezpośredniego wywoływania klasy kanału:

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
 217
Author: apdastous,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2013-01-31 17:14:35

Dzięki za JanC, dodałem trochę modyfikacji dla przykładu i przetestowałem w Python3, to naprawdę przydatne dla mnie.

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()
 5
Author: perillaseed,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2013-10-13 12:43:03

W moim przypadku problemem było buforowanie wyjścia. Z powodu buforowania wyjścia z aplikacji nie wychodzą w sposób nieblokujący. Odpowiedź na pytanie Jak wydrukować wyjście bez buforowania znajdziesz tutaj: Wyłącz buforowanie wyjścia . W skrócie, po prostu uruchom Pythona z opcją-u w ten sposób:

> python -u script.py

 0
Author: Youngmin Kim,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2018-05-22 06:12:08