expect介绍及使用示例

expect介绍及使用示例

expect是一种能够按照脚本内容里面设定的方式与交互式程序进行“会话”的程序。根据脚本内容,Expect可以知道程序会提示或反馈什么内容以及 什么是正确的应答。它是一种可以提供“分支和嵌套结构”来引导程序流程的解释型脚本语言。

shell功能很强大,但是不能实现有交互功能的多机器之前的操作,例如ssh和ftp.而expect可以帮助我们来实现.

一,安装expect

yum install expect  

其实expect根bash形势上差不多的.

二,实例

1,ssh实现自动登录,并停在登录服务器上

#!/usr/bin/expect -f  
 set ip [lindex $argv 0 ]     //接收第一个参数,并设置IP  
 set password [lindex $argv 1 ]   //接收第二个参数,并设置密码  
 set timeout 10                   //设置超时时间  
 spawn ssh root@$ip       //发送ssh请滶  
 expect {                 //返回信息匹配  
 "*yes/no" { send "yes\r"; exp_continue}  //第一次ssh连接会提示yes/no,继续  
 "*password:" { send "$password\r" }      //出现密码提示,发送密码  
 }  
 interact          //交互模式,用户会停留在远程服务器上面. 

运行结果如下:

root@ubuntu:/home/zhangy# ./test.exp 192.168.1.130 admin  
spawn ssh root@192.168.1.130  
Last login: Fri Sep  7 10:47:43 2012 from 192.168.1.142  
[root@linux ~]#  

这个例子有统一的接口,根据IP和密码可以连接到不同的机器.如果你嫌输入IP和密码麻烦,看下面的例子

#!/usr/bin/expect -f  
 set ip 192.168.1.130  
 set password admin  
 set timeout 10  
 spawn ssh root@$ip  
 expect {  
 "*yes/no" { send "yes\r"; exp_continue}  
 "*password:" { send "$password\r" }  
 }  
 interact 

运行结果如下:

root@ubuntu:/home/zhangy# ./web.exp  
spawn ssh root@192.168.1.130  
Last login: Fri Sep  7 12:59:02 2012 from 192.168.1.142  
[root@linux ~]#  

2,ssh远程登录到服务器,并且执行命令,执行完后并退出

#!/usr/bin/expect -f  
 set ip 192.168.1.130  
 set password admin  
 set timeout 10  
 spawn ssh root@$ip  
 expect {  
 "*yes/no" { send "yes\r"; exp_continue}  
 "*password:" { send "$password\r" }  
 }  
 expect "#*"  
 send "pwd\r"  
 send  "exit\r"  
 expect eof

运行结果如下:

root@ubuntu:/home/zhangy# ./test3.exp  
spawn ssh root@192.168.1.130  
root@192.168.1.130's password:  
Last login: Fri Sep  7 14:05:07 2012 from 116.246.27.90  
[root@localhost ~]# pwd  
/root  
[root@localhost ~]# exit  
logout  
Connection to 192.168.1.130 closed.  

3,远程登录到ftp,并且下载文件

#!/usr/bin/expect -f  
 set ip [lindex $argv 0 ]  
 set dir [lindex $argv 1 ]  
 set file [lindex $argv 2 ]  
 set timeout 10  
 spawn ftp $ip  
 expect "Name*"  
 send "zwh\r"  
 expect "Password:*"  
 send "zwh\r"  
 expect "ftp>*"  
 send "lcd $dir\r"  
 expect {  
 "*file"  { send_user "local $_dir No such file or directory";send "quit\r" }  
 "*now*"  { send "get $dir/$file $dir/$file\r"}  
 }  
 expect {  
 "*Failed" { send_user "remote $file No such file";send "quit\r" }  
 "*OK"     { send_user "$file has been download\r";send "quit\r"}  
 }  
 expect eof  

运行结果如下:

root@ubuntu:/home/zhangy# ./test2.exp 192.168.1.130 /var/www/www aaa.html  
spawn ftp 192.168.1.130  
Connected to 192.168.1.130.  
220 (vsFTPd 2.0.5)  
Name (192.168.1.130:root): zwh  
331 Please specify the password.  
Password:  
230 Login successful.  
Remote system type is UNIX.  
Using binary mode to transfer files.  
ftp> lcd /var/www/www  
Local directory now /var/www/www  
ftp> get /var/www/www/aaa.html /var/www/www/aaa.html  
local: /var/www/www/aaa.html remote: /var/www/www/aaa.html  
200 PORT command successful. Consider using PASV.  
150 Opening BINARY mode data connection for /var/www/www/aaa.html (66 bytes).  
226 File send OK.  
66 bytes received in 0.00 secs (515.6 kB/s)  
quit aaa.html has been download  
221 Goodbye. 

4.改装scp程序,使其不需要再输入密码(该脚本文件名为:fun)

