web-dev-qa-db-ja.com

Paramikoを使用してPythonでsshを介した対話型シェルを実装しますか?

Sshを介してリモートシェルで複数のコマンドを実行するプログラム(Windows 7ではPython 3.x))を作成します。paramikos 'exec_command() functionコマンドは環境変数(前のコマンドで設定)に依存し、1つのexec_command()に連結できないため、(コマンドの実行後にチャネルが閉じられるため)ユースケースに適さないことに気付きましたプログラムで異なる時間に実行されるように呼び出します。

したがって、同じチャネルでコマンドを実行したいと思います。私が検討した次のオプションは、paramikosのinvoke_Shell()関数を使用して対話型シェルを実装することでした。

ssh = paramiko.SSHClient()
ssh.set_missing_Host_key_policy(paramiko.AutoAddPolicy())
ssh.connect(Host, username=user, password=psw, port=22)

channel = ssh.invoke_Shell()

out = channel.recv(9999)

channel.send('cd mivne_final\n')
channel.send('ls\n')

while not channel.recv_ready():
    time.sleep(3)

out = channel.recv(9999)
print(out.decode("ascii"))

channel.send('cd ..\n')
channel.send('cd or_fail\n')
channel.send('ls\n')

while not channel.recv_ready():
    time.sleep(3)

out = channel.recv(9999)
print(out.decode("ascii"))

channel.send('cd ..\n')
channel.send('cd simulator\n')
channel.send('ls\n')

while not channel.recv_ready():
    time.sleep(3)

out = channel.recv(9999)
print(out.decode("ascii"))

ssh.close() 

このコードにはいくつかの問題があります。

  1. 最初のprintは、常にls出力を印刷するとは限りません(2番目のprintにのみ印刷される場合もあります)。
  2. 最初のcdおよびlsコマンドは常に出力に存在します(出力の一部としてrecvコマンドを介して取得します)が、後続のcdおよびlsコマンドは時々印刷され、時々印刷されません。
  3. 2番目と3番目のcdおよびlsコマンド(印刷時)は、常に最初のls出力の前に表示されます。

私はこの「非決定論」と混同しており、あなたの助けにとても感謝しています。

14
misha
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()

    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)

        return shin, shout, sherr
18
misha

基本的にすべてのコードを使用し、forループを追加しています。

commands = ["ls","command2","command3"]
conn_one = ShellHandler(Host,name,pwd)
for command in commands:
      conn_one.execute(command)

2つのコマンドを正しい出力で実行しますが、その後はただそこに座っています。コードのどこかでdelを呼び出す必要があるかどうか疑問に思っています。

0
magicsword