一、EOF说明
Shell中通常将EOF与 << 结合使用,表示后续的输入作为子命令或子Shell的输入,直到遇到EOF为止,再返回到主调Shell。可以把EOF替换成其他东西,意思是把内容当作标准输入传给程序。
回顾一下< <的用法。当shell看到< <的时候,它就会知道下一个词是一个分界符。在该分界符以后的内容都被当作输入,直到shell又看到该分界符(位于单独的一行)。这个分界符可以是你所定义的任何字符串。
使用Cat和EOF多行输入
cat >> /root/test.txt <<EOF
java
hello word !
hadoop
EOF
上的脚本命令是,在test.txt文件后面加上三行代码。
二、Expect概述
- expect 是基于tcl 演变而来的,所以很多语法和tcl 类似,Expect 是用来进行自动化控制和测试的工具。主要解决shelI脚本中不可交互的问题。对于大规模的linux运维很有帮助
- 在linux运维和开发中,我们经常需要远程登录服务器进行操作,登录的过程是一个交互的过程,可能会需要输入yes/no,password等信息。为了模拟这种输入,可以使用Expect脚本
Expect安装
- yum -y install tcl-devel
- yum -y install expect
三、基本命令
send:向进程发送字符串,用于模拟用户的输入, 该命令不能自动回车换行,一般要加\r(回车)
expect: expect的一个内部命令,判断上次输出结果里是否包含指定的字符串,如果有则立即返回,否则就等待超时时间后返回,只能捕捉由spawn启动的进程的输出expect
spawn:启动进程,并跟踪后续交互信息
interact:执行完成后保存交互状态,把控制权交给控制台
● 单位是:秒
● timeout -1 为永不超时
● 默认情况下,timeout是10秒
set timeout 30:设置超时时间为30秒(默认的超时时间是 10 秒,通过 set 命令可以设置会话超时时间, 若不限制超时时间则应设置为-1)
exp_continue: 允许expect继续向下执行指令meout:指定超时时间,过期则继续执行后续指令
send_user: 回显命令,相当于echo
$argv参数数组:Expect脚本可以接受从bash传递的参数,可以使用 [lindex $argv n] 获得,n从0开始,分别表示第一个$1,第二个$2,第三个$3……参数 ($argvn没有空格则表示脚本名称 ; $argv n有空格则代表下标)
一般流程:spawn 启动追踪 —> expect 匹配捕捉关键字 ——> 捕捉到将触发send 代替人为输入指令—> interact /expect eof
Expect脚本必须以interact或expect eof 结束,执行自动化任务通常expect eof就够了
● expect eof 是在等待结束标志。由spawn启动的命令在结束时会产生一个eof标记,expect eof 即在等待这个标记
四、Expect语法
send命令不具备回车换行功能,一般要加 \r 或 \n
expect “password:” {send “mypassword\r”;}
方式一:
expect “aaa” {send “AAA\r”}
expect “bbb” {send “BBB\r”}
expect “ccc” {send “CCC\r”}
方式二:只要匹配了aaa或bbb或ccc中的任何一个,执行了了相应的send语句后将会退出该expect语句
expect {
“aaa” {send “AAA\r“}
“bbb” {send “BBB\r”}
“ccc” {send “CCC\r”}
}
方法三:exp_continue表示继续后面的匹配,如果匹配了aaa,执行完send语句后还要继续向下匹配bbb
exoect {
“aaa” {send “AAA\r“;exp_continue}
“bbb” {send “BBB\r”;exp_continue}
“ccc” {send “CCC\r”}
}
五、Expect执行方式
案例:ssh远程免交互登录
[root@localhost ~]#vi login.sh
#!/usr/bin/expect //expect文件路径
set timeout 60
set hostname [lindex $argv 0]
set password [lindex $argv 1]
spawn ssh root@$hostname
expect {
"*yes/no" {send "yes\r";exp_continue}
"*password" {send "$password\r"}
}
interact
[root@localhost ~]#chmod +x a.sh
[root@localhost ~]#./login.sh 192.168.100.101 123abc
[root@localhost ~]#vi login.sh
#!/bin/bash
hostname=$1
password=$2
set timeout 60
/usr/bin/expect<<-EOF //加载expect文件路径
spawn ssh root@${hostname}
expect {
"(yes/no)" {send "yes\r";exp_continue}
"*password" {send "$password\r"}
}
expect "*]#" //匹配进入后的页面
send "exit\r" //触发退出
expect eof //等待结束
EOF //expect结束标志,EOF前后不能有空格
[root@localhost ~]#chmod +x a.sh
[root@localhost ~]#./login.sh 192.168.100.101 123abc
案例一:创建用户tom,密码tom123**
- [root@localhost ~]#vim b.sh
- #!/bin/bash
- user=$1
- password=$2
- useradd $user
- expect<<-EOF //加载expect文件路径
- spawn passwd $user //开启进程跟踪passwd
- expect {
- "New password:" {send "$password\r";exp_continue} //匹配关键字并自动输入密码
- "*password" {send "$password\r"} //匹配关键字并自动输入密码
- }
- expect eof //等待结束
- EOF
- [root@localhost ~]#chmod +x b.sh
- [root@localhost ~]#./b.sh tom tom123
案例二:SSH登录
● 首次登录
● 正常登录
● 连接被拒绝,可能是ssh没开,或者端口不对,或者防火墙限制
● 没有连接地址
- [root@localhost ~]#vim c.sh
- #!/usr/bin/expect //expect文件路径
- set timeout 20
- set hostname [lindex $argv 0]
- set password [lindex $argv 1]
- spawn ssh root@$hostname
- expect {
- "Connection refused" exit
- "Name or service not know" exit
- "(yes/no)" {send "yes\r";exp_continue}
- "*password:" {send "$password\r"}
- }
- interact
- exit
- [root@localhost ~]#chmod +x a.sh
- [root@localhost ~]#./c.sh 192.168.100.100 123abc
案例三:FTP登录
- [root@localhost ~]#vim d.sh
- #!/bin/bash
- hostname=$1
- expect<<-EOF
- spawn ftp $hostname
- expect {
- "Name" {send "ftp\r";exp_continue}
- "Password" {send "\r"}
- }
- expect eof
- EOF
- [root@localhost ~]#chmod +x d.sh
- [root@localhost ~]#./d.sh 192.168.100.100 //本机要开启ftp,对方也要开启