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