Python实现ssh批量登录并执行命令


Posted in Python onOctober 25, 2016

局域网内有一百多台电脑,全部都是linux操作系统,所有电脑配置相同,系统完全相同(包括用户名和密码),ip地址是自动分配的。现在有个任务是在这些电脑上执行某些命令,者说进行某些操作,比如安装某些软件,拷贝某些文件,批量关机等。如果一台一台得手工去操作,费时又费力,如果要进行多个操作就更麻烦啦。

或许你会想到网络同传,网络同传是什么?就是在一台电脑上把电脑装好,配置好,然后利用某些软件,如“联想网络同传”把系统原样拷贝过去,在装系统时很有用,只要在一台电脑上装好,同传以后所有的电脑都装好操作系统了,很方便。同传要求所有电脑硬件完全相同,在联想的电脑上装的系统传到方正电脑上肯定会出问题的。传系统也是很费时间的,根据硬盘大小,如果30G硬盘,100多台电脑大约要传2个多小时,反正比一台一台地安装快!但是如果系统都传完了,发现忘了装一个软件,或者还需要做些小修改,再同传一次可以,但是太慢,传两次半天时间就没了。这时候我们可以利用ssh去控制每台电脑去执行某些命令。

先让我们回忆一下ssh远程登录的过程:首先执行命令 ssh username@192.168.1.x ,第一次登录的时候系统会提示我们是否要继续连接,我们要输入“yes”,然后等一段时间后系统提示我们输入密码,正确地输入密码之后我们就能登录到远程计算机,然后我们就能执行命令了。我们注意到这里面有两次人机交互,一次是输入‘yes',另一次是输入密码。就是因为有两次交互我们不能简单的用某些命令去完成我们的任务。我们可以考虑把人机交互变成自动交互,python的pexpect模块可以帮我们实现自动交互。下面这段代码是用pexpect实现自动交互登录并执行命令的函数:

#!/usr/bin/env python 
# -*- coding: utf-8 -*- 
 
import pexpect 
 
def ssh_cmd(ip, passwd, cmd): 
  ret = -1 
  ssh = pexpect.spawn('ssh root@%s "%s"' % (ip, cmd)) 
  try: 
    i = ssh.expect(['password:', 'continue connecting (yes/no)?'], timeout=5) 
    if i == 0 : 
      ssh.sendline(passwd) 
    elif i == 1: 
      ssh.sendline('yes\n') 
      ssh.expect('password: ') 
      ssh.sendline(passwd) 
    ssh.sendline(cmd) 
    r = ssh.read() 
    print r 
    ret = 0 
  except pexpect.EOF: 
    print "EOF" 
    ssh.close() 
    ret = -1 
  except pexpect.TIMEOUT: 
    print "TIMEOUT" 
    ssh.close() 
    ret = -2  
  return ret

利用pexpect模块我们可以做很多事情,由于他提供了自动交互功能,因此我们可以实现ftp,telnet,ssh,scp等的自动登录,还是比较实用的。根据上面的代码相信读者已经知道怎么实现了(python就是那么简单!)。

用上面的代码去完成任务还是比较费时间的,因为程序要等待自动交互出现,另外ubuntu用ssh连接就是比较慢,要进行一系列的验证,这样才体现出ssh的安全。我们要提高效率,在最短的时间内完成。后来我发现了python里面的paramiko模块,用这个实现ssh登录更加简单。看下面的代码:

#-*- coding: utf-8 -*- 
#!/usr/bin/python  
import paramiko 
import threading 
def ssh2(ip,username,passwd,cmd): 
  try: 
    ssh = paramiko.SSHClient() 
    ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy()) 
    ssh.connect(ip,22,username,passwd,timeout=5) 
    for m in cmd: 
      stdin, stdout, stderr = ssh.exec_command(m) 
#      stdin.write("Y")  #简单交互,输入 ‘Y'  
      out = stdout.readlines() 
      #屏幕输出 
      for o in out: 
        print o, 
    print '%s\tOK\n'%(ip) 
    ssh.close() 
  except : 
    print '%s\tError\n'%(ip) 
if __name__=='__main__': 
  cmd = ['cal','echo hello!']#你要执行的命令列表 
  username = "" #用户名 
  passwd = ""  #密码 
  threads = []  #多线程 
  print "Begin......" 
  for i in range(1,254): 
    ip = '192.168.1.'+str(i) 
    a=threading.Thread(target=ssh2,args=(ip,username,passwd,cmd))  
    a.start()

