使用I/O重定向的方式将命令列表提供给交互式程序
标准输入的一种替代品
Here Document 是标准输 入的一种替代品,可以帮助脚本开发人员不必使用临时文件来构建输入信息,而是直接就地 生产出一个文件并用作命令的标准输入,Here Document 可以与非交互式程序和命令一起使用
命令<< 标记> 文件
....
......
>标记
1 标记可以使用任意的合法字符(通用的字符是EOF)
2 结尾的标记一定要顶格写,前面不能有任何字符(包括空格)
3 结尾的标记后面也不能有任何字符(包括空格)
4 开头标记前后空格会被省略掉
EOF 标记之间的两行是输入的密码和确认密码,两行内容必须保持一致,否则密 码设置不成功。此脚本执行后不会输出任何信息,可另开一个终端使用 jerry 用户登录,输 入新修改的密码来验证密码是否修改正确
[root@localhost ~]#vim mcb4.repo.sh
[root@localhost ~]#bash mcb4.repo.sh
[root@localhost ~]#cat mcb4.repo.sh
#!/bin/bash
cat > /http.repo <
-a
-i
tee 与cat 区别
tee直接生成文件,cat >需要导一下
是建立在tcl(tool command language)语言基础上的一个工具,常被用于进行自动化控制和测试,解决shell脚本中交互的相关问题
spawn ##表示开启监控、捕捉模式,后面跟着命令,如ssh,scp等
expect ##表示开始捕捉,即是从进程开始接收字符串,后面是花括号,表示捕捉的内容{}
send ##表示遇到捕捉的关键字,send免交互要做的事,即向进程发送字符串
exp_continue ##表示继续捕捉,一般是连续的操作,比如向ssh需要先回答yes,然后继续捕捉password
expect eof ##表示停留10秒以后,自动退出,与spawn相对应
interact ##表示停留,不再退出
expect 脚本中首先引入文件,表明使用的事哪一种shell
#!/usr/bin/expect
spawn 后面通常跟一个Linux执行命令,表示开启一个会话、进程,并跟踪后续交互信息
例: spawn passwd root
判断上次输出结果中是否包含指定的字符串,如果有则立即返回,否则就等待超时时间后返回;只能捕捉有swpan启动的进程输出;用于接受命令执行后的输出,然后和期望的字符串匹配
向进程发送字符串,用于模拟用户的输入,该命令不能自动回车换行,要加 \r (回车) 或者\ n
学习相关的命令
#!/usr/bin/expect
spawn scp /etc/passwd 192.168.11.10:/opt
#监控scp命令,出现scp命令开始捕捉屏幕内容
expect {
"yes/no" {send "yes\n";exp_continue}
"password" {send "123\n";}
}
#捕捉屏幕上的关键字,出现yes/no输入yes
#exp_continue 代表继续捕捉password
#出现 password 输入密码。 \n \r 回车
expect eof
(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 set ip 192.168.20.30 set user root
定义变量
[root@localhost ~]# str=abc123我好
[root@localhost ~]# echo ${#str}
8
经典案例
[root@localhost ~]#echo {a..z}
a b c d e f g h i j k l m n o p q r s t u v w x y z
[root@localhost ~]#echo {a..z}|tr -d " " 去空格
abcdefghijklmnopqrstuvwxyz
[root@localhost ~]#str=`echo {a..z}|tr -d " "` 定义变量
[root@localhost ~]#echo $str 输出变量
abcdefghijklmnopqrstuvwxyz
[root@localhost ~]#echo ${#str}
26
[root@localhost ~]#echo ${str:3:-3} #跳过最前面的 三个字符, 和最后三个字符
defghijklmnopqrstuvw
[root@localhost ~]#echo ${str: -6:-3}
#倒数6个到 倒数3个
uvw
[root@localhost ~]#url="http://www.baidu.com/index.html"
[root@localhost ~]#echo ${url#*/}
#从左边开始匹配,遇到/ 就把/左边的都删除包括 / 本身
/www.baidu.com/index.html
[root@localhost ~]#url="http://www,baidu.com:8080/index.html"
[root@localhost ~]#echo ${url#*:}
//www,baidu.com:8080/index.html
[root@localhost ~]#echo $url
http://www,baidu.com:8080/index.html
[root@localhost ~]#echo ${url%:*}
#懒惰模式, 遇到: 会把左边删除留下右边
http://www,baidu.com
[root@localhost ~]#echo ${url%%:*}
#贪婪模式, 会一直找到所有的 : 会把左边删除留下右边
http
url trl 是个变量,任意赋值
[root@localhost ~]#echo ${url/http/https}
#把 http换成 https
https://www,baidu.com:8080/index.html
[root@localhost ~]#echo ${url}
http://www,baidu.com:8080/index.html
案例继续刨根问底
[root@localhost ~]#echo ${user^^}
LISI:X:1003:1003::/HOME/LISI:/BIN/BASH
[root@localhost ~]#echo ${user,,}
lisi:x:1003:1003::/home/lisi:/bin/bash