#!/usr/bin/expect -f
#scp 远程copy文件
set user root
set passwd zytech@e800
#接受传入的第一个参数
set file [lindex $argv 0]
#接受传入的第二个参数
set ip [lindex $argv 1]
set timeout 10//超时时间10秒
#spawn用来启动一个新的进程
spawn scp -r /home/zhu/softs/$file $user@192.168.8.$ip:/opt
#expect用来等待你所期望的字符串,send用来发送字符串
expect {
"*yes/no" { send "yes\r";exp_continue}
"*password:" { send "$passwd\r" }
}
#行完成后保持交互状态,把控制权交给控制台,这个时候就可以手工操作了。
#果没有这一句登录完成后会退出,而不是留在远程终端上。如果你只是登录过去执行一段命令就退出,可改为[expect eof]
#interact
expect eof

5.关闭防火墙:

#关闭防火墙-方法一
#! /usr/bin/expect -f
spawn sudo service iptables stop
expect "password"
send "123456\r"
expect eof
#关闭防火墙-方法二
#! /usr/bin/expect -f
spawn sudo service iptables stop
expect {
  "password" {send "123456\r";exp_continue}
}
expect eof


运行:fun oozie.tar.gz 168

三、对于上述示例中相关命令的解释

1  [#!/usr/bin/expect]

这一行告诉操作系统脚本里的代码使用那一个shell来执行。这里的expect其实和linux下的bash、windows下的cmd是一类东西。

注意:这一行需要在脚本的第一行。


2. [set timeout 30]

基本上认识英文的都知道这是设置超时时间的,现在你只要记住他的计时单位是:秒


3. [spawn ssh -l username 192.168.1.1]

spawn是进入expect环境后才可以执行的expect内部命令,如果没有装expect或者直接在默认的SHELL下执行是找不到spawn命令的。所以不要用 “which spawn“之类的命令去找spawn命令。好比windows里的dir就是一个内部命令,这个命令由shell自带,你无法找到一个dir.com 或 dir.exe 的可执行文件。


它主要的功能是给ssh运行进程加个壳,用来传递交互指令。


4. [expect "password:"]


这里的expect也是expect的一个内部命令,有点晕吧,expect的shell命令和内部命令是一样的,但不是一个功能,习惯就好了。这个命令的意思是判断上次输出结果里是否包含“password:”的字符串,如果有则立即返回,否则就等待一段时间后返回,这里等待时长就是前面设置的30秒


5. [send "ispass/r"]


这里就是执行交互动作,与手工输入密码的动作等效。

温馨提示: 命令字符串结尾别忘记加上 “/r”,如果出现异常等待的状态可以核查一下。


6. [interact]


执行完成后保持交互状态,把控制权交给控制台,这个时候就可以手工操作了。如果没有这一句登录完成后会退出,而不是留在远程终端上。如果你只是登录过去执行一段命令就退出,可改为[expect eof]


四、expect提供的常用命令使用参考

1. send

用来发送一个字符串,比如 send "hello world"。

初始情况下,这个字符串会发送到标准输出。如果你用的是max OSX或者linux,可以在Terminal下直接输入expect命令并回车,就进入了expect交互环境,此时,输入send "hello world"就可以看到结果。

一旦你的程序已经与其他程序进行交互,字符串就会被发送到其他程序那里。如上面的例子脚本中,我们调用send ”$password\r"就是把密码发送给SSH连接的服务器端指定端口。

2. expect

与send相反,expect用来等待你所期望的字符串。比如expect "hello"

在expect后面跟的字符串中,你可以指定一个正则表达式。

expect会一直等待下去,除非收到的字符串与预期的格式匹配,或者到了超期时间。

3. spawn

spawn用来启动一个新的进程,比如上面的spawn ssh -D $port $user@$host,Expect会执行命令“ssh -D $port $user@$host”。

在交互式的场景中,当你输入命令后,可能服务器端会返回一些操作提示符,以让你输入命令。Expect提供了这样三个常用的命令,spawn, expect和send,恰好满足这种需要。把它们结合起来使用,可以实现很多简单的自动化脚本。

其它常用的命令还有:interact,比如你通过脚本自动连接到了某个ftp,并输入了用户名密码,此时需要人工输入一些命令,就可以使用interact命令,它会把脚本的控制权交给用户;sleep,等待多少秒等等。

由于expect是从Tcl继承下来的,所以也支持Tcl的语法和命令,比如变量声明、流程控制等等。


上面脚本的一些解释:

1. set timeout 300:设置超时时间300s。如果设为-1,代表永不超时。

2. expect eof:等待接受文件结束符。

五、关于expect的参考网址:


blog.51yip.com/linux/1462.html
rfyiamcool.blog.51cto.com/1030776/980343
linux.chinaunix.net/techdoc/desktop/2007/02/14/950495.shtml



  • 0
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值