Shell编程之免交互

目录

引言

一、Here Document免交互

1、Here Document概述

2、基本语法格式

3、特殊字符“<<”

4、注意事项

二、Here Document常规用法

1、直接把输入的从命令行输出

2、输入保存到文件

3、追加输入保存到文件

4、输入保存到文件

5、创建yum源

6、通过 read 命令接收输入并打印

7、通过 passwd 给用户设置密码

8、将变量替换成实际值

9、整体赋值给一个变量

10、关闭变量替换的功能

11、标记前面加‘-’,表示要抑制各行首 TAB 的作用

12、":"代表什么都不做的空命令

二、Expect

1、expect概述

2、基本命令

四、Expect常规用法

1、ssh无交互登录到远程服务器

2、如果想要在对方服务器上进行一下操作后再退出可执行以下脚本

3、用set关键字定义变量,变量名和变量的值中间用空格分开,其他用法与shell脚本一致

4、创建用户并设置用户密码

总结


引言

在编写shell脚本的过程中,一般会遇到一些操作时,需要用到指定的参数,这个时候使用免交互,就能快捷方便。

一、Here Document免交互

1、Here Document概述

(1)使用I/O重定向的方式将命令列表提供给交互式程序或命令,比如 ftp、cat 或 read 命令。

(2)标准输入的一种替代品,可以帮助脚本开发人员不必使用临时文件来构建输入信息,而是直接就地生产出一个文件并用作命令的标准输入。

2、基本语法格式

命令 <<标记
......
(内容)
......
标记

3、特殊字符“<<”

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

4、注意事项

(1)标记可以使用任意的合法字符;

(2)结尾的标记一定要顶格写,前面不能有任何字符;

(3)结尾的标记后面也不能有任何字符(包括空格);

(4)开头的标记前后的空格会被省略。

二、Here Document常规用法

1、直接把输入的从命令行输出

[root@localhost ~]# cat <<EOF> hello world
> EOF
hello world

[root@localhost ~]# cat <<EOF
> hello
> world
> EOF
hello
world

2、输入保存到文件

[root@localhost ~]# cat > name.txt <<EOF
> lisi
> zhangsan
> wangwu
> EOF
[root@localhost ~]# cat name.txt 
lisi
zhangsan
wangwu

3、追加输入保存到文件

[root@localhost ~]# cat >> name.txt  <<EOF
> zhaosi
> liuda
> EOF
[root@localhost ~]# cat name.txt 
lisi
zhangsan
wangwu
zhaosi
liuda

4、输入保存到文件

[root@localhost file]# tee test1.txt <<EOF
> zhangsan
> lisi
> wangwu
> EOF
zhangsan
lisi
wangwu
[root@localhost file]# cat test1.txt 
zhangsan
lisi
wangwu

5、创建yum源

[root@localhost file]# cat >local.repo <<EOF
> [local]
> name=local
> baseurl=file:///mnt
> enbaled=1
> gpgcheck=0
> EOF
[root@localhost file]# cat local.repo 
[local]
name=local
baseurl=file:///mnt
enbaled=1
gpgcheck=0




[root@localhost file]# cat <<EOF > local1.repo 
> [local]
> name=local
> baseurl=file:///mnt
> enbaled=1
> gpgcheck=0
> EOF
[root@localhost file]# cat local.repo 
[local]
name=local
baseurl=file:///mnt
enbaled=1
gpgcheck=0

6、通过 read 命令接收输入并打印

[root@localhost ~]# read a <<EOF
> 10
> EOF
[root@localhost ~]# echo $a
10

7、通过 passwd 给用户设置密码

[root@localhost file]# useradd lisi
[root@localhost file]# passwd lisi <<EOF
> 123456
> 123456
> EOF
更改用户 lisi 的密码 。
新的 密码:无效的密码: 密码少于 8 个字符
重新输入新的 密码:passwd:所有的身份验证令牌已经成功更新。

8、将变量替换成实际值

