Shell编程免交互及相应场景配置解析

Here Document

Here Document概述

  • 使用I/O重定向的方式将命令列表提供给交互程序
  • 标准输入的一种代替品
  • 语法格式:
    命令 <<标记


    标记

Here Document使用注意事项

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

Here Document免交互

示例1,通过read命令接受输入并打印

[root@server2 ~]# vi aa.sh
#!/bin/bash
# read免交互
read i <<T
hello!       
T
echo $i
[root@server2 ~]# chmod +x aa.sh
[root@server2 ~]# ./aa.sh
hello!  #输出

例2,通过passwd给用户设置密码

[root@server2 ~]# useradd lisi    #需有账户
[root@server2 ~]# vi passwd.sh
#!/bin/ bash
#给用户设置密码
passwd lisi <<A
abc123      #因为设置密码需要输入两次,所以输入两遍
abc123
A
[root@server2 ~]# chmod +x passwd.sh
[root@server2 ~]# ./passwd.sh
更改用户 lisi 的密码 。
新的 密码:无效的密码: 密码少于 7 个字符
重新输入新的 密码:passwd:所有的身份验证令牌已经成功更新。

Here Document变量设定

  • 变量替换
[root@server2 ~]# vi file.sh
a=/root/a.txt
b=nihao
cat > $a <<EOF      #重定向给文件a
HELLO $b
EOF
[root@server2 ~]# chmod +x file.sh
[root@server2 ~]# ./file.sh
[root@server2 ~]# cat a.txt
HELLO nihao #输出
  • 变量设定
[root@server2 ~]# vi var.sh
#!/bin/bash
# here document的值赋给变量
var=a
b=$(cat <<EOF
this is a book.
i have a dog.
$var
EOF
)
echo $b
[root@server2 ~]# chmod +x var.sh
[root@server2 ~]# ./var.sh
this is a book. i have a dog. a#输出

Here Document格式控制

  • 关闭变量替换功能
[root@server2 ~]# vi var.sh
#!/bin/bash
# here document的值赋给变量
var=a
b=$(cat <<'EOF'       #选项加单引号
this is a book.
i have a dog.
$var            #看定义的变量是否被引用
EOF
)
echo $b
[root@server2 ~]# chmod +x var.sh
[root@server2 ~]# ./var.sh
this is a book. i have a dog. $var  #输出   #变量未被引用
  • 去除每行之前的TAB字符
[root@server2 ~]# vi abc.sh
#!/bin/bash
cat <<-'EOF'
ABC123     #不是空格,是TAB字符
EOF
[root@server2 ~]# chmod +x abc.sh
[root@server2 ~]# ./abc.sh
ABC123 #输出

Here Document多行注释

通过Here Document方式使Bash支持多行注释

语法格式:
:<<DO-NOTHING
第一行注释
第二行注释
....
DO-NOTHING

示例

[root@server2 ~]# vi test1.sh
#!/bin/bash
# 注释
:<<E
ni hao
hello
lisi
E
echo zhangsan
[root@server2 ~]# chmod +x test1.sh
[root@server2 ~]# ./test1.sh
zhangsan

Expect

Expect概述

  • 建立在tcl(基本语言工具)之上的一个工具
  • 用于进行自动化控制和测试(屏幕捕捉)
  • 解决shell脚本中交互相关的问题

Expect安装

[root@server2 ~]# yum -y install expect    #yum源安装
[root@server2 ~]# rpm -qa | grep ecpect   #检查是否安装
expect-5.45-14.e17_1.x86_64       

Expect执行方式

直接执行

在这里插入[root@server2 ~]# vi test1.sh
#!/usr/bin/expect     #不再是/bin/bash,直接是expect,后面的命令结构必须是expect结构不然无法识别
代码片

嵌入执行

[root@server2 ~]# vi test1.sh
#!/bin/bash
zhangsan=$1
lisi=$2
usr/bin/expect<<-EOF   #定义完变量后,启用expect命令解释接下来的语法
...
EOF    #expect结束
...     #后面还能接着打/bin/bash能识别的命令

基础命令

  • spawn #发起进程

启动进程,并跟踪后续交互信息

  • expect #屏幕捕捉,匹配关键词

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

  • send #匹配关键词成功,输出字符串,不能自动换行,加 \r (回车) 或 \n

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

  • 结束符

expect eof
等待执行结束
interact
执行完成后保持交互状态,把控制权交给控制台

  • set

