expect简介和使用例子(转)

expect简介

expect是一款自动化的脚本解释型的工具。

expect基于tcl脚本,expect脚本的运行需要tcl的支持。

expect对一些需要交互输入的命令很有帮助,比如ssh ftp scp telnet。

远程登录linux服务器的时候,ssh命令需要手工输入密码,当登录多台机器的时候就会非常繁琐。

expect就可以根据设定的规则,自动帮我们输入密码,大大节省了时间。

expect安装

一般机器不会自带expect,需要手动安装。

系统为RHEL/CentOS:

yum install expect

系统为Debian/Ubuntu:

apt-get install expect

expect基础知识

expect脚本

脚本开头

expect脚本一般以#!/usr/bin/expect -f开头,类似bash脚本。

常用后缀

expect脚本常常以.exp或者.ex结束。

expect主要命令

  • spawn 新建一个进程,这个进程的交互由expect控制
  • expect 等待接受进程返回的字符串,直到超时时间,根据规则决定下一步操作
  • send 发送字符串给expect控制的进程
  • set 设定变量为某个值
  • exp_continue 重新执行expect命令分支
  • [lindex $argv 0] 获取expect脚本的第1个参数
  • [lindex $argv 1] 获取expect脚本的第2个参数
  • set timeout -1 设置超时方式为永远等待
  • set timeout 30 设置超时时间为30秒
  • interact 将脚本的控制权交给用户,用户可继续输入命令
  • expect eof 等待spawn进程结束后退出信号eof

expect命令分支

expect命令采用了tcl的模式-动作语法,此语法有以下几种模式:

单一分支语法

set password 123456
expect "*assword:" { send "$password\r" }

当输出中匹配*assword:时,输出password变量的数值和回车。

多分支模式语法

set password 123456
expect {
      "(yes/no)?" { send "yes\r"; exp_continue }
      "*assword:" { send "$password\r" } }

当输出中包含(yes/no)?时,输出yes和回车,同时重新执行此多分支语句。

当输出中匹配*assword:时,输出password变量的数值和回车。

expect详细讲解

ssh远程登录expect脚本

下面是一个自动登录系统hostname1和hostname2执行uname -a后断开连接的脚本。

先建立login.exp

touch login.exp
chmod +x login.exp
vim login.exp

内容如下:

#!/usr/bin/expect -f

set timeout -1  //永远等待,不会超时
spawn ssh root@hostname1   //spawn 后面跟命令名称和参数

//如果匹配到*assword,那么发送密码,并进入下面的expect语句(uname -a语句)。 //如果匹配到yes/no,那么发送yes,并重新执行这个expect语句。 expect { "*assword" {send "123456\r";} "yes/no" {send "yes\r";exp_continue} } //匹配到*]#,那么运行uname -a命令 expect "*]#" {send "uname -a\r"} send "exit\r" //退出远程登录 expect eof //结束spawn //开始下一个命令 spawn ssh root@hostname2 expect { "*assword" {send "123456\r";} "yes/no" {send "yes\r";exp_continue} } expect "*]#" {send "uname -a\r"} send "exit\r" //退出远程登录 expect eof //结束spawn exit //退出expect脚本 

ssh远程登录shell脚本(嵌套expect)

shell中使用expect -c "expect脚本内容"来完成嵌套。

注意:

expect脚本里面的"都需要在前面加上转义符号。

每个expect语句后面加上分号";"。

vim expect_in_shell.sh

#!/usr/bin/bash

HOSTS="hostname1 hostname2"  
  
for host in $HOSTS do expect -c " set timeout 5; spawn ssh root@${host}; expect { \"*assword\" { send \"123456\r\" } \"yes/no\" { send \"yes\r\"; exp_continue } } ; expect \"*]#\" {send \"uname -a\r\" } ; send \"exit\r\" //退出远程登录 expect eof " done 

带参数的expect脚本ssh登录

vim login_arg.exp

#!/usr/bin/expect -f
set ip [lindex $argv 0]   //第一个参数赋值给变量ip
set password [lindex $argv 1] //第二个参数复制给变量password set timeout -1 spawn ssh root@$ip expect { "password" {send "$password\r";} "yes/no" {send "yes\r";exp_continue} } interact //停留在远程shell 

带参数运行login_arg.exp

chmod +x login_arg.exp
./login_arg.exp 127.0.0.1 123456

猜你喜欢

转载自www.cnblogs.com/alpha1981/p/10966906.html