Python模块之pexpect

一、pexpect模块介绍

Pexpect使Python成为控制其他应用程序的更好工具。可以理解为Linux下的expect的Python封装,通过pexpect我们可以实现对ssh,ftp,passwd,telnet等命令行进行自动交互,
而无需人工干涉来达到自动化的目的

 

二、Pexpect的安装

#方法一:
pip3 install pexpect

#方法二:源码安装

三、简单示例

import pexpect
#ssh登录,账号root,主机:114.114.114.114,端口5000
child = pexpect.spawn('/usr/binssh root@114.114.114.114 -p 5000')
#expect方法等待子程序产生的输出,判断是否匹配定义的字符串
child.expect('Password')
#匹配后则发送密码串进行回应
child.sendline('hello123')

注:spawn类无法在windows下运行
如果child.sendline('
copy tftp://10.100.255.220/vip_user_list.txt flash:vip_user_list.txt')是一个下载文件命令,而下载过程又比较长,又不想将全局的timeout超时时间统一改太长?
解决方法如下:
child.timeout=1200  #超时时间12分钟
child.send(下载文件)
chile.expect('#')
child.timeout=30  #执行完下载命令,改回默认超时时间30秒即可

 

四、pexpect的核心组件

  4.1 spawn类

#功能:启动和控制子应用程序,以下是它的构造函数定义:
class pexpect.spawn(command, args=[], timeout=30, maxread=2000,
searchwindowsize=None, logfile=None, cwd=NOne, env=None, ignore_sighup=True)

#各参数解析
#1. command参数可以使任意已知的系统命令,比如:
child = pexpect.spawn('/usr/bin/ftp')    #启动ftp客户端命令
child = pexpect.spawn('/usr/bin/ssh root@114.114.114.114')#启动ssh远程连接命令
child = pexpect.spawn('ls -lattr /tmp')    #运行ls显示/tmp目录内容命令

#2. args=[] 当子程序需要参数时,可以使用Python列表来代替参数项,如:
child = pexpect.spawn('/usr/bin/ftp',[])
child = pexpect.spawn('/usr/bin/ssh',[root@114.114.114.114'])
child = pexpect.spawn('ls ' ['-lattr' ,'/tmp'])


#3. timeout 为等待结果的超时时间;
#4. maxread为pexpect从终端控制台一次读取的最大字节数
#5.searchwindowsize 为匹配缓冲区字符串的位置,默认是从开始位置匹配
#6.logfile 为写入到日志(打开文件或者终端sys.stdout),logfile也可单独出来,如
fout=open('mylog.txt','wb')
child.logfile = fout    #打印到终端,child.logfile = sys.stdout打印到屏幕(在python3中,打印到屏幕需要在spawn(encodig='utf-8')指定编码类型),最后记得关掉fout.close()

ps:会存在会话结束,但是保存数据或打印到终端数据不完全问题,解决办法:延长执行命令的次数,跟时间无关,即多次使用无意义的expect(),send()来让数据有足够的时间传回来保存 注:pexpect无法解析shell命令中的元字符(>,|,*),解决办法: #方法一: child = pexpect.spawn('/bin/bash -c "ls -l | grep LOG > logs.txt"') child.expect(pexpect.EOF) #方法二: shell_cmd = 'ls -l | grep LOG > logs.txt' child=pexpect.spawn('/bin/bash',['-c',shell_cmd]) child.pexpect(pexpect.EOF) ================================================================================== #expect方法 #expect定义了一个子程序的匹配规则 #方法定义: expect(pattern,timeout=-1,searchwindowsize=-1) #参数介绍 #1.pattern为字符串或正则表达式或pexpect.EOF(指向缓冲区尾部,无匹配项)或pexpect.TIMEOUT(匹配等待超时)或者前四种组成的列表,返回列表的索引值 #2.timeout为等待匹配结果的超时时间,超时会触发pexpect.TIMEOUT #3.searchwindowsize为匹配缓冲区字符串的位置,默认是从开始位置匹配 #pattern为列表的使用: #方法一: p = pexpect.spawn('xxx') index = p.expect(['good','bad',pexpect.EOF,pexpect.TIMEOUT]) if index == 0: fun1() elif index ==1: fun2() elif index ==2: fun3() elif index==3: fun4() #方法二: p = pexpect.spawn('xxx') index = p.expect(['good','bad']) try: if index ==0: fun1() elif index ==1: fun2() expect EOF: fun3() expect TIMEOUT: fun4() #expect方法里的before和after #before:保存了最近匹配成功的内容,俩次child.expect()之间的内容,不包含pattern部分 #after:保存了最近匹配成功之后的内容,即child.expect(这里面的内容) #before与after例子 import pexpect import sys dhild = pexpect.spawn('ssh root@192.168.2.1') child.expect(['password:']) child.sendline('hello123') print ('before:',child.before) print ('after:',child.after)
==========================================================================================
#read相关方法,与expect捕捉相配套使用 send(self,s) #发送命令,不回车 sendline(self,s='') #发送命令,回车 sendcontrol(self,char) #发送控制字符,如child.sendcontrol('c')等价于‘ctrl + c’ sendeof() #发送eof

   4.2 run函数

#run格式
pexpect.run(command,timeout=-1,withexitstatus=False,events=None,\
            extra_args=None,logfile=None,cwd=None,env=None)
#参数解析:
command:系统已知命令
event:字典,定义了expect与sendline对应关系

#实例
from pexpect import *
child=spawn('scp foo user@example.com:.')
child.expect('(?i)password')
child.sendline(mypassword)

#等同于
from pexpect import *
run('scp foo user@example.com:.',events={'(?i)password':mypassword})

  4.3 pxssh类

#pxssh格式:
class pexpect.pxssh.pxssh(timeout=30,maxread=2000,searchwindowsize=None,logfile=None,cwd=None,env=None)
#常见的三种方法
login() #建立ssh连接
logout()    #断开连接
prompt()    #等待系统提示符,用于度鞥带命令执行结束

#pxssh例子:ssh登录设备执行命令,取回结果
from pexpect import pxssh
import getpass
try:
    s=pxssh.pxssh() #创建pxssh对象
    hostname=input('hostname:').strip()
    username=input('username:').strip()
    password=getpass.getpass('password:').strip()   #接收密码输入
    s.login(hostname,username,password) #建立ssh连接
    s.sendline('uptime')    #运行uptime命令
    s.prompt()      #匹配系统提示符
    print (s.before.decode'utf-8')    #打印出现系统提示符前的命令输出
    s.sendline('ls -l')
    s.prompt()
    print (s.before.decode'utf-8')
    s.sendline('df')
    s.prompt()
    print(s.before.decode'utf-8')
    s.logout()  #断开ssh连接
except pxssh.ExceptionPxssh as e:
    print ('pxssh failed on login.')
    print (str(e))
    
    

 

posted @ 2018-05-25 00:16  森林326  阅读(5675)  评论(0编辑  收藏  举报