Linux - 通過expect工具實現指令碼的自動互動
目錄
- 1 安裝expect工具
- 2 expect的常用命令
- 3 作用原理簡介
- 3.1 示例指令碼
- 3.2 指令碼功能解讀
- 4 其他指令碼使用示例
- 4.1 直接通過expect執行多條命令
- 4.2 通過shell呼叫expect執行多條命令
- 5 spawn not found 的解決
1 安裝expect工具
expect
是建立在tcl基礎上的一個自動化互動套件, 在一些需要互動輸入指令的場景下, 可通過指令碼設定自動進行互動通訊. 其互動流程是:
spawn啟動指定程序 -> expect獲取指定關鍵字 -> send想指定程序傳送指定指令 -> 執行完成, 退出.
由於expect
是基於tcl的, 所以需要確保系統中安裝了tcl:
# 檢查是否安裝了tcl: [root@localhost ~]# whereis tcl tcl: /usr/lib64/tcl8.5 /usr/include/tcl.h /usr/share/tcl8.5 # 如果沒有安裝, 使用yum安裝tcl和expect: [root@localhost ~]# yum install -y tcl [root@localhost ~]# yum install -y expect # 檢視expect的安裝路徑: [root@localhost ~]# command -v expect /usr/bin/expect
2 expect的常用命令
命 令 | 說 明 |
---|---|
spawn | 啟動新的互動程序, 後面跟命令或者指定程式 |
expect | 從程序中接收資訊, 如果匹配成功, 就執行expect後的動作 |
send | 向程序傳送字串 |
send exp_send | 用於傳送指定的字串資訊 |
exp_continue | 在expect中多次匹配就需要用到 |
send_user | 用來列印輸出 相當於shell中的echo |
interact | 允許使用者互動 |
exit | 退出expect指令碼 |
eof | expect執行結束, 退出 |
set | 定義變數 |
puts | 輸出變數 |
set timeout | 設定超時時間 |
3 作用原理簡介
3.1 示例指令碼
這裡以ssh遠端登入某臺伺服器的指令碼為例進行說明, 假設此指令碼名稱為remote_login.sh
:
#!/usr/bin/expect
set timeout 30
spawn ssh -l root 172.16.22.131
expect "password*"
send "123456\r"
interact
3.2 指令碼功能解讀
(1) #!/usr/bin/expect
上述內容必須位於指令碼檔案的第一行, 用來告訴作業系統, 此指令碼需要使用系統的哪個指令碼解析引擎來執行.
具體路徑可通過command -v expect
命令檢視.
注意:
這裡的expect和Linux的bash、Windows的cmd等程式一樣, 都是一種指令碼執行引擎.
指令碼需要有可執行許可權(
chmod +x remote_login.sh
, 或chmod 755 auto_login.sh
), 然後通過命令./remote_login.sh
執行即可;如果輸入
sh remote_login.sh
, 意義就不一樣了: 明確呼叫sh
引擎去執行此指令碼, 此時首行的#!/usr/bin/expect
就失效了.
(2) set timeout 30
設定連線的超時時間為30秒.
(3) spawn ssh -l root 172.16.22.131
spawn、send等命令是expect工具中的內部命令, 如果沒有安裝expect工具, 就會出現"spawn not found"等錯誤.
不要用
which spawn
之類的命令去找spawn, 因為並沒有這樣的程式.
(4) expect "password*"
這個命令用來判斷上次輸出結果裡是否包含"password*"的字串, 如果有則立即返回, 否則就等待一段時間後返回. 這裡的等待時長就是前面設定的timeout, 也就是30秒.
(5) send "123456\r"
這裡就是執行互動動作, 作用等同於手工輸入密碼.
提示: 命令字串結尾加上\r
, 這樣的話, 如果出現異常等待的狀態就能夠停留下來, 作進一步的核查.
(6) interact
expect執行完成後保持使用者的互動狀態, 這個時候使用者就可以手工操作了.
如果沒有這一句, expect執行完成後就會退出指令碼剛剛遠端登入過去的終端, 使用者也就不能繼續操作了.
4 其他指令碼使用示例
4.1 直接通過expect執行多條命令
注意首行內容, 這種情況下就只能通過./script.sh
來執行這類指令碼了:
#!/usr/bin/expect -f
set timeout 10
# 切換到root使用者, 然後執行ls和df命令:
spawn su - root
expect "Password*"
send "123456\r"
expect "]*" # 萬用字元
send "ls\r"
expect "#*" # 萬用字元的另一種形式
send "df -Th\r"
send "exit\r" # 退出spawn開啟的程序
expect eof # 退出此expect互動程式
4.2 通過shell呼叫expect執行多條命令
注意首行內容, 這種情況下可通過sh script.sh
、bash script.sh
或./script.sh
, 都可以執行這類指令碼:
#!/bin/bash
ip="172.16.22.131"
username="root"
password="123456"
# 指定執行引擎
/usr/bin/expect <<EOF
set time 30
spawn ssh $username@$ip df -Th
expect {
"*yes/no" { send "yes\r"; exp_continue }
"*password:" { send "$password\r" }
}
expect eof
EOF
5 spawn not found 的解決
出現這個錯誤的基本上都是出學者: Linux 執行shell指令碼有兩種方式:
一種是將指令碼作為sh的命令列引數, 如
sh remote_login.sh
, 或sh /data/remote_login.sh
;一種是將指令碼作為具有執行許可權的可執行指令碼, 如
./remote_login.sh
, 或/data/remote_login.sh
.
而作為sh命令列引數來執行, 那麼指令碼第一行的#!/usr/bin/expect
就會失效, 所以才會出現spawn not found
、send not found
等錯誤, 所有上面的automate_login.sh指令碼必須用以下命令執行:
./automate_expect.sh
參考資料
Linux expect 介紹和用法
expect spawn、linux expect 用法小記
版權宣告
作者: 瘦風(https://healchow.com)
出處: 部落格園 瘦風的部落格(https://www.cnblogs.com/shoufeng)
感謝閱讀, 如果文章有幫助或啟發到你, 點個[好文要頂