web-dev-qa-db-ja.com

paramikoを使用してsshセッションが期限切れにならないようにするにはどうすればよいですか?

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セッションが閉じるのを止める方法はありますか?またはparamikoの代替手段はありますか?

更新
Linuxサーバーに接続しているときにMacbookでexec_commandを呼び出し続けることができましたが、Linuxサーバーに接続するとexec_commandの後にsshセッションが自動的に閉じましたスイッチそして上げた
SSHException: paramiko.ssh_exception.SSHException: SSH session not active

>>> print ssh.get_transport()  
>>> <paramiko.Transport at 0xf00216d0L (unconnected)>  
>>> print ssh.get_transport().is_active()  
>>> False  
>>> print ssh.get_transport().is_authenticated()  
>>> False

Paramiko sshセッションを常にアクティブに保つ方法はありますか?

Paramikoデバッグモード情報は次のように返されます。

開始スレッド(クライアントモード):0x2657e10L
接続済み(バージョン1.99、クライアントComware-5.20)
kex algos:[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 ']クライアントmac:[u'hmac-sha1'、u'hmac-sha1-96 '、u'hmac-md5'、u'hmac-md5-96 ']サーバーmac:[u'hmac-sha1 '、u'hmac-sha1-96'、u'hmac-md5 '、u'hmac-md5-96'] client compress:[u'none '] server compress:[u'none'] client lang:[u ''] server lang:[u ''] kexが続きますか?False
暗号が同意しました:local = aes128-cbc、remote = aes128-cbc
kexdiffie-hellman-group14-sha1を使用;サーバーキータイプssh-rsa;暗号:ローカルaes128-cbc、リモートaes128-cbc; mac:ローカルhmac-sha1、リモートhmac-sha1;圧縮:ローカルなし、リモートなし
新しいキーに切り替えます.。
userauthはOKです
認証(パスワード)に成功しました!
[chan 0]最大パケット数:32768バイト
[chan 1]最大パケット数:32768バイト
[chan 0]最大パケット出力:32496バイト
Secshチャネル0が開かれました。
Secshチャネル2のオープンに失敗しました:
リソース不足:リソース不足
[chan 0] Seschチャンネル0リクエストOK
[chan 0] EOF送信済み(0)

9
J.Wang

Paramikoを使用してインタラクティブシェルを実装できます。これにより、リモートシェルでコマンドが実行された後にチャネルが閉じられなくなります。

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
4
misha

接続呼び出しでtimeoutパラメータを使用しているようです。

ssh.connect(Host, 22, user, passwd, timeout=3)

ドキュメントから:

タイムアウト(フロート)– TCP接続のオプションのタイムアウト(秒単位)

私のスクリプトの1つでは、次のようにします。

ssh = paramiko.SSHClient()
ssh.connect(Host, username=settings.user)

電話をかけるまで接続を開いたままにします

ssh.close()
2
LarsVegas