文章目录
一、Here Document免交互
1.Here Document 概述
Here Document使用I/O重定向的方式将命令列表提供给交互式程序如ftp、cat、tee、read 、passwd、su等等;它是标准输入的一种替代品。
2.语法格式
命令 <<标记
......
传入的内容
......
标记
#常使用EOF做标记
3.注意事项
- 标记可以使用任意合法字符,但开始标记与结尾标记需要保持一致。(常用标记EOF,使用其他字母也可以)
- 结尾的标记一定要顶格写,前面不能有任何字符。
- 结尾的标记后面也不能有任何字符(包括空行)。
- 开头标记前后的空格会被省略掉。
4.案例
-
通过read命令接收输入并打印
-
免交互方式实现对行数的统计
-
免交互修改用户密码
-
多行重定向到文件
-
Here Document支持变量替换
如果标记之间有变量被使用,会先替换变量值。 如果想要将一些内容写入文件,除了常规的方法外,也可以使用 Here Document。
如果写入的内容中包含变量,在写入文件时要先将变量替换成实际值,再结合cat命令完成输出。
-
多行注释
Bash 的默认注释是“#”,该注释方法只支持单行注释;Here Document 的引入解决了多行注释的问题。
:(冒号)代表什么都不做的空命令。中间标记区域的内容不会被执行,会被bash忽略掉,因此可达到批量注释的效果。
二、expect
expect是建立在tcl(tool command language)语言基础上的一个工具,常被用于进行自动化控制和测试,解决shell脚本中交互的相关问题。
expext是需要安装的,yum install -y expect 进行安装
1.expect脚本相关命令
-
#!/usr/bin/expect
脚本解释器,expect 脚本中首先引入文件,表明使用的事哪一种shell -
spawn 启动新的进程(监控,捕捉)
spawn后面通常跟一-个Linux执行命令,表示开启一个会话、启动进程,并跟踪后续交互信息 例: spawn passwd root -
expect 从进程接收字符串
- 判断上次输出结果中是否包含指定的字符串,如果有则立即返回,否则就等待超时时间后返回;
- 只能捕捉由spawn启动的进程的输出;
- 用于接收命令执行后的输出,然后和期望的字符串匹配
-
send 用于向进程发送字符串
- 向进程发送字符串,用于模拟用户的输入;
- 该命令不能自动回车换行,一般要加\r (回车)或者\n
-
exp_continue 匹配多个字符串在执行动作后加此命令
exp_ continue 类似于控制语句中的continue 语句。表示允许expect 继续向下执行指令 -
expect eof
- 表示交互结束,等待执行结束,退回到原用户,与spawn对应
- 比如切换到root用户,expect脚本默认的是等待10s当执行完命令后,默认停留10s后,自动切回了原用户
-
interact 允许用户交互
- 会停留在目标终端而不会退回到原终端,这个时候就可以手工操作了,interact后的命.令不起作用;
- 比如interact后添加exit,并不会退出root用户。而如果没有interact则登录完成后会退出,而不是留在远程终端上。
- 使用interact会保持在终端而不会退回到原终端;
-
set
expect 默认的超时时间是10秒,通过set 命令可以设置会话超时时间,若不限制超时时间则应设置为-1,例如: set time out 30 -
send_users
表示回显命令与echo相同 -
接收参数
expect 脚本可以接受从bash命令行传递参数,使用 [lindex $argv n]获得。其中你从0开始,分别表示第一个,第二个,第三个…参数,例如:
set hostname [lindex $argv 0] 相当于hostname=$1
set password [lindex $argv 1] 相当于passswd=$2
2.实例
2.1 免交互su切换用户
vim expect.sh
#!/usr/bin/excpect #定义解释器
spawn su [lindex $argv 0] #追踪su命令
expect "密码" {send "XXXXXX(用户密码)"} #匹配到“密码”字段就输入密码
expect "]#" {send_user "ni hao"} #匹配到“]#”字符就输出ni hao
interact #将控制权交给控制台
2.2 ssh免交互
vim ssh.sh
#!/usr/bin/expect
spawn ssh 192.168.41.44 #追踪ssh 192.168.41.44
expect {
"yes/no" { send "yes\n";exp_continue } #如果是第一次进行ssh,会出现一段文字,询问要不要yes,这里是匹配第一次ssh的yes/no字段,当第二次再执行ssh时就不会出现询问字段,exp_continue的作用是匹配到了yes/no,就先输入yes,然后继续;如果是第二次ssh,就直接继续执行下面的指令。
"password" { send "XXXXXX\n"}
}
interact