[root@localhost file]# aa=$(cat <<EOF
> $a
> EOF
> )
[root@localhost file]# echo $aa
123

[root@localhost file]# vim test1.sh
#!/bin/bash
doc_file="file1.txt"
i="file1"
cat > $doc_file <<EOF
请将软件放入到 $i
EOF
[root@localhost file]# sh test1.sh 
[root@localhost file]# cat file1.txt 
请将软件放入到 file1

9、整体赋值给一个变量

[root@localhost file]# vim TX.sh
#!/bin/bash
ivar="Great! Beautyful!"
myvar=$(cat <<EOF     #(将Here Document 整体赋值给变量)
this is Line 1.
tan are Sun, Moon and starts.
$ivar     #(输出时会进行变量替换)
EOF
)
echo $myvar
[root@localhost file]# sh TX.sh 
this is Line 1. tan are Sun, Moon and starts. Great! Beautyful!

10、关闭变量替换的功能

[root@localhost file]# aa=$(cat <<'EOF'
> 20
> $a
> EOF
> )
[root@localhost file]# echo $aa
20 $a
[root@localhost file]# echo $a
123

11、标记前面加‘-’,表示要抑制各行首 TAB 的作用

[root@localhost file]# vim TX2.sh
#!/bin/bash
cat <<EOF     
        hello world
EOF

cat <<-EOF
        hello world
EOF
[root@localhost file]# sh TX2.sh 
	hello world
hello world

12、":"代表什么都不做的空命令

[root@localhost file]# vim TX3.sh
#!/bin/bash
:  cat <<-EOF
the second comment.
test line.
EOF
echo "请下载软件!"
[root@localhost file]# sh TX3.sh 
请下载软件!

二、Expect

1、expect概述

(1)建立在tcl语言基础上的一个工具,常被用于进行自动化控制和测试,解决shell脚本中交互相关的问题。

(2)expect它是一个程序,所以它也是需要提前安装才可以使用的。

[root@localhost file]# rpm -q expext
未安装软件包 expext 
[root@localhost file]# rpm -q tcl
未安装软件包 tcl 
[root@localhost file]# yum install expect  tcl -y
[root@localhost ~]# rpm -q expect
expect-5.45-14.el7_1.x86_64
[root@localhost ~]# rpm -q tcl
tcl-8.5.13-8.el7.x86_64
[root@localhost ~]# which expect 
/usr/bin/expect
[root@localhost ~]# which tcl
/usr/bin/which: no tcl in (/usr/local/sbin:/usr/local/bin:/usr/sb
in:/usr/bin:/root/bin)

2、基本命令

(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 或*password。如果匹配 yes/no则输出 yes 并再次执行判断;如果匹配*password则输出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

四、Expect常规用法

1、ssh无交互登录到远程服务器

[root@localhost /]# vim dome1.sh
#!/usr/bin/expect     #(这里需要用expect自己的解释器,注意不要写成bash否则无法识别)

spawn ssh root@192.168.32.130     #(开启一个程序,这个程序是ssh远程登录)

expect {
        "password:"
        { send "123456\r"; }     #(捕获内容,当出现password的时候,就会向程序发送密码,默认是不换行的,所以需要\r回车换行,多个条件需要花括号括起来,注意格式问题!)
}

interact     #(交互,否则会直接退出远程服务器)

:wq

[root@localhost /]# chmod +x dome1.sh 
[root@localhost /]# ./dome1.sh 
spawn ssh root@192.168.32.130
root@192.168.32.130's password: 
Last login: Wed Sep 15 12:22:52 2021 from 192.168.32.1

2、如果想要在对方服务器上进行一下操作后再退出可执行以下脚本

[root@localhost /]# vim dome2.sh
#!/usr/bin/expect

spawn ssh root@192.168.32.130

expect {
        "password:"
        { send "123456\r"; }
}

expect "#"     #(当捕获到#的时候)
send "ls\r"    #(执行ls命令)
send "ifconfig ens33\r"     #(执行ifconfig ens33命令)
send "exit\r"     #(执行完exit退出登陆)

expect eof     #(不需要进行交互了,意味着结束expect程序,如果不写的话不会执行操作直接退出;如果不写,写interact也无法在对方机器上执行命令,eof可以替换
)

