Linux Shell 自動互動功能

2022-08-06 18:00:31

需求背景:

  近日,在安裝某軟體流程,發現在安裝過程需要輸入一些資訊才能繼續下一步操作,在機器數量較少情況下,我們可以單臺登入上去完成安裝操作,但當機器數量超過一定時,如果再手動登入操作,就會產生大量重複性操作,既不能帶來有效學習能力提升,同時也會極大產生不確定性,引發工作效率下降,那麼如何自動化完成某些操作呢,尤其是帶有互動功能的步驟呢,例如需要輸入賬號密碼?

如有興趣轉載,請標註來源:https://www.cnblogs.com/xiong97/p/16557437.html  謝謝!

1. EOF 多文字輸入

需求案例 1

  新交付了一批機器,每臺機器只分配了一塊落盤 ,現在根據需求對該盤進行分割區並實現掛載,如何實現?

需求分析:

對於一個盤,實現分割區掛載到不同目錄,通常思路有兩條:

方法一: 將整塊盤作為一個PV ,整合成VG卷,再根據劃分不同LV卷大小分給不同目錄

方法二: 通過fdisk 將盤直接分割成對應需求的大小,再對磁碟初始化,完成掛載

方案解決

這裡我們為了演示互動功能,選擇方法二,實現指令碼如下:

#!/bin/bash
fdisk /dev/sdb <<EOF
n
p
1
wq
EOF
 
 mkfs.xfs /dev/sdb1 &&  mkdir -p /data && mount /dev/sdb1 /data
echo '/dev/sdb1 /data xfs defaults 0 2' >> /etc/fstab

 分析上述指令碼,我們發現使用了 一個關鍵字 EOF

EOF是END Of File的縮寫,表示自定義終止符.既然自定義,那麼EOF就不是固定的,可以隨意設定別名,在linux按ctrl-d就代表EOF.
EOF一般會配合cat能夠多行文字輸出.
其用法如下:
<<EOF        //開始

....                //需要輸入的內容

EOF            //結束

例如使用 cat、<<、EOF>以互動方式編寫bash指令碼,如下所示。

cat << EOF > script.sh
#!/bin/bash

printf "Hello\n"
printf "Wordl!\n"
EOF

合理 利用這三個,即可以完成對應多文字互動輸入,例如修改使用者密碼,正常情況下,需要連續輸入兩次密碼,兩次密碼一致才能修改成功,如下:

上面我們學會了EOF 這個關鍵字,那麼我們試試通過它來修改密碼。指令碼如下:

#!/bin/bash

cat << EOF| passwd 
新密碼
新密碼,與上述需一致
EOF

# or 不使用管道符

passwd << EOF
新密碼
新密碼,與上述需一致
EOF

實戰結果,成功修改密碼:

2. Expect  自動互動

需求案例 2

  新交付了一批機器,需要給每臺機器分發檔案,如何實現?

需求分析:

   遠端拷貝檔案常用密令是scp 或者 rsync ,但是在給每臺機器傳輸時需要若輸入密碼,有的機器可能還需要輸入YES,錄入機器指紋資訊,如下:

Expect  是在tcl基礎上的一個自動化互動套件, 在一些需要互動輸入指令的場景下, 可通過指令碼設定自動進行互動通訊. 其互動流程主要有以下5步:

0 定義變數

1 spawn啟動指定指令碼或命令

2  expect匹配結果關鍵詞

3  send針對指定關鍵詞傳送指定指令

4  執行完成, 退出

但可惜的是os預設沒有安裝,因此需要先安裝才能使用

Expect is a tcl application for automating and testing  interactive applications such as telnet, ftp, passwd, fsck, rlogin, tip, etc. Expect makes it easy for a script to control another program and interact with it.

方案解決:

  1. 先檢查本機是否安裝了expect ,如果沒有安裝,需要手動安裝

# 檢查是否安裝了tcl:
[root@localhost ~]# rpm -qi expect
Name        : expect
Version     : 5.45
Release     : 14.el7_1
Architecture: x86_64
Install Date: Fri 05 Aug 2022 07:26:04 AM CST
Group       : Development/Languages
....

# 如果沒有安裝, 使用yum安裝expect ,通常會順帶把依賴包tcl 也安裝了:
[root@localhost ~]# yum install -y expect  # -y  其實也是安裝過程中一個互動,發現沒,只是作為引數傳入了
[root@localhost ~]# yum install -y tcl  # 如果上述命令提示已安裝tcl了,此步可以忽略

# 檢視expect的安裝路徑:
[root@localhost ~]#  which expect
/usr/bin/expect

 2 .對應功能指令碼開發,本案例指令碼參考如下:

[root@test01 ~]# cat scp.exp
#! /usr/bin/expect
set file [lindex $argv 0]
set file2 [lindex $argv 1]
spawn scp -rp $file $file2 [email protected]:/tmp
expect {
 "(yes/no)" {send "yes\r";exp_continue}
 "*password:*" {send "Password\r"}

}
expect eof
exit -onexit {
 send_user "bye \n"
}

3. 分析上述指令碼,有幾個點需要說明

#!/usr/bin/expect

指令碼檔案的第一行指明expect 安裝位置,具體可以參考2 中命令檢視,指明指令碼解析器,和Shell類似,表示程式使用Expect解析,這裡與一般bash 指令碼不同,因此需要注意,通常我們會將expect指令碼字尾修改成exp來和bash 指令碼 sh區別

set 設定變數值

set file [lindex $argv 0] 將傳入的第一個引數賦給file ,類似第二、三個引數[lindex $argv 1] [lindex $argv 2] 等,後續呼叫時使用 $file ,和shell 一樣。特殊引數:

$argc表示傳參的個數,$argv0表示指令碼的名字

 spawn  表名要執行的指令碼或程式命令,如ssh、scp等

格式: spawn [選項] [需要自動互動的命令或程式]

例如:spawn scp -rp $file $file2 [email protected]:/tmp #<==執行scp命令(注意開頭必須要有spawn, 否則無法實現互動)

expect 

需和spawn 配合使用 ,表示匹配spawn指定的指令碼或命令的輸出結果,如果與expect後面的字串匹配,就執行下面的send命令,表示對結果響應反饋

有時命令的輸出提示資訊有可能會變化,所以可以在expect中使用模糊匹配,比如*

注意:匹配的動作也可以放在下一行,這樣就不需要使用{}(大括號)了

send

在expect命令匹配指定的字串後,傳送指定的字串給系統,這些命令可以支援一些特殊跳脫符號,例如:\r表示回車、\n表示換行、\t表示製表符等

exp_continue

從命令的拼寫就可以看出命令的作用,即讓Expect程式繼續匹配的意思,如果需要一次匹配多個字串,那麼不同的匹配之間就要加上exp_continue,否則expect將不會自動輸入指定的字串。最後一個的結尾就不需要加上exp_continue了,因為前面都已完成了,它是最後一個啦

exit

功能類似於Shell中的exit,即直接退出指令碼,還可以利用這個命令對指令碼做一些關閉前提示等工作

send_user

列印Expect指令碼資訊,類似Shell裡的echo. 例如列印變數資訊,驗證資料傳入是否正常

在掌握expect 基本使用方式後,我們寫一個批次檢視機器負載資訊的小指令碼,加強記憶

#! /usr/bin/expect
    set time 30
    set ip [lindex $argv 0]

    spawn ssh root@$ip uptime
    expect {
        "*yes/no" { send "yes\r"; exp_continue }
        "*password:" { send "$password\r" }
    }
    expect eof

實戰結果: