Paramiko捕获命令输出

时间:2016-10-17 19:46:47

标签: python python-2.7 ssh paramiko

我有一个让我头疼几天的问题。我在Python 2.7.10中使用Paramiko模块,我想向Brocade路由器发出多个命令,但只返回给定命令之一的输出,如下所示:

#!/usr/bin/env python
import paramiko, time

router = 'r1.test.example.com'
password = 'password'
username = 'testuser'

ssh = paramiko.SSHClient()
ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())
ssh.connect(router, username=username, password=password)
print('Successfully connected to %s' % router)

remote_conn = ssh.invoke_shell()
output = remote_conn.recv(1000)

# Disable paging on Brocade.
remote_conn.send('terminal length 0\n')
# Check interface status.
remote_conn.send('show interfaces ethernet 0/1\n') # I only want output from this command.
time.sleep(2)
output = remote_conn.recv(5000)
print(output)

如果我要打印完整输出,它将包含发给路由器的所有内容,但我只想查看show interfaces ethernet 0/1\n命令的输出。

任何人都可以帮忙解决这个问题吗?

我想问的最后一件事。我想过滤output变量并检查字符串的出现,如“向上”或“向下”,但我似乎无法让它工作,因为输出中的所有内容都显示在新行上?

例如:

如果我在for循环中迭代output变量,我会得到变量中的所有字符,如下所示:

for line in output:
    print(line)

我得到这样的输出:

ë

[R

名词

1

1

ë

名词

ħ

0

有什么方法吗?

再次

提前感谢您的帮助。

致以最诚挚的问候,

Aaron C。

3 个答案:

答案 0 :(得分:2)

对于你的第二个问题:虽然我不是paramiko的专家,但我看到函数recv according to the doc返回一个字符串。如果对字符串应用 for 循环,则会获得字符(而不是人们可能期望的行)。换行是由您使用打印功能引起的,如on this page, at paragraph 6.3所述。

我还没有研究过paramiko建议做什么。但为什么不把完整的字符串视为一个单一的实体呢?例如,您可以检查" up"为:

if "up" in output:

或者,如果这更适合您的需求,您可以split the string into lines然后做您想做的任何测试:

for line in output.split('\n'): 

答案 1 :(得分:2)

在阅读完所有评论后,我做了以下更改:

#!/usr/bin/env python
import paramiko, time

router = 'r2.test.example.com'
password = 'password'
username = 'testuser'

ssh = paramiko.SSHClient()
ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())
ssh.connect(router, username=username, password=password)
print('Successfully connected to %s' % router)

remote_conn = ssh.invoke_shell()
output = remote_conn.recv(1000)

# Disable paging on Brocade.
remote_conn.send('terminal length 0\n')
time.sleep(2)
# Clearing output.
if remote_conn.recv_ready():
    output = remote_conn.recv(1000)

# Check interface status.
remote_conn.send('show interfaces ethernet 4/1\n') # I only want output from this command.
time.sleep(2)
# Getting output I want.
if remote_conn.recv_ready():
    output = remote_conn.recv(5000)
print(output)

# Test: Check if interface is up.
for line in output.split('\n'):
    if 'line protocol is up' in line:
        print(line)

现在一切都很好。

感谢您的帮助。

致以最诚挚的问候,

Aaron C。

答案 2 :(得分:0)

如果可以,exec_command()调用提供了一种更简单的机制来调用命令。我看到Cisco交换机突然断开尝试exec_command()的连接,因此可能无法与Brocade设备一起使用。

如果您必须转到invoke_shell()路线,请确保在连接后和send('terminal length 0\n')之后清除所有待处理的输出,在致电recv_ready()之前检查recv()以避免阻止阅读可能永远不会到达的数据。由于您正在控制交互式shell,因此可能需要sleep()次调用以允许服务器有足够的时间来处理和发送数据,或者可能需要通过识别shell来轮询输出字符串以确认您的上一个命令已完成提示字符串。