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