Внедрить интерактивную оболочку поверх ssh в Python с помощью Paramiko?

Я хочу написать программу (в Python 3.x в Windows 7), которая выполняет несколько команд на удаленной оболочке через ssh. Посмотрев на функцию exec_command () paramikos, я понял, что он не подходит для моего использования (потому что канал закрывается после выполнения команды), так как команды зависят от переменных среды (заданных предварительными командами) и не могут быть объединены в один вызов exec_command (), поскольку они должны выполняться в разное время в программе.

Таким образом, я хочу выполнять команды в одном канале. Следующим вариантом, который я рассмотрел, была реализация интерактивной оболочки с использованием функции invoke_shell () paramikos:

  • Как выполнить сценарий удаленно в python с помощью ssh?
  • Запуск команд оболочки Windows с помощью python
  • Внедрение оболочки Python внутри программы Python
  • входные данные против raw_input: приложение Interactive Shell Python?
  • Python Fabric: Как обрабатывать произвольное приглашение командной строки для ввода?
  • Как использовать python для запуска интерактивной программы оболочки?
  • 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. Первая «печать» не всегда печатает вывод «ls» (иногда он печатается только во второй «печати»).
    2. Первые команды «cd» и «ls» всегда присутствуют в выходе (я получаю их через команду «recv», как часть вывода), в то время как все следующие команды «cd» и «ls» печатаются иногда, и иногда это не так.
    3. Вторая и третья команды «cd» и «ls» (при печати) всегда появляются перед первым выходом «ls».

    Я смущен этим «не-детерминизмом» и очень ценю вашу помощь.

  • Не удается выполнить сценарий оболочки из подпроцесса python: разрешено отклонение
  • Python unicode popen или ошибка чтения Popen unicode
  • Могу ли я использовать Python в качестве замены Bash?
  • Как я могу проверить, запущен ли скрипт Python в интерактивном режиме?
  • Как перейти к REPL (Read, Eval, Print, Loop) из кода Python
  • Изменить рабочий каталог в оболочке с помощью скрипта python
  • 2 Solutions collect form web for “Внедрить интерактивную оболочку поверх ssh в Python с помощью 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() 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 

    Я в основном использую весь ваш код и просто добавляю цикл for:

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

    Он выполняет 2 команды с правильным выходом, но затем он просто сидит там. Мне интересно, нужно ли мне звонить дель где-то в коде.

    Python - лучший язык программирования в мире.