gpt4 book ai didi

python - 使用 Paramiko 在 Python 中通过 ssh 实现交互式 shell?

转载 作者:塔克拉玛干 更新时间:2023-11-03 01:24:48 25 4
gpt4 key购买 nike

我想编写一个程序(在 Windows 7 上的 Python 3.x 中)通过 ssh 在远程 shell 上执行多个命令。在查看 paramikos 的 exec_command() 函数后,我意识到它不适合我的用例(因为 channel 在命令执行后关闭),因为命令取决于环境变量(由先前设置命令)并且不能串联成一个 exec_command() 调用,因为它们将在程序的不同时间执行。

因此,我想在同一个 channel 中执行命令。我研究的下一个选项是使用 paramikos 的 invoke_shell() 函数实现交互式 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 输出(有时它只在第二个 print 上打印)。
  2. 第一个 cdls 命令总是出现在输出中(我通过 recv 命令获取它们,作为输出的一部分),而以下所有 cdls 命令有时会打印,有时不会。
  3. 第二个和第三个 cdls 命令(打印时)总是出现在第一个 ls 输出之前。

我对这种“非确定性”感到困惑,非常感谢您的帮助。

最佳答案

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

关于python - 使用 Paramiko 在 Python 中通过 ssh 实现交互式 shell?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37115365/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com