Shell编程--免交互

发布于:2023-01-21 ⋅ 阅读:(377) ⋅ 点赞:(0)

Here Document

  • 使用 I/O 重定向的方式将命令列表提供给交互式程序或命令,比如 ftp、cat 或 read 命令。
  • HereDocument是标准输入的一种替代品,可以帮助脚本开发人员不必使用临时文件来构建输入信息,而是直接就地生产出一个文件并用作命令的标准输入。

基本语法格式

特殊字符“<<”在标记和命令之前,这样做的目的是将命令块的输出重定向到程序或命令的 stdin。
标记的选择要确保不会出现在其他地方,避免出现混淆;
两个标记之间的内容被当做是一个文件并用作“命令”的标准输入。
另外 Here Document 也可以与非交互式程序和命令一起使用。
 

四点注意事项:

  • 标记可以使用任意的合法字符
  • 结尾的标记一定要顶格写,前面不能有任何字符
  • 结尾的标记后面也不能有任何字符(包括空格)
  • 开头的标记前后的空格会被省略

 

 

Expect 免交互

expect 是由Don Libes基于Tcl(Tool Command Language )语言开发的,主要应用于自动化交互式操作的场景,是一个免费的编程工具,用来实现自动的交互式任务,而无需人为干预。说白了,expect就是一套用来实现自动交互功能的软件。

环境安装

expect 并非是 shell 能够解释的,所以想要执行它来完成你的脚本,你需要先安装两个软件包,当然,在使用yum的情况下,我们执行这条命令就可以了

脚本解释器

expect 脚本中首先引入文件,表明使用的是哪一个shell

#!/usr/bin/expect
 

spawn

spawn 后面通常跟一个Linux执行命令,表示开启一个会话、启动进程,并跟踪后续交互信息

spawn passwd root


expect

  • 判断上次输入结果中是否包含指定的字符串,如果有则立即返回,否则就等待超时时间后返回
  • 只能捕捉由 spawn 启动的进程的输出
  • 用于接收命令执行后的输出,然后和期望的字符串匹配

send

向进程发送字符串,用于模拟用户的输入,该命令不能自动回车换行,一般要加 \r (回车) 或者 \n

expect "密码" {send "abc123\r"}   
或
expect "密码"
send "abc123\r"

#同一行send部分要有{}   换行send部分不需要有{}
expect          
"密码1"{ send "abc123\r"}
"密码2"{ send "123456\r"}
"密码3"{ send "123123\r"}

#expect支持多个分支,只要匹配了其中一个情况,执行相应的send语句后退出该expect语句

结束符

expect eof

表示交互结束,等待执行结束,退回到原用户,与spawn对应

比如:切换到root用户,expect脚本默认的是等待10s,当执行完命令后,默认停留10s后,就会自动切回原用户。

interact

执行完成后保持交互状态,把控制权交给控制台,会停留在目标终端,这个时候就可以手工操作了,interact 后的命令不起作用
比如 interact 会保持在终端而不会退回到原终端,比如切换到 root 用户,会一直在 root 用户状态下
比如2 ssh 到另一服务器,会一直在目标服务器终端,而不会切回的原服务器
 

set

expect 默认的超时时间是10秒,通过 set 命令可以设置会话超时时间,若不限制超时时间则应设置为-1

set timeout 20

exp_continue

  • exp_continue 附加于某个 expect 判断项之后,可以使该项匹配后,还能继续匹配该 expect 判断语句内的其他项
  • exp_continue 类似于控制语句中 continue 语句,表示允许 expect 继续向下执行指令

例如,将判断交互输出中是否存在 yes/no 或 *password , 如果匹配 yes/no 则输出 yes 并再次执行判断;如果匹配 *password 则输出 123123 并结束该段 expect 语句,操作如下:

 expect
"(yes/no)" {send "yes\r"; exp_ continue; }
"*password" {set timeout 300; send "abc123\r";}

SSH服务

#!/bin/bash
net=192.168.100
password=123123
iplist="
101
102
"

for i  in $iplist
do
ip=$net.$i

/usr/bin/expect <<EOF

spawn ssh root@$ip

expect {
        "(yes/no)"
        {send "yes\r";exp_continue}
        "*password"
        {send "$password\r"}
}
expect "*]#" { send "useradd test\n" }
expect "*]#" { send "exit\r"   }
expect eof
EOF
done