目录
expect概述
-
expect 是建立在 tcl 语言基础上的一个工具,它可以让一些需要交互的任务自动化地完
成,相当于模拟了用户和命令行的交互操作。expect 是用来进行自动化控制和测试的工具。
主要解决 shell 脚本中不可交互的问题。对于大规模的 Linux 运维很有帮助。
-
在 Linux 运维和开发中,经常需要远程登录服务器进行操作,登录的过程是一个交互的
过程,可能会需要输入 yes/no, password 等信息。为了模拟这种输入,可以使用 expect
脚本。
-
在实际的生产环境中,有一个常用的场景就是批量配置集群无密钥登录。如果集群的机
器数量很多,手动一台一台地去每台机器去配置无密钥是非常糟糕的事情。使用 expect 功
能,可以远程登录机器,并通过交互方式进行无密钥登录。
-
安装
yum install -y expect
基本命令介绍
脚本解释器
-
expect 脚本中首先引入文件,表明使用的是哪一个 shell
#!/usr/bin/expect
expect/send
-
expect 命令用来判断上次输出结果里是否包含指定的字符串,如果有则立即返回,否
则就等待超时时间后返回,只能捕捉由 spawn 启动的进程的输出。
-
expect 接收命令执行后的输出,然后和期望字符串匹配,若匹配成功则执行相应的
send 向进程发送字符串,用于模拟用户的输入。Send 发送的命令不能自动回车换行,一般
要加\r(回车)。其常见语法形式有以下三种。
方式一
expect "$case1" {send "$respond1\r"}
方式二
expect "$case1"
send "$response1\r"
方式三
expect
{
"$case1" {send "$response1\r"}
"$case2" {send "$response2\r"}
"$case3" {send "$response3\r"}
}
-
上述语法结构中$case1 代表检测命令的输出结果,如果输出内容和$case1 一致,通过
send 命令模拟用户发送内容到终端
spawn
-
spawn 后面通常跟一个命令,表示开启一个会话、启动进程,并跟踪后续交互信息
spawn Linux 执行命令
spawn passwd
结束符
-
expect eof :等待执行结束,若没有这一句,可能导致命令还没执行,脚本就结束了。
-
interact : 执行完成后保持交互状态, 把控制权交给控制台,这时可以手动输入信息。
需要注意的是,expect eof 与 interact 只能二选一。
set
-
expect 默认的超时时间是 10 秒,通过 set 命令可以设置会话超时时间, 若不限制超时
时间则应设置为-1。例如执行以下命令即可将超时时间设置为 30 秒。
set timeout 30
exp_continue
-
exp_continue 表示允许 expect 继续向下执行指令。
send_user
-
send_user 表示回显命令,相当于 echo。
接收参数
-
expect 脚本可以接受从 bash 传递的参数,使用[lindex $argv n]获得。其中 n 从 0 开始,
分别表示第一个,第二个,第三个....参数。
-
参数存在 argv 中,使用第一个参数如下:
set param0 [lindex $argv 0]
-
$argc 表示参数个数,判断语句如下:
if {$argc< 1} {
#do something
send_user "usage: $argv0 <param1><param2> ... "
exit
}
-
在上述脚本中,$argv0 是脚本名,但[lindex $argv 0]是第一个参数 param1, [lindex
$argv 1]是第二个参数 param2, 以此类推。send_user 用来显示信息到父进程(一般为用户
的 shell)的标准输出。
expect 语法
单一分支语法
-
单一分支用于简单的用户交互,当监控命令的标准输出满足 expect 指定的字符串时,
向标准输入发送 send 指定的字符串。具体用法如下所示。默认情况下,send 不会向标准
输入发送回车键,所以需要通过\r 手动换行。
expect "password:" {send "mypassword\r“;}
多分支语法
-
多分支用于复杂的用户交互,一般情况下输出内容可能有多个,根据不同的输出内容,
分别向标准输入发送不同的内容。其语法格式如下所示,只要匹配了 aaa、bbb 或 ccc 中的
任何一个,就执行相应的 send 语句,然后退出该 expect 语句。
expect
{
"aaa" {send "AAA\r"}
"bbb" {send "BBB\r"}
"ccc" {send "CCC\r"}
}
-
除了上述的多分支结构之外,还有另外一种多分支结构,具体使用方法如下所示。
exp_continue 表示继续后面的匹配,假如配了 aaa,执行完 send 语句后还要继续向下匹配
bbb。
expect
{
"aaa" {send “AAA”;exp_continue}
"bbb" {send “BBB”;exp_continue }
"ccc" {send "CCC"}
}
expect执行方式
直接执行
cat direct.sh
###########################################################################################
#!/usr/bin/expect
# expect 将超时时间设置为 60 秒
set timeout 60
# 记录日志文件
log_file test.log
# 开启控制台输出,为0时,控制台不输出
log_user 1
# $argc 表示参数个数
if {$argc<1} {
# send_user相当于shell的echo
send_user "usage: $argv0 <param1><param2> ... "
exit
}
set hostname [lindex $argv 0]
set password [lindex $argv 1]
# spawn 后面通常跟一个命令,表示开启一个会话、启动进程,并跟踪后续交互信息
spawn ssh root@${hostname}
expect {
"(yes/no)"
# 匹配的(yes/no)时自动输入yes,exp_continue 表示允许 expect 继续向下执行指令
{send "yes\r"; exp_continue}
"*password"
{send "$password\r"}
}
expect "*#" {send "ifconfig\r"}
# expect eof :等待执行结束,若没有这一句,可能导致命令还没执行,脚本就结束了
# interact : 执行完成后保持交互状态, 把控制权交给控制台,这时可以手动输入信息。 需要注意的是,expect eof 与 interact 只能二选一。
interact
###########################################################################################
chmod +x direct.sh
./direct.sh 127.0.0.1 123456
嵌入执行
cat expect_ch_passwd.sh
###########################################################################################
#!/bin/bash
user=$1
password=$2
useradd $user
expect << EOF
# 开启一个进程跟踪 passwd 命令,expect 只能捕捉该进程信息
spawn passwd $user
# 匹配输出信息"New password:"
expect "New password:"
# 自动输入密码
send "${password}\r"
# 匹配输出信息"Retype new password:"
expect "Retype new password:"
# 自动输入密码
send "${password}\r"
# 等待结束标记
expect eof;
EOF
###########################################################################################