有一些需要自動輸入密碼的場景,可以用expect
自動與交互式程序會話
腳本
#!/usr/bin/expect -f
argv:接收參數
argc:參數個數
argv0:腳本名
選項
-c "cmd":指定在執行腳本之前執行的命令,這些命令最好用雙引號括起,防止被shell分開解釋,可以反覆使用-c
-d:輸出調試信息,報告expect和interact等命令執行時的內部行為,腳本開頭寫"exp_internal 1"也可以達到同樣的效果
-D:交互式調試器
-f:指定命令文件
-i:交互式提示輸入命令
--:劃定選項尾,如腳本中寫 #! /usr/bin/expect - ,則任何命令行的選項都會被解釋為參數,被argv接收
命令
setfile 「xxx」 設置參數file的值為"xxx"
close:關閉與當前進程的連結
expect和interact都可以檢查到進程的退出,隱含執行一個close命令,如果用exec kill pid方式殺進程,則需要顯示調用一次close指令
-slave 同時關閉從屬進程
-onexec 0 保持spawn_id 開啟,1關閉當前的spawn_id
-i 指定關閉的spawn_id
debug:調試
now立即啟動調試器,0停止,1啟動
disconnect:從終端斷開與克隆進程的連接,克隆進程會在後臺作為獨立進程組繼續運行,IO被重定向到/dev/null
send_user "password?\ "
expect_user -re "(.*)\n"
for {} 1 {} {
if [fork]!=0 {sleep 3600;continue}
disconnect #克隆進程繼續運行
spawn priv_prog
expect Password:
send "$expect_out(1,string)\r"
. . .
exit
}
Exp_continue[-continue_timer]
這個命令可以使expect繼續執行而不是正常的返回.默認情況下,exp_continue會重高超時時鐘,-continue_timer選項會阻止時鐘重新計數(連續計數).
Exp_internal [-f file] value
如果是value非零的話,使接下來的命令將調試信息輸出到Expect和標準錯誤輸出.如果是0的話,輸出的信息將會被屏蔽.調試信息包括收到的每條信息和每次嘗試用當前輸出與腳本中的模式相匹配的信息.如果設置了輸出文件,那么正常的和調試的信息都會被寫到這個文件當中.(忽略上面value選項的值).任何之前打開的調試輸出文件將會被關閉.-info選項使exp_internal返回最近關於non-info參數的描述
Exp_open [args] [-I spawn_id]
它返回對應於原始spawn id的文件描述符.這樣這個文件描述符就可以被使用了,就好像這個文件是被Tcl的open指令打開的一樣.(這個spawn id將不再使用,wait指令將不能用在這個進程.).-leaveopen選項使spawn id保持打開,以便供Expect命令使用
Exp_pid [-i spawn_id]
它將返回對應於當前被跟蹤進程的ID.如果使用-i選項,將返回對應於指定的spawn id的進程ID.
Exp_send:send別名
Exp_send_error:它是Send_error
Exp_send_log:Send_log
Exp_send_tty:Send_tty
Exp_send_user:Send_user
Exp_version [[-exit] version]
它用於確保腳本程序與當前的Expect兼容。在沒有參數的情況下,返回當前Expect的版本.這個版本就會編譯到腳本中.如果你確切的知道你的腳本程序不需要最新版本的特性,可以指定一個以前的版本
expect_tty [expect_args ],輸入是一個tty
expect_user [expect_args],輸入是stdin
expect [[-opts] pat1 body1] ... [-opts] patn [bodyn]
等待被監控進程的輸出,如果匹配到指定輸出字符串、遇到文件尾、或超時時,則執行對應的body體,如果body是空的,將被忽略
-nocase 匹配時不區分大小寫
-timeout:選項使得Expect使用選項後面的數值做為超時時間,而不是timeout變量中設置的時間
expect_befor模式,在所有匹配之前隱含調用
expect_after模式,在所有匹配之後隱含調用
expect {
busy {puts busy\n ; exp_continue} # 輸出值有busy執行{}裡面的語句
failed abort # 輸出有failed,執行abort,abort是預定義的一個函數
"invalid password" abort #輸出字符串中間有空格,需要用雙引號括起來
-re "failed|invalid password" abort #使用正則表達式
-ex "failed|invalid *password" abort #精準正則,不對* ^等轉義
-i $proc2 busy {puts busy\n ; exp_continue} #匹配$proc2進程的輸出,默認是當前進程的輸出
-i $proc3 # 匹配到執行XXX
-i $proc4 # 匹配到執行XXX
any_spawn_id XXX
timeout abort #超時時觸發
full_buffer abort #輸出的值超過最大的match_max設置的值是觸發
null #輸出ascii 0時觸發
connected
}
匹配時,任何輸出都會被保存到expect_out緩衝區中,匹配到的9個字串分別被保存在expect_out(1, string) ~ expect_out(9, string),如果在模式前面使用了 -indeces 選項,則這9個字串的起始位置和結束位置保存在expect_out(X, start)和expect(X, end)中,X∈[0, 9]。expect(0,*)是匹配到的整個字符串,下面三條語句等效
expect "cd"
Set expect_out(0,string) cd
Set expect_out(buffer) abcd
-i選項還可以定義一個全局變量,裡面存儲著spawn_id列.當變量內容發生變化時,它會被重新讀取.這樣就可以在程序執行的時候改變I/O源.以這種方式提供的spawn_id被稱為」indirect spawn_id」
Fork
複製一個進程,返回新進程ID,失敗返回-1
interact [string1 body1] ... [stringn [bodyn]]
將控制權給用戶,根據用戶的輸入,觸發相應的動作
set CTRLZ \032
interact {
-reset $CTRLZ {exec kill -STOP [pid]} # 按下ctrl+z觸發
\001 {send_user "you typed a control-A\n"; # 發送一個字符串到終端
send "\001"
}
$ {send_user "The date is [exec date]."}
\003 exit
foo {send_user "bar"} # 用戶輸入foo時觸發
~~
}
send:發送字符串給當前進程
send_user:發送字符串到標準輸出
send_error:發送字符串到標準錯誤
send_log:發送字符串到日誌文件
send_tty:發送字符串到tty
sleep seconds:休眠
spawn [args] program [args]
創建一個執行program命令的進程,這個進程的三個標準IO都被重定向到expect,
spawn su - ftomcat -c "tar -xzf /tmp/ftomcat.tgz . 1>/dev/null 2>/dev/null"
expect "Password"
send "ftomcat\r"
expect eof
exit