批量複製及執行命令shell指令碼
平時在處理一個或幾個機器執行環境時,一個機器一個機器處理也能接受,但是如果是一批機器,幾十或幾百臺,要是一臺一臺去安裝環境,光是輸入同一的命令,估計你自己都想吐,所有聰明的人會想一些偷懶的辦法,確實可以找到一些省時省力的方法,比如寫一個批量處理shell指令碼,這幾天在處理一批(八九十臺)機器環境,找了一些批量處理的指令碼,包括批量傳輸(scp)檔案到多臺機器上、批量執行命令到多臺機器、還有需要互動的命令,下面記錄一些這些命令:
機器IP檔案:ip.txt
192.168.10.201 192.168.10.202 192.168.10.203 192.168.10.204 192.168.10.205
另外所有機器要可以從一臺機器通過ssh無密登入其他機器
1.批量傳輸(scp)檔案到一批機器上
建立指令碼檔案,並修改為可執行檔案
touch xscp.sh
chmod +x xscp.sh
指令碼命令:
#!/bin/bash for line in `cat $1` do if [ "$3" == "" ] then # dir echo scp -r $2/ $line:$2/ scp -r $2/ $line:$2/ else # files echo scp ${@:3} $line:$2/ scp ${@:3} $line:$2/ fi done
示例1:指定要傳輸的檔案
./xscp.sh ip.txt /opt/soft/ filename1 filename2 ...
第一個引數($1):ip.txt 是一個以換行符分割的IP集合,每個IP佔一行
第二個引數($2):/opt/soft/ 檔案要傳輸到目標機器的目錄
第三個及以後n個引數(${@:3}):為要傳輸的檔案或檔案集合
示例2:不指定檔案
./xscp.sh ip.txt /opt/soft/
如果不指定要傳輸的檔案,會預設傳輸第二個引數($2)目錄(/opt/soft)下的所有檔案,到目標機器對應的目錄(/opt/soft)下
2.批量執行命令指令碼(無互動)
建立指令碼檔案,並修改為可執行檔案
touch xcall.sh chmod +x xcall.sh
指令碼命令:
#!/bin/bash params=${@:2} for line in `cat $1` do echo "============= {print %s, $line} $params =============" ssh $line "$params" done
示例1:
./xcall.sh ip.txt hostname
第一個引數($1):ip.txt 要通過ssh執行命令的機器ip集合
第二個及後續n個引數(${@:2}):視為要執行的整個命令,最好整個命令用雙引號引起,如下示例
示例2:
./xcall.sh ip.txt "rpm -qa | grep lzo"
3.批量執行命令指令碼(有互動)
有些時候批量向其他機器執行指令碼,需要互動,比如安裝軟體時,需要確認(輸入yes/no),需要輸入密碼等,如果人工輸入,上百臺機器估計夠費勁,所有在執行批量指令碼時,讓其自動互動,顯得很實用,下面以expect來實現自動互動互動舉幾個小示例。
expect是一個自動互動功能的工具,可以滿足代替我們實際工作中需要從終端手動輸入某些內容來使得程式或命令繼續執行的目的。如安裝軟體是時的一些提示,ssh遠端主機執行命令時需要多次輸入密碼的情況。
首先在執行指令碼的機器上要安裝expect軟體包
yum -y install expect
一些expect基本命令:
- spawn :啟動新程序,用於執行shell命令;
- expect :從發起互動的命令的程序接受字串,用於匹配我們預想的字串;
- send :用於向發起互動的命令的程序傳送字串;
- interact:允許使用者互動,即此命令後,互動將不會由expect進行,將交回給使用者;
示例1:
建立指令碼檔案並修改為可執行檔案:
touch xcall-keytool-list.sh chmod +x xcall-keytool-list.sh
指令碼命令:
#!/bin/bash params=${@:2} for line in `cat $1` do echo "============ $line $params =============" /usr/bin/expect << EOF set time 20 spawn ssh $line "$params" expect { "*password:" { send "changeit\r" } } expect eof EOF done
命令執行 在查詢java證書庫時需要輸入java證書庫密碼(changeit):
./xcall-keytool-list.sh ip.txt "/home/software/java/bin/keytool -list -keystore /home/software/java/jre/lib/security/cacerts | grep baidu"
第一個引數($1):ip.txt 要執行命令的機器IP集合檔案
第二個及後面n個引數(${@:2}):為要執行的命令
示例2:更為通用的指令碼命令
建立指令碼檔案並修改為可執行檔案:
touch xcall-interaction.sh chmod +x xcall-interaction.sh
指令碼命令:
#!/bin/bash # IP檔案 iptxt=$1 # 要執行的命令 command=$2 # 互動時輸入項(yes/no|password) initem=$3 # 互動輸入內容(yes | 密碼) input=$4 echo "--- command:$command ---" echo "--- 輸入項:$initem ---" echo "--- 輸入內容:$input ---" for line in `cat $iptxt` do echo "============ $line $command =============" if [ "$initem" = "" ] then ssh $line "$command" else /usr/bin/expect << EOF set time 20 spawn ssh $line "$command" expect { "*$initem:" { send "$input\r" } } expect eof EOF fi done
執行命令:
./xcall-interaction.sh ip.txt "/home/software/java/bin/keytool -list -keystore /home/software/java/jre/lib/security/cacerts | grep baidu" password changeit
第一個引數($1):ip.txt 要執行命令的機器IP集合檔案
第二個引數($2):為要執行的命令,用雙引號
第三個引數($3):為輸入項,這裡是password(輸入密碼),或者輸入yes/no等,如果沒有第三個引數,則當成非互動命令執行
第四個引數($4):為輸入項要輸入的內容,如輸入項是password時,此項輸入密碼;如果輸入項是yes/no時,一般輸