Python 如何使用paramiko保持ssh会话不过期?

Python 如何使用paramiko保持ssh会话不过期?,python,python-2.7,ssh,paramiko,switching,Python,Python 2.7,Ssh,Paramiko,Switching,我打算使用paramiko在远程主机上运行几个命令,但ssh会话在运行命令后关闭。 代码如下: from paramiko import SSHClient import paramiko ssh = SSHClient() ssh.load_system_host_keys() ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy()) ssh.connect(host, 22, user, passwd, timeout=3)

我打算使用paramiko在远程主机上运行几个命令,但ssh会话在运行命令后关闭。
代码如下:

from paramiko import SSHClient  
import paramiko  
ssh = SSHClient()
ssh.load_system_host_keys()
ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())
ssh.connect(host, 22, user, passwd, timeout=3)
stdin, stdout, stderr = ssh.exec_command('uname -a')
那么有没有办法阻止ssh会话关闭?帕拉米科还有其他选择吗

更新

当连接到Linux服务器时,我可以在Macbook上继续调用
exec\u命令
,但当连接到交换机时,在Linux服务器上执行
exec\u命令
一次后,ssh会话会自动关闭,并引发
SSHException:paramiko.ssh\u异常。SSHException:ssh会话未激活

>>> print ssh.get_transport()  
>>> <paramiko.Transport at 0xf00216d0L (unconnected)>  
>>> print ssh.get_transport().is_active()  
>>> False  
>>> print ssh.get_transport().is_authenticated()  
>>> False
打印ssh.get_transport() >>> >>>print ssh.get_transport()是否处于活动状态() >>>假的 >>>print ssh.get_transport()是否经过身份验证() >>>假的 有没有办法让paramiko ssh会话始终处于活动状态

返回的paramiko调试模式信息如下:

启动线程(客户端模式):0x2657e10L
已连接(版本1.99,客户端Comware-5.20)
kex算法:[u'diffie-hellman-group-exchange-sha1',u'diffie-hellman-group14-sha1',u'diffie-hellman-group1-sha1']服务器密钥:[u'ssh-rsa']客户端加密:[u'aes128-cbc',u'3des-cbc',u'des-cbc']服务器加密:[u'aes128-cbc',u'3des-cbc',u'des cbc'][u'hmac-sha1',u'hmac-sha1-96',u'hmac-md5',u'hmac-md5-96']客户端压缩:[u'none']服务器压缩:[u'none']客户端语言:[u']服务器语言:[u']kex后面是?False
同意的密码:本地=aes128 cbc,远程=aes128 cbc
使用kex diffie-hellman-group14-sha1;服务器密钥类型ssh rsa;密码:本地aes128 cbc,远程aes128 cbc;mac:本地hmac-sha1,远程hmac-sha1;压缩:本地无,远程无
切换到新键…
userauth正常
身份验证(密码)成功!
[chan 0]最大数据包输入:32768字节
[chan 1]最大数据包输入:32768字节
[chan 0]最大数据包输出:32496字节
第0频道已打开。
Secsh通道2打开失败:
资源短缺:资源短缺
[chan 0]Sesch通道0请求正常
[chan 0]已发送EOF(0)


我看到您在connect调用中使用了
timeout
参数:

ssh.connect(host, 22, user, passwd, timeout=3)
从文件中:

超时(浮点)–TCP连接的可选超时(以秒为单位)

在我的一个脚本中,我只需执行以下操作:

ssh = paramiko.SSHClient()
ssh.connect(host, username=settings.user)
在我打电话之前保持连接畅通

ssh.close()

您可以使用paramiko实现交互式shell,这样在远程shell上执行命令后,通道不会关闭

import paramiko
import re


class ShellHandler:

    def __init__(self, host, user, psw):
        self.ssh = paramiko.SSHClient()
        self.ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())
        self.ssh.connect(host, username=user, password=psw, port=22)

        channel = self.ssh.invoke_shell()
        self.stdin = channel.makefile('wb')
        self.stdout = channel.makefile('r')

    def __del__(self):
        self.ssh.close()

    @staticmethod
    def _print_exec_out(cmd, out_buf, err_buf, exit_status):
        print('command executed: {}'.format(cmd))
        print('STDOUT:')
        for line in out_buf:
            print(line, end="")
        print('end of STDOUT')
        print('STDERR:')
        for line in err_buf:
            print(line, end="")
        print('end of STDERR')
        print('finished with exit status: {}'.format(exit_status))
        print('------------------------------------')
        pass

    def execute(self, cmd):
        """

        :param cmd: the command to be executed on the remote computer
        :examples:  execute('ls')
                    execute('finger')
                    execute('cd folder_name')
        """
        cmd = cmd.strip('\n')
        self.stdin.write(cmd + '\n')
        finish = 'end of stdOUT buffer. finished with exit status'
        echo_cmd = 'echo {} $?'.format(finish)
        self.stdin.write(echo_cmd + '\n')
        shin = self.stdin
        self.stdin.flush()

        shout = []
        sherr = []
        exit_status = 0
        for line in self.stdout:
            if str(line).startswith(cmd) or str(line).startswith(echo_cmd):
                # up for now filled with shell junk from stdin
                shout = []
            elif str(line).startswith(finish):
                # our finish command ends with the exit status
                exit_status = int(str(line).rsplit(maxsplit=1)[1])
                if exit_status:
                    # stderr is combined with stdout.
                    # thus, swap sherr with shout in a case of failure.
                    sherr = shout
                    shout = []
                break
            else:
                # get rid of 'coloring and formatting' special characters
                shout.append(re.compile(r'(\x9B|\x1B\[)[0-?]*[ -/]*[@-~]').sub('', line).
                             replace('\b', '').replace('\r', ''))

        # first and last lines of shout/sherr contain a prompt
        if shout and echo_cmd in shout[-1]:
            shout.pop()
        if shout and cmd in shout[0]:
            shout.pop(0)
        if sherr and echo_cmd in sherr[-1]:
            sherr.pop()
        if sherr and cmd in sherr[0]:
            sherr.pop(0)

        self._print_exec_out(cmd=cmd, out_buf=shout, err_buf=sherr, exit_status=exit_status)
        return shin, shout, sherr

这个脚本还有更多吗?你应该能够继续调用<代码> Excel命令>代码> hello。你在脚本中运行多个命令或者运行一个命令的多个脚本?我可以在MacBook上保持调用<代码> Excel命令>代码,但是它在Linux服务器上不起作用,引发了<代码> sHeExp:PAR。amiko.ssh_exception.SSHException:ssh会话未激活Mac上的Python版本为2.7.11,但Linux服务器上的Python版本为2.6.6。@tdelaney非常奇怪。这是一个特别糟糕的开关吗?!您可以启用日志记录(请参阅)在调试级别获取更多详细信息。从Linux服务器连接到交换机时的结果相同,但从另一台服务器连接到Linux服务器时工作正常。无论是否添加
timeout
参数,似乎都没有什么不同。@LarsVegashave您看到过这个()?它确实解决了非活动ssh会话的问题,但当我并行运行它时,在发送第一个命令后,通道一直挂在那里。@lars