上面的程序还是有些技巧的:

  • 利用多线程,同时发出登录请求,同时去连接电脑,这样速度快很多,我试了一下,如果不用多线程,直接一个一个挨着执行的话,大约5~10秒钟才能对一台电脑操作完,具体时间要根据命令的来决定,如果是软件安装或者卸载时间要更长一些。这样下来怎么也要一二十分钟,用多线程后就快多了,所有的命令执行完用了不到2分钟!
  • 最好用root用户登录,因为安装或者卸载软件的时候如果用普通用户又会提示输入密码,这样又多了一次交互,处理起来就比较麻烦!安装软件时apt-get install xxx 最好加上“-y”参数,因为有时安装或删除软件时提示是否继续安装或卸载,这又是一次自动交互!加上那个参数后就没有人机交互了。
  • 循环时循环所有ip,因为计算机的ip是路由器自动分配的,保险起见,最好全部都执行,保证没有遗漏的主机
  • 远端执行命令时如果有交互,可以这样用 stdin.write("Y")来完成交互,“Y”就是输入“Y”。
  • 把所有的命令放到一个列表里面,遍历列表可以依次执行列表里面的命令
  • 为了更好的进行控制,最好在电脑上提前把root用户打开,装好ssh服务器并让其开机自动执行。

希望本文所述对你有所帮助,Python实现ssh批量登录并执行命令内容就给大家介绍到这里了。希望大家继续关注我们的网站!想要学习Python可以继续关注本站。

Python 相关文章推荐
Python装饰器使用示例及实际应用例子
Mar 06 Python
python字符串编码识别模块chardet简单应用
Jun 15 Python
python实现将html表格转换成CSV文件的方法
Jun 28 Python
Python实现将sqlite数据库导出转成Excel(xls)表的方法
Jul 17 Python
python中pip的使用和修改下载源的方法
Jul 08 Python
对Python函数设计规范详解
Jul 19 Python
Python代码实现http/https代理服务器的脚本
Aug 12 Python
Python根据服务获取端口号的方法
Sep 25 Python
python 怎样将dataframe中的字符串日期转化为日期的方法
Sep 26 Python
Python定义一个Actor任务
Jul 29 Python
详解python中的异常捕获
Dec 15 Python
python实现简单的井字棋游戏(gui界面)
Jan 22 Python
详解Python的Lambda函数与排序
Oct 25 #Python
Python脚本实现Web漏洞扫描工具
Oct 25 #Python
python+django快速实现文件上传
Oct 24 #Python
python实现简单爬虫功能的示例
Oct 24 #Python
简单谈谈Python中的反转字符串问题
Oct 24 #Python
Python 内置函数complex详解
Oct 23 #Python
Python检测生僻字的实现方法
Oct 23 #Python
You might like
PHP、Python和Javascript的装饰器模式对比
2015/02/03 PHP
php实现兼容2038年后Unix时间戳转换函数
2015/03/18 PHP
使用PHP接受文件并获得其后缀名的方法
2015/08/05 PHP
Linux下从零开始安装配置Nginx服务器+PHP开发环境
2015/12/21 PHP
PHP迭代器接口Iterator用法分析
2017/12/28 PHP
从数据结构的角度分析 for each in 比 for in 快的多
2013/07/07 Javascript
文本框水印提示效果的简单实现代码
2014/02/22 Javascript
JS 打印界面的CSS居中代码适用所有浏览器
2014/03/19 Javascript
JS动态增加删除UL节点LI及相关内容示例
2014/05/21 Javascript
javascript+html5实现绘制圆环的方法
2015/07/28 Javascript
Javascript实现快速排序(Quicksort)的算法详解
2015/09/06 Javascript
JavaScript实现图片滑动切换的代码示例分享
2016/03/06 Javascript
在DWR中实现直接获取一个JAVA类的返回值的两种方法
2016/12/25 Javascript
vue文件树组件使用详解
2018/03/29 Javascript
微信小程序出现wx.getLocation再次授权问题的解决方法分析
2019/01/16 Javascript
JS判断数组里是否有重复元素的方法小结
2019/05/21 Javascript
通过JS深度判断两个对象字段相同
2019/06/14 Javascript
写给新手同学的vuex快速上手指北小结
2020/04/14 Javascript
Vue 3.0中jsx语法的使用
2020/11/13 Javascript
收集的几个Python小技巧分享
2014/11/22 Python
Python中用pycurl监控http响应时间脚本分享
2015/02/02 Python
用Python实现通过哈希算法检测图片重复的教程
2015/04/02 Python
Android分包MultiDex策略详解
2017/10/30 Python
使用python绘制3维正态分布图的方法
2018/12/29 Python
python 去除二维数组/二维列表中的重复行方法
2019/01/23 Python
python使用opencv对图像mask处理的方法
2019/07/05 Python
深入了解Python iter() 方法的用法
2019/07/11 Python
python实现单目标、多目标、多尺度、自定义特征的KCF跟踪算法(实例代码)
2020/01/08 Python
Python3读写ini配置文件的示例
2020/11/06 Python
使用CSS3的rem属性制作响应式页面布局的要点解析
2016/05/24 HTML / CSS
合伙协议书范本
2014/04/21 职场文书
网站推广策划方案
2014/06/04 职场文书
聘用意向书
2014/07/29 职场文书
2014年信息技术工作总结
2014/12/16 职场文书
大学生简历自我评价2015
2015/03/03 职场文书
pytorch 两个GPU同时训练的解决方案
2021/06/01 Python