1. [#!/usr/bin/expect]
這一行告訴作業系統腳本裡的代碼使用那一個shell來執行。這裡的expect其實和linux下的bash、windows下的cmd是一類東西。
注意:這一行需要在腳本的第一行。
2. [set timeout 30]
基本上認識英文的都知道這是設定逾時時間的,現在你隻要記住他的計時機關是:秒 。timeout -1 為永不逾時
3. [spawn ssh -l username 192.168.1.1]
spawn是進入expect環境後才可以執行的expect内部指令,如果沒有裝expect或者直接在預設的SHELL下執行是找不到spawn指令的。是以不要用 “which spawn“之類的指令去找spawn指令。好比windows裡的dir就是一個内部指令,這個指令由shell自帶,你無法找到一個dir.com 或 dir.exe 的可執行檔案。
它主要的功能是給ssh運作程序加個殼,用來傳遞互動指令。
4. [expect "password:"]
這裡的expect也是expect的一個内部指令,有點暈吧,expect的shell指令和内部指令是一樣的,但不是一個功能,習慣就好了。這個指令的意思是判斷上次輸出結果裡是否包含“password:”的字元串,如果有則立即傳回,否則就等待一段時間後傳回,這裡等待時長就是前面設定的30秒
5. [send "ispass\r"]
這裡就是執行互動動作,與手工輸入密碼的動作等效。
溫馨提示: 指令字元串結尾别忘記加上“\r”,如果出現異常等待的狀态可以核查一下。
6. [interact]
執行完成後保持互動狀态,把控制權交給控制台,這個時候就可以手工操作了。如果沒有這一句登入完成後會退出,而不是留在遠端終端上。如果你隻是登入過去執行
7.$argv 參數數組
expect腳本可以接受從bash傳遞過來的參數.可以使用[lindex $argv n]獲得,n從0開始,分别表示第一個,第二個,第三個....參數
下面的expect腳本的例子
執行這個檔案./launch.exp 1 2 3
螢幕上就會分别列印出參數
send_user用來發送内容給使用者。
參數運用方面還有很多技巧
比如$argc 存儲了參數個數,args被結構化成一個清單存在argv。$argv0 被初始化為腳本名字。
除此之外,如果你在第一行(#!那行)使用-d (debug參數),可以在運作的時候輸出一些很有用的資訊
比如你會看見
argv[0] = /usr/bin/expect argv[1] = -d argv[2] = ./launch.exp argv[3] = 1 argv[4] = 2 argv[5] = 3
使用這些也可以完成參數傳遞
8.expect的指令行參數參考了c語言的,與bash shell有點不一樣。其中,$argc為指令行參數的個數,$argv0為腳本名字本身,$argv為指令行參數。[lrange $argv 0 0]表示第1個參數,[lrange $argv 0 4]為第一個到第五個參數。與c語言不一樣的地方在于,$argv不包含腳本名字本身。
9.exp_continue的用法
<code>#!/usr/bin/expect -f</code>
<code>set</code> <code>ipaddr </code><code>"localhost"</code>
<code>set</code> <code>passwd </code><code>"iforgot"</code>
<code>spawn ssh root@$ipaddr #spawn 意思是執行指令,expect内指令,shell中不存在</code>
<code>expect {</code>
<code>"yes/no"</code> <code>{ send </code><code>"yes\r"</code><code>; exp_continue}</code>
<code>"password:"</code> <code>{ send </code><code>"$passwd\r"</code> <code>}</code>
<code>}</code>
<code>expect </code><code>"]# "</code>
<code>send </code><code>"touch a.txt\r"</code> <code>#意思為發送指令</code>
<code>send </code><code>"exit\r"</code>
<code>expect eof</code>
<code>exit</code>
exp_continue可以繼續執行下面的比對,簡單了許多。還有一點,讓我認識到比對不見得要比對最後幾個字元。
10.拿來小例子
設定變量 set PASSWD abcd123
<code># Expect script to supply root/admin password </code><code>for</code> <code>remote ssh server</code>
<code># and execute command.</code>
<code># This script needs three argument to(s) connect to remote server:</code>
<code># password = Password of remote UNIX server, </code><code>for</code> <code>root user.</code>
<code># ipaddr = IP Addreess of remote UNIX server, no hostname</code>
<code># scriptname = Path to remote script which will execute on remote server</code>
<code># If you username and passwd has not pass the rsa trust, your login will fail.</code>
<code># Usage For example:</code>
<code># ./sshlogin.exp password </code><code>192.168</code><code>.</code><code>1.11</code> <code>who</code>
<code># ------------------------------------------------------------------------</code>
<code># Copyright (c) </code><code>2004</code> <code>nixCraft project <http:</code><code>//cyberciti.biz/fb/></code>
<code># This script </code><code>is</code> <code>licensed under GNU GPL version </code><code>2.0</code> <code>or above</code>
<code># -------------------------------------------------------------------------</code>
<code># This script </code><code>is</code> <code>part of nixCraft shell script collection (NSSC)</code>
<code># Visit http:</code><code>//bash.cyberciti.biz/ for more information.</code>
<code># ----------------------------------------------------------------------</code>
<code># </code><code>set</code> <code>Variables</code>
<code>set</code> <code>password [lrange $argv </code><code>0</code> <code>0</code><code>]</code>
<code>set</code> <code>ipaddr [lrange $argv </code><code>1</code> <code>1</code><code>]</code>
<code>set</code> <code>scriptname [lrange $argv </code><code>2</code> <code>2</code><code>]</code>
<code>set</code> <code>arg1 [lrange $argv </code><code>3</code> <code>3</code><code>]</code>
<code>set</code> <code>timeout -</code><code>1</code>
<code># now connect to remote UNIX box (ipaddr) </code><code>with</code> <code>given script to execute</code>
<code>spawn ssh yourusername@$ipaddr $scriptname $arg1</code>
<code>match_max </code><code>100000</code>
<code># Look </code><code>for</code> <code>passwod prompt</code>
<code>expect </code><code>"*?assword:*"</code>
<code># Send password aka $password</code>
<code>send -- </code><code>"$password\r"</code>
<code># send blank line (\r) to make sure we </code><code>get</code> <code>back to gui</code>
<code>send -- </code><code>"\r"</code>
==============================================================================
<code>#!/usr/bin/expect </code>
<code> </code>
<code> </code><code># 設定逾時時間為 </code><code>60</code> <code>秒</code>
<code> </code><code>set</code> <code>timeout </code><code>60</code>
<code> </code><code># 設定要登入的主機 IP 位址</code>
<code> </code><code>set</code> <code>host </code><code>192.168</code><code>.</code><code>1.46</code>
<code> </code><code># 設定以什麼名字的使用者登入</code>
<code> </code><code>set</code> <code>name root </code>
<code> </code><code># 設定使用者名的登入密碼</code>
<code> </code><code>set</code> <code>password </code><code>123456</code>
<code> </code><code>#spawn 一個 ssh 登入程序</code>
<code> </code><code>spawn ssh $host -l $name </code>
<code> </code><code># 等待響應,第一次登入往往會提示是否永久儲存 RSA 到本機的 know hosts 清單中;等到回答後,在提示輸出密碼;之後就直接提示輸入密碼</code>
<code> </code><code>expect { </code>
<code> </code><code>"(yes/no)?"</code> <code>{ </code>
<code> </code><code>send </code><code>"yes\n"</code>
<code> </code><code>expect </code><code>"assword:"</code>
<code> </code><code>send </code><code>"$pasword\n"</code>
<code> </code><code>} </code>
<code> </code><code>"assword:"</code> <code>{ </code>
<code> </code><code>send </code><code>"$password\n"</code>
<code> </code><code>} </code>
<code> </code><code>expect </code><code>"#"</code>
<code> </code><code># 下面測試是否登入到 $host </code>
<code> </code><code>send </code><code>"uname\n"</code>
<code> </code><code>expect </code><code>"Linux"</code>
<code> </code><code>send_user </code><code>"Now you can do some operation on this terminal\n"</code>
<code> </code><code># 這裡使用了 interact 指令,使執行完程式後,使用者可以在 $host 終端進行互動操作。</code>
<code> </code><code>Interact</code>
用expect實作ssh自動登入對伺服器進行批量管理
1.實作ssh自動登入完成任務的expect腳本
<code>set</code> <code>ipaddress [lindex $argv </code><code>0</code><code>]</code>
<code>set</code> <code>passwd [lindex $argv </code><code>1</code><code>]</code>
<code>set</code> <code>timeout </code><code>30</code>
<code>spawn ssh shellqun@$ipaddress</code>
<code>"yes/no"</code> <code>{ send </code><code>"yes\r"</code><code>;exp_continue }</code>
<code>expect </code><code>"*from*"</code>
<code>send </code><code>"mkdir -p ./tmp/testfile\r"</code>
<code>#send </code><code>"exit\r"</code>
<code>expect </code><code>"#"</code> <code>指令運作完, 你要期待一個結果, 結果就是傳回shell提示符了(是# 或者$)</code>
<code>#最後一句第</code><code>13</code><code>行的解釋:</code>
其實寫成 interact 的最大好處是登入後不會退出,而會一直保持會話連接配接,可以後續手動處理其它任務,請根據實際情況自行選擇了。
2.調用login.exp完成批量管理
<code>#!/bin/bash</code>
<code>for</code> <code>i </code><code>in</code> <code>`awk </code><code>'{print $1}'</code> <code>passwd.txt`</code>
<code>do</code>
<code>j=`awk -v I=</code><code>"$i"</code> <code>'{if(I==$1)print $2}'</code> <code>passwd.txt`</code>
<code>expect /root/shell/login.exp $i $j</code>
<code>done</code>
<code>3</code><code>.passwd.txt</code>
<code>192.168</code><code>.</code><code>0.2</code> <code>password2</code>
<code>192.168</code><code>.</code><code>0.3</code> <code>password3</code>
<code> </code><code>13</code><code>.</code>
<code>"?assword:"</code> <code>{ </code>
<code> </code><code>#此大括号内是逐條執行,不存在</code><code>if</code><code>關系</code>
<code>send </code><code>"$PASSWORD\r"</code>
<code>exp_continue</code>
轉載:http://www.cnblogs.com/iloveyoucc/archive/2012/05/11/2496433.html
本文轉自奔跑在路上部落格51CTO部落格,原文連結http://blog.51cto.com/qiangsh/1850747如需轉載請自行聯系原作者
qianghong000