Paramiko exec_command с несколькими командами на маршрутизаторе Cisco, не дающими никаких выходных данных

login_user = 'xyz'
login_pass = 'xyz'
ssh = paramiko.SSHClient()
ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())
ssh.connect(device, username=login_user, password=login_pass, look_for_keys=False, timeout=5)
print "success loggedin"
stdin, stdout, stderr = ssh.exec_command("term len 0 ; show int desc | i Tu ; show ip interface brief | in Tunnel ; show ip bgp vpnv4 vrf AWS summary | i 169.25 ; show ip route vrf AWS bgp")
all_output = stdout.read()
print all_output

Выше приведен мой фрагмент кода, при передаче только одной команды результаты печатаются очень хорошо, однако с несколькими командами, как указано выше, он не работает (нет вывода). Устройство, на которое выполняется вход, — это Cisco ASR1006. Это как-то связано с устройством?

Любая помощь высоко ценится!


Поскольку несколько команд не сработали, я использую приведенный ниже код с несколькими exec_command, но это занимает около 30-40 секунд на каждое выполнение. Требуется ssh.connect для выполнения каждой команды?

ssh.connect(device, username=login_user, password=login_pass, look_for_keys=False)
stdin, stdout, stderr = ssh.exec_command('show int desc | i Tu')
vpc_ints = stdout.read()
ssh.connect(device, username=login_user, password=login_pass, look_for_keys=False)
stdin, stdout, stderr = ssh.exec_command('show ip interface brief | in Tunnel')
vpc_peers = stdout.read()
ssh.connect(device, username=login_user, password=login_pass, look_for_keys=False)
stdin, stdout, stderr = ssh.exec_command('show ip bgp vpnv4 vrf AWS summary | i 169.25')
vpc_bgp_routes = stdout.read()
ssh.connect(device, username=login_user, password=login_pass, look_for_keys=False)
stdin, stdout, stderr = ssh.exec_command('show ip route vrf AWS bgp')
ip_routes_list = stdout.read()

Как было предложено в комментариях, я также пытался вызывать exec_command несколько раз по одному соединению:

ssh.connect(device, username=login_user, password=login_pass, look_for_keys=False)
stdin, stdout, stderr = ssh.exec_command('show int desc | i Tu')
vpc_ints = stdout.read()
#ssh.connect(device, username=login_user, password=login_pass, look_for_keys=False)
stdin, stdout, stderr = ssh.exec_command('show ip interface brief | in Tunnel')
vpc_peers = stdout.read()
#ssh.connect(device, username=login_user, password=login_pass, look_for_keys=False)
stdin, stdout, stderr = ssh.exec_command('show ip bgp vpnv4 vrf AWS summary | i 169.25')
vpc_bgp_routes = stdout.read()
#ssh.connect(device, username=login_user, password=login_pass, look_for_keys=False)
stdin, stdout, stderr = ssh.exec_command('show ip route vrf AWS bgp')
ip_routes_list = stdout.read()
ssh.close()

Но это не удается с:

File "C:\Python27\myvpndashboard\myvpnapp\tunnel_summary.py", line 179, in dataset_build
cisco_show(device)
File "C:\Python27\myvpndashboard\myvpnapp\tunnel_summary.py", line 36, in cisco_show
stdin, stdout, stderr = ssh.exec_command('show ip interface brief | in Tunnel')
File "C:\Users\sduraisami\Envs\myvpndashboard\lib\site-packages\paramiko\client.py", line 472, in exec_command
chan = self._transport.open_session(timeout=timeout)
File "C:\Users\sduraisami\Envs\myvpndashboard\lib\site-packages\paramiko\transport.py", line 765, in open_session
timeout=timeout)
File "C:\Users\sduraisami\Envs\myvpndashboard\lib\site-packages\paramiko\transport.py", line 889, in open_channel
raise e
EOFError

person Sivasubramanian Duraisami    schedule 12.01.2018    source источник


Ответы (1)


Известно, что (по крайней мере, некоторые) маршрутизаторы Cisco не поддерживают несколько команд в одном запросе «exec»: https://the.earth.li/~sgtatham/putty/latest/htmldoc/Chapter3.html#using-cmdline-m

На некоторых серверах (в частности, системах Unix) вы даже можете поместить несколько строк в этот файл и выполнить несколько команд последовательно или целый сценарий оболочки; но, возможно, это злоупотребление, и нельзя ожидать, что оно будет работать на всех серверах. В частности, известно, что он не работает с некоторыми «встроенными» серверами, такими как маршрутизаторы Cisco.


Обычно вы можете запускать exec_command несколько раз по одному и тому же соединению. Но с Cisco это тоже не представляется возможным.

Затем я бы рекомендовал вам придерживаться повторного открытия сеанса для каждой команды. Хотя это неэффективно, это надежный подход.


Если неэффективность неприемлема, вы можете использовать канал оболочки, например:

channel = ssh.invoke_shell()
channel.send('command 1\n')
channel.send('command 2\n')
channel.send('command 3\n')
while not channel.recv_ready():
    time.sleep(3)
out = channel.recv(9999)
print out

Но это не надежно. Трудно сказать, когда заканчивается вывод одной команды и начинается другая. Также трудно сказать, получили ли вы уже весь вывод или нет. А поскольку Paramiko всегда использует PTY, вы получаете много нежелательного вывода, например подсказки, эхо-команды и т. д.

person Martin Prikryl    schedule 17.01.2018