如何使用Paramiko从远程机器上执行的命令返回stderr?

时间:2017-01-08 20:19:19

标签: python python-2.7 unix paramiko

请检查以下代码(我认为这是不言自明的)。

代码:

import sys
import paramiko

def execute_command_on_remote_machine(ip_addr, command):
    try:
        client = paramiko.SSHClient()
        client.set_missing_host_key_policy(paramiko.AutoAddPolicy())

        client.connect(str(ip_addr), username='root', password='****')
        chan = client.get_transport().open_session()
        #chan.get_pty()

        stdin, stderr, stdout = [], [], []
        stdin, stdout, stderr = client.exec_command(command, get_pty=True)

        err_list = [line for line in stderr.read().splitlines()]
        out_list = [line for line in stdout.read().splitlines()]

        client.close()
        return out_list, err_list
    except Exception, e:
        print str(e)
        sys.exit(1)


output, error = execute_command_on_remote_machine("*.*.*.*", "dinesh")

print "Output is - ",output
print "Error is - ",error

输出

D:\dsp_jetbrains\AWS_Persistence>python chk.py
Output is -  ['bash: dinesh: command not found']
Error is -  []

问题:

作为输入,我传递了错误的命令,我期待" dinesh:command not found"将打印为"错误是"。然而,它正在出现"输出是"。

问题:

exec_commandparamiko的输出和错误如何运作?

其他测试用例:

我也尝试使用命令,但是输入了错误的命令。示例 - lsof -f dinesh

但结果是一样的。

D:\dsp_jetbrains\AWS_Persistence>python chk.py
Output is -  ['lsof: unknown file struct option: d', 'lsof: unknown file struct
option: i', 'lsof: unknown file struct option: n', 'lsof: unknown file struct op
tion: e', 'lsof: unknown file struct option: s', 'lsof: unknown file struct opti
on: h', 'lsof 4.87', ' latest revision: ftp://lsof.itap.purdue.edu/pub/tools/uni
x/lsof/', ' latest FAQ: ftp://lsof.itap.purdue.edu/pub/tools/unix/lsof/FAQ', ' l
atest man page: ftp://lsof.itap.purdue.edu/pub/tools/unix/lsof/lsof_man', ' usag
e: [-?abhKlnNoOPRtUvVX] [+|-c c] [+|-d s] [+D D] [+|-f[gG]] [+|-e s]', ' [-F [f]
] [-g [s]] [-i [i]] [+|-L [l]] [+m [m]] [+|-M] [-o [o]] [-p s]', '[+|-r [t]] [-s
 [p:s]] [-S [t]] [-T [t]] [-u s] [+|-w] [-x [fl]] [-Z [Z]] [--] [names]', "Use t
he ``-h'' option to get more help information."]
Error is -  []

1 个答案:

答案 0 :(得分:1)

您可能不应该使用get_pty,除非您知道自己需要它。考虑删除:

chan.get_pty()

并改变:

stdin, stdout, stderr = client.exec_command(command, get_pty=True)

为:

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

来自DOCS

  

从服务器请求伪终端。这通常在创建客户端通道后立即使用,要求服务器为使用invoke_shell调用的shell提供一些基本的终端语义。如果您要使用exec_command执行单个命令,则不必(或不希望)调用此方法。

如果您希望ssh连接看起来像用户已连接,则使用pseudo-terminal。需要这个是不常见的。一个人为的例子就是如果你想通过发送和接收ansi-terminal序列在远端运行VIM。

如果没有get_pty,远程执行的命令将只是连接一个stdin / stdout / stderr trio,就像它在piepline中一样。

get_pty的罕见性很可能是为什么它不是默认值。