Expect概述
Expect是建立在tcl基础上的一个工具,Expect是用来进行自动化控制和测试的工具。主要解决shell脚本中不可交互的问题。对于大规模的Linux运维很有帮助
在Linux运维和开发中,我们经常需要远程登录服务器进行操作,登录过程是一个交互的过程,可能会需要输入yes/no,password等信息。为了模拟这种输入,可以使用Expect脚本。
基本命令
send:向进程发送字符串,用于模拟用户的输入
- 该命令不能自动回车换行,一般要加\r(回车)
expect
- expect的一个内部命令,判断上次输出结果里是否包含指定的字符串,如果有则立即返回,否则就等待超时时间后返回
- 只能捕捉由spawn启动的进程的输出
spawn:启动进程,并跟踪后续交互信息
interact:执行完成后保持交互状态,把控制权交给控制台
Timeout:指定超时时间,过期则继续执行后续指令
- 单位是:秒
- timeout -1 为永不超时
- 默认情况下,timeout是10秒
exp_continue
- 允许expect继续向下执行指令
send_user
- 回显命令,相当于echo
$argv 参数数组
- Expect脚本可以接受从bash传递的参数,可以使用[lindex $argv n]获得,n从0开始,分别表示第一个,第二个,第三个...参数
Expect脚本必须以interact或expect eof结束,执行自动化任务通常expect eof就够了
- expect eof是在等待结束标志。由spawn启动的命令在结束时会产生一个eof标记,expect eof即在等待这个标记
Expect语法
单一分支语法
expect "password:" {send "mypassword\r";}
多分支模式语法
send命令不具备回车换行功能,一般要加\r或\n
expect "aaa" {send "AAA\r"}
expect "aaa" {send "AAA\r"}
expect "aaa" {send "AAA\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\r";exp_continue}
"bbb" {send "BBB\r";exp_continue}
"ccc" {send "CCC\r"}
}
-re参数表示匹配正则表达式
Expect执行方式
直接执行
SSH登录
首次登陆
正常登录
连接被拒绝,可能ssh没开,或端口号不对,亦或是防火墙限制
没有该连接地址
#!/usr/bin/expect //Expect二进制文件的路径
#超时时间
set timeout 20 //20秒等待时间
log_file test.log //日志文件
log_user 1 //日志用户
#参数传入
set hostname [lindex $argv 0] //追加参数0,统计变量并加载第1位置参数
set password [lindex $argv 1] //追加参数1,统计变量并加载第2位置参数
#追踪命令
spawn ssh [email protected]$hostname //追踪命令
#捕捉信息并匹配免交互执行
expect { //捕捉提示信息
"Connection refused" exit //捕捉到拒接连接信息即退出
"service not konwn" exit //捕捉到服务为开启信息即退出
"(yes/no)" // 捕捉yes或no参数
{send "yes\r";exp_continue} //输入yes并继续执行
"*password" // 捕捉参数
{send "$password\r"} //输入密码参数
}
#控制权交予控制台
interact //控制权转交控制台人为输入
[[email protected] ~]# ./expect.sh 192.168.109.132 baby520./
spawn ssh [email protected]
[email protected]‘s password:
Last login: Thu Oct 10 16:52:48 2019 from 192.168.109.10
[[email protected] ~]#
Expect执行方式
嵌入执行
#!/bin/bash
user=$1
password=$2
#非交互命令放expect外面
useradd $user
#开始交互
expect <<-EOF //Expect开始标志,标准性输入,等同于stdin
spawn passwd $user
expect "新的*"
send "$password\r"
expect "重新*"
send "$password\r"
expect eof;
EOF //Expect结束语句,EOF前后不能有空格
[[email protected] ~]# ./expect.sh 192.168.109.132 baby520./
spawn ssh [email protected]
[email protected]‘s password:
Last login: Thu Oct 10 17:38:16 2019 from 192.168.109.133
[[email protected] ~]# exit
登出
Connection to 192.168.109.132 closed.
案例1
创建自定义用户,密码自定义
#!/bin/bash
username=$1
password=$2
useradd $username
/usr/bin/expect <<-EOF
spawn passwd $username
expect {
"新的*"
{send "$password\r";exp_continue}
"重新*"
{send "$password\r"}
}
EOF
[[email protected] ~]# ./tom.sh test2 123123
spawn passwd test2
更改用户 test2 的密码 。
新的 密码:
无效的密码: 密码少于 8 个字符
重新输入新的 密码:
passwd:所有的身份验证令牌已经成功更新。
案例2
利用expect完成FTP登录过程
#!/usr/bin/expect -f
set timeout 20
spawn ftp 192.168.109.133
expect "Name*"
send "ftp\r"
expect "Password:*"
send "\r"
expect "ftp>*"
interact
expect eof
谢谢阅读!!!
原文地址:https://blog.51cto.com/14080162/2441338
时间: 2024-09-28 16:14:55