设置超时时间,过期则继续执行后续指令
单位是秒
timeout-1表示永不超时
默认情况下,timeout是10秒

  • exp_continue

允许expect继续向下执行指令

  • send_user

回显命令,相当于echo

  • 接受参数

Expect脚本可以接受从bash传递的参数
可以使用[lindex $argv n]获得
n从0开始,分别表示第一个,第二个,第三个…参数
lindex $argv 0=$1

Expect语法

单一分支语法

expect "password: " {send "mypassword\r"}

多分支模式语法

示例1

expect "aaa" {send "AAA\r"}
expect "bbb" {send “BBB\r"}
expect "ccc" {send "CCC\r"}

示例2

expect {
"aaa" {send "AAA\r"}
"bbb" {send “BBB\r"}
"ccc" {send "CCC\r"}
}

示例3

expect {
"aaa" {send "AAA\r"; exp_continue}
"bbb" {send “BBB\r"; exp_continue}
"ccc" {send "CCC\r"; exp_continue}
}

项目操作

创建用户并设置密码

[root@server2 ~]# yum -y install expect   #先安装Expect
[root@server2 ~]# vi expect1.sh
#!/bin/bash
# 新建用户设置密码的免交互
hostname=$1
password=$2
useradd $hostname      
/usr/bin/expect <<-EOF    #启用expect命令
spawn passwd $hostname     #发起进程
expect "新的" {send "$password\r"}    #捕捉关键词‘新的’,输出变量$password
expect "重新输入新的" {send "$password\r"}   #捕捉关键词‘重新输入新的’,输出变量$password
expect eof                  #允许向下执行
EOF
[root@server2 ~]# chmod +x expect1.sh 
[root@server2 ~]# ./expect1.sh ag1 123456     #新建账户密码

ssh登录免交互

未指定帐号密码,登录时需带上ip地址及密码

需先生成密钥面交互成再登录免交互

[root@server2 ~]# yum -y install expect   #先安装Expect
[root@server2 ~]# vi sshmy.sh
#!/bin/bash
# ssh 生成密钥的免交互及登录免交互
/usr/bin/expect <<-EOF
spawn ssh-keygen -t rsa       #发起生成密钥进程
expect "(/root/.ssh/id_rsa)" {send "\r"}     #捕捉关键词输出换行
expect "(empty for no passphrase)" {send "\r"}    #捕捉关键词输出换行
expect "again" {send "\r"}    #捕捉关键词输出换行
expect eof      #允许向下执行
EOF
/usr/bin/expect <<-EOF
spawn ssh root@$hostname     #发起登录免交互进程
expect "yes/no" {send "yes\r"}        #捕捉关键词输出yes换行
expect "password" {send "$password\r"}    #捕捉关键词输出密码换行
expect "*]#" {send "exit\r"}      #最后退出
expect eof       #允许向下执行
EOF
[root@server2 ~]# chmod +x sshmy.sh 
[root@server2 ~]# ./sshmy.sh 20.0.0.10 123456

指定ip地址及密码,登录时无需带上帐号密码

[root@server2 ~]# yum -y install expect   #先安装Expect
[root@server2 ~]# vi sshmy.sh
#!/bin/bash
# 免密登录
yum -y install expect &> /dev/null
/usr/bin/expect<<-EOF
spawn ssh-keygen -t rsa      #发起生成密钥进程
expect "/root/.ssh/id_rsa" {send "\r"}
expect "empty for no passphrase" {send "\r"}
expect "passphrase again" {send "\r"}
expect eof
EOF
/usr/bin/expect <<-EOF
spawn ssh-copy-id 192.168.100.110       #密钥发给客户机
expect "yes/no" {send "yes\r"}
expect "root@192.168.100.110's password" {send "123123\r"}
expect eof
EOF
ssh root@192.168.100.110
[root@server2 ~]# chmod +x sshmy.sh 
[root@server2 ~]# ./sshmy.sh  #直接登录上,本地会生成.ssh/文件,下次免密登录前需要删除
[root@server2 ~]# rm -rf .ssh/   #下次再免密登录时需提前删除

利用expect完成FTP登录过程

先配置ftp服务,再设置ftp登录免交互

服务器配置ftp服务

[root@server2 ~]# vi nmftp.sh    
#!/bin/bash
# ftp安装和匿名用户登录
A=vsftpd
B=/etc/vsftpd/vsftpd.conf
yum -y install $A         #安装ftp服务
sed -i -e '/local_enable/s/YES/NO/g' $B           #修改为不允许本地账户登录
sed -i -e '/listen/s/NO/YES/g' $B                 #修改为监听IPV4地址的请求
sed -i -e '/listen_ipv6/s/YES/NO/g' $B               #修改为不监听IPV6地址的请求
sed -i -e 's/^#anon_upload_enable=YES/anon_upload_enable=YES/g' $B    #修改为允许匿名账户上传
sed -i -e 's/^#anon_mkdir_write_enable=YES/anon_mkdir_write_enable=YES/g' $B   #修改为允许匿名账户新建文件或目录
sed -i '$aanon_umask=022' $B        #添加设置匿名账户的权限为755
sed -i '$aanon_other_write_enable=YES' $B       #添加允许匿名账户的其他写权限
systemctl start $A          #启动服务
netstat -anpt | grep $A          #检查是否启动
grep -v "#" $B | grep -v "^$"     #检查配置文件

设置权限,创建文件用来测试匿名用户

[ root@server1 ~]# cd /var/ftp/ pub                                #进目录创建文件
[ root@server1 pub]# echo "abc123" > a.txt                  #创建a.txt文件内容abc123
[ root@server1 pub]# echo "bcd123" > b.txt                  #创建b.txt文件内容bcd123
[ root@server1 ~]# chown -R ftp.ftp /var/ftp/pub            #设置文件归属
[ root@server1 ~]# chmod -R 755 /var/ftp/pub               #设置文件目录权限
[ root@server1 pub]# ls -lh 

客户机免密登录

[root@client1 ~]# vi ftpdl.sh
#!/bin/bash
# ftp一键登录
hostname=$1
password=$2
/usr/bin/expect <<-EOF
spawn ftp $hostname              #发起ftp登录进程
expect "Name" {send "ftp\r"}            #捕捉关键词输出ftp换行
expect "Password:" {send "$password\r"}    #捕捉关键词输出变量$password换行
expect "ftp>" {send "ls -lh\r"}       #捕捉关键词输出ls -lh换行,查看服务器文件
interact
EOF
[root@client1 ~]# chmod +x ftpdl.sh 
[root@client1 ~]# ./ftpdl.sh 20.0.0.11 123456

磁盘初始化

新添加一块磁盘
在这里插入图片描述

[root@client1 ~]# ls /dev/sd*    #查看下磁盘详细信息

在这里插入图片描述

[root@client1 ~]# fdisk /dev/sdb     #新建磁盘

在这里插入图片描述
初始化一块磁盘,格式化,挂载

[root@client1 ~]# vi cppz.sh
#!/bin/bash
# 初始化一块磁盘,格式化,挂载
 /usr/bin/expect<<-EOF
 spawn fdisk /dev/sdb               #发起新建磁盘进程
 expect {
     "命令(输入 m 获取帮助)" {send "n\r";exp_continue}          #捕捉关键词...,输出...
     "Select (default p)" {send "p\r";exp_continue}
     "分区号 (1-4,默认 1)" {send "1\r";exp_continue}
     "起始 扇区 (2048-41943039,默认为 2048)" {send "\r";exp_continue}
     "+size{K,M,G}" {send "\r"}}
     expect "命令(输入 m 获取帮助)" {send "p\r";send "wq\r";exp_continue}
EOF
fdisk -l                     #查看新建磁盘
mkdir /data           
mkfs.ext4 /dev/sdb1         #格式化
mount /dev/sdb1  /data     #挂载
df -Th              #查看挂载
[root@client1 ~]# chmod +x cppz.sh 
[root@client1 ~]# ./cppz.sh 

还原初始状态
[root@client1 ~]# umount /data/    #卸载挂载
[root@client1 ~]# rm -rf /data/    删除挂载点
[root@client1 ~]# fdisk -d /dev/sdb     删除磁盘
[root@client1 ~]# vi cpgs.sh
#!/bin/bash
a=`ls /dev/sd* | grep -o sd[b-z] | uniq`      #显示出/dev/sda/外的所有磁盘
for var in $a
do
   echo -e "n\np\n\n\n\nwq\n" | fdisk /dev/$var   支持转义字符
   mkfs.ext4 /dev/${var}"1"                             重分发
   mkdir -p /data/${var}"1"                             创建
   mount /dev/${var}"1" /data/${var}"1"                 挂载
done
[root@client1 ~]# chmod +x cpgs.sh 
[root@client1 ~]# ./cpgs.sh 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值