:wq

[root@localhost ~]# chmod +x dome2.sh 
[root@localhost ~]# ./dome2.sh 
spawn ssh root@192.168.32.130
root@192.168.32.130's password: 
Last login: Wed Sep 15 12:35:08 2021 from 192.168.32.128
[root@localhost ~]# ls
1 = 0                 letest.txt  web.sh       公共  下载
1.txt                 lstest.txt  wjf          模板  音乐
2.txt                 test.txt    xxx.tar.bz2  视频  桌面
anaconda-ks.cfg       tmp.tar.gz  xxx.tar.gz   图片
initial-setup-ks.cfg  WEBJK.sh    yum.conf     文档
[root@localhost ~]# ifconfig ens33
ens33: flags=4163<UP,BROADCAST,RUNNING,MULTICAST>  mtu 1500
        inet 192.168.32.130  netmask 255.255.255.0  broadcast 192
.168.32.255        inet6 fe80::c5b8:3bdf:ea88:c9a3  prefixlen 64  scopeid 0x
20<link>        ether 00:0c:29:80:36:1a  txqueuelen 1000  (Ethernet)
        RX packets 64  bytes 7167 (6.9 KiB)
        RX errors 0  dropped 0  overruns 0  frame 0
        TX packets 62  bytes 10322 (10.0 KiB)
        TX errors 0  dropped 0 overruns 0  carrier 0  collisions 
0
[root@localhost ~]# exit
登出
Connection to 192.168.32.130 closed.

3、用set关键字定义变量,变量名和变量的值中间用空格分开,其他用法与shell脚本一致

[root@localhost /]# vim dome3.sh
#!/usr/bin/expect
set user root
set ip [lindex $argv 0]     #(设置第一个位置变量为ip)
set pass [lindex $argv 1]     #(设置第二个位置变量为登陆密码)

spawn ssh $user@$ip

expect {
        "password:"
        { send "$pass\r"; }

}

expect "#"
send "ls\r"
send "exit\r"

expect eof
                                                              
:wq  

[root@localhost /]# chmod +x dome3.sh 
[root@localhost /]# ./dome3.sh 192.168.32.130 123456     #(执行时需要加位置变量)
spawn ssh root@192.168.32.130
root@192.168.32.130's password: 
Last login: Wed Sep 15 12:24:59 2021 from 192.168.32.128
[root@localhost ~]# ls
1 = 0                 letest.txt  web.sh       公共  下载
1.txt                 lstest.txt  wjf          模板  音乐
2.txt                 test.txt    xxx.tar.bz2  视频  桌面
anaconda-ks.cfg       tmp.tar.gz  xxx.tar.gz   图片
initial-setup-ks.cfg  WEBJK.sh    yum.conf     文档
[root@localhost ~]# exit
登出
Connection to 192.168.32.130 closed.

4、创建用户并设置用户密码

[root@localhost ~]# vim dome4.sh
#!/bin/bash
username=$1

useradd $username

/usr/bin/expect <<-EOF

spawn passwd $username

expect {
        "新的 密码"
        { send "123456\r";exp_continue }
        "重新输入新的 密码"
        { send "123456\r"; }
}
EOF

:wq

[root@localhost ~]# ./dome4.sh liwu
useradd:用户“liwu”已存在
spawn passwd liwu
更改用户 liwu 的密码 。
新的 密码:
无效的密码: 密码少于 8 个字符
重新输入新的 密码:
passwd:所有的身份验证令牌已经成功更新。

总结

1、免交互是使用I/O重定向的方式将命令列表提供给交互式程序或命令。
2、免交互是标准输入的一种替代品。
3、建立在tcl语言基础上的一个工具,常被用于进行自动化控制和测试,解决shell脚本中交互相关的问题。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值