使用I/O重定向的方式将命令列表提供给交互式程序或命令,比如ftp、cat或read命令
是标准输入的一种替代品可以帮助脚本开发人员不必使用临时文件来构建输入信息,而是直接就地产生出一个“文件”并用作“命令”的标准输入。Here Document 也可以与非交互式程序和命令一起使用
命令 << 标记
……
内容——————标记之间是传入内容
……
标记
注意事项:
> wc -l <EOF
> Line 1
> Line2
> EOF
> read i <<EOF
> Hi
> EOF
> echo $i
> passwd lisi <<EOF
> abc1234——————这两行是输入的密码和确认密码
> abc1234
> EOF
在写入文件时会先将变量替换成实际值,再结合cat命令完成写入
> #!/bin/bash
> file="EOF1.txt"
> i="school"
> cat > $file <<EOF
> I am going to $i
> EOF
> cat EOF1.txt
> #!/bin/bash
> var="Great! I am going to school!"
> myvar=$(cat <<EOF
> This is Line 1.
> Today is Monday.
> $var
> EOF
> )
> echo $myvar
>#!/bin/bash
>var="Great! I am going to school!"
>myvar=$(cat <<'EOF'——————对标记加单引号,即可关闭变量替换
>This is Line 1.
>Today is Monday.
>$var
>EOF
>)
>echo $myvar
> #!/bin/bash
> var="Great! I am going to school!"
> myvar=$(cat <<-'EOF'——————在标记前加”-“,即可抑制个行首TAB或空格
> This is Line 1.
> Today is Monday.
> $var
> EOF
> )
> echo -e "$myvar"
Bash的默认注释是”#“,该注释方法只支持单行注释,Here Document的引入解决了多行注释的问题
":"代表上面都不做的空命令,中间标记区域的内容不会倍执行,会被bash忽略掉,因此可达到批量注释的效果
> #!/bin/bash
> var="Great! I am going to school!"
> : <<-‘EOF’——————多行注释,“:”开头的 Here Document 标记内容不会被执行
> This is Line 1.
> Today is Monday.
> $var
> EOF
> echo myvar
建立在tcl语言基础上的一个工具,常被用于进行自动化控制和测试,解决shell脚本中交互相关的问题
rpm -1 expect
rpm -q tcl
yum -y install expect
(1)脚本解释器
expect脚本种首先引入文件,表明使用的是哪一个shell
#!/usr/bin/expect
(2)spawn
spawn后面通常跟一个Linux执行命令,表示开启一个会话、启动进程,并跟踪后续交互信息。例:spawn passwd root
(3)expect
判断上次输出结果中是否包含指定的字符串,如果有则立即返回,否则就等待超时时间后返回
只能捕捉由spawn启动的进程的输出
用于接收命令执行后的输出,然后和期望的字符串匹配
(4)send
向进程发送字符串,用户模拟用户的输入;该命令不能自动回车换行,一般要加\r(回车)或者\n
例:
方式一:
expect “密码” {send “abc123\r”}————同一行send部分要有{}
方式二:
expect “密码”
send ”$abc123\r“————换行send部分不需要有{}
方式三:
expect支持多个分支
> expect——————只要匹配了其中一个情况,执行相应的send语句后退出该expect语句
> {
> "密码1" {
send "abc123\r"}
> "密码2" {
send "123456\r"}
> "密码3" {
send "123123\r"}
> }
(5)结束符
expect eof
表示交互结束,等待执行结束,退回到原用户,与spawn对应
比如切换到root用户,expect脚本默认的是等待10s,当执行完命令后,默认停留10s后,自动切回了原用户
interact
执行完成后保持交互状态,把控制权交给控制台,会停留在目标终端而不会退回到原终端,这个时候就可以手工操作了,interact后的命令不起作用,比如interact后添加exit,并不会退出root用户,而如果没有interact则登录完成后会退出,而不是留在远程终端上
使用interact会保持在终端而不会退回到原终端,比如切换到root用户,会一直在root用户状态下,比如ssh到另一服务器,会一直在目标服务器终端,而不会切回到原服务器
注意:expect eof 与interact只能二选一
(6)set
expect默认的超时时间是10秒,通过set命令可以设置会话超时时间,若不限制超时时间则应设置为-1
例:set timeout 30
(7)exp_continue
exp_continue附加于某个expect判断项之后,可以使该项被匹配后,还能继续匹配该expect判断语句内的其他项,exp_continue类似于控制语句中的continue语句,表示允许expect继续向下执行指令
例如下例将判断交互输出中是否存在yes/no或者*assword。如果匹配yes/no则输出yes并再次执行判断;如果匹配*assword则输出abc123并结束该段expect语句
> expect {
> "(yes/no)" {
send "yes\r"; exp_continue}
> "\*password" {
set timeout 300;send "abc123\r";}
> }
注意:使用exp_continue时,如果跟踪像passwd这样的输入密码后就结束进程的命令,expect{}外不要再加上expect eof,因为spawn进程结束后会默认向expect发送eof,会导致后面的expect eof执行报错
(8)send_user
send_user表示回显命令,相当于echo
(9)接收参数
expect脚本可以接收从bash命令行传递的参数,使用[lindex $argv n]获得,其中n从0开始,分别表示第一个,第二个,第三个…参数
set hostname [lindex $argv 0]————相当于hostname=$1
set password [lindex $argv 1]————相当于password=$2
(10)expect直接执行
需要使用expect命令执行脚本
> #!/usr/bin/expect ————声明解释器
> set timeout 5 ————设置超时时间
> set username [lindex $argv 0] ————参数传入
> set password [lindex $argv 1] ————参数传入
> spawn su $username ———— 开始追踪命令
> expect "密码"
> send "$password\r"
> expect "*]#"
> send_user "ok" 免交互执行,捕捉信息匹配
> interact ————把控制权交给控制台
>或者 expect eof
嵌入执行模式,将expect过程融入shell当中,方便执行和处理
> #! /bin/bash
> user=$1
> password=$2
> useradd $user ———— 非交互命令放在expect外面
> /usr/bin/expect <<-EOF ———————— 开始免交换执行 , expect开始标志
> spawn passwd $user —— 开启-一个进程跟踪passwd命令,expect只能捕捉该进程信息
> expect "新的*"
> send "$ {password}\r"
> expect "重新*"
> send "$ {password} \r"
> expect eof
> EOF
> #! /usr/bin/ expect
> set timeout 5
> set hostname [l index $argv 0 ]
> set password [l index $argv 1]
> spawn ssh $hostname
> expect {
> "Connection refused" exit ———— 连接失败情况,比如对方ssh服务关闭
> "Name or service not known" exit —————— 找不到服务器,比如输入的IP地址不正确
> " (yes/no)" {
send "yes\r" ;exp_ continue}
> "password:" {
send "$password\r"}
> interact
> exit ———————— interact后的命令不起作用
>#!/bin/bash
>/usr/bin/expect <<EOF
>spawn fdisk /dev/$1
>expect "命令(输入 m 获取帮助)" {
send "n\n"}
>expect "Select (default p)" {
send "p\n"}
>expect "分区号" {
send "\n"}
>expect "起始 扇区" {
send "\n"}
>expect "Last 扇区, +扇区" {
send "\n"}
>expect "命令(输入 m 获取帮助)" {
send "w\n"}
>expect eof
>EOF
>fdisk -l
>mkfs.xfs -f /dev/$11 &>/dev/null
>if [ $? -ne 0 ];then
>echo "未格式化成功,请检查脚本"
>else
>echo "格式化成功"
>fi