經過上面的一系列分析后,進入到獲取安裝包的步驟,作者在此處單獨寫了一個腳本,get_all.sh,我們繼續分析這個腳本
_ROOTDIR="$(pwd)"CONF_DIR="${_ROOTDIR}/../conf". ${CONF_DIR}/global. ${CONF_DIR}/core. ${CONF_DIR}/iredadmin
程序的意思是利用pwd獲取當前目錄,并定義為_ROOTDIR, 接著根據_ROOTDIR,獲得CONF目錄的具體位置。從而引入global, core ireadmin
iredadmin 文件里定義了一系列的關于版本和關于iredadmin的變量,估計在后面下載文件時可能會用到,如圖
接著分析。。。。
check_user rootcheck_hostnamecheck_runtime_dir
根據我的經驗,顯然check_user check_hostname check_runtime_dir 都是SHELL的函數,我們來一一分析
check_user 定義在conf/core文件中,具體的代碼如下:
check_user(){ # Check special user PRivilege to execute this script. if [ X"$(id -u)" != X"$(id -u ${1})" ]; then ECHO_ERROR "Please run this script as user: ${1}." exit 255 else if [ X"$(id -u)" == X"0" ]; then export PATH="/usr/kerberos/sbin:/usr/kerberos/bin:/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin:/root/bin" else : fi fi}
id -u 顯然是獲得登錄用戶的ID,在調用函數時,check_user root ,顯然是根據返回結果判斷是否為root,如果不是root,就返回錯誤。
exit 255 關于這個,大家參考這篇文章,http://www.CUOXin.com/tangdoudou/archive/2013/10/23/3385149.html
else之后的判斷不知道是否有點多余,不過作者可能是為了保險,再次判斷是否登錄用戶的ID為0,從而設置PATH的值
$PATH:決定了shell將到哪些目錄中尋找命令或程序,PATH的值是一系列目錄,當您運行一個程序時,linux在這些目錄下進行搜尋編譯鏈接。
目前設置的這個值里,比正常的多了/root/bin/,不知道這步在后面的程序中是否起著什么作用,暫時看不出來。
接著。。。
check_hostname 是分析服務器主機名是否符合規則的,代碼如下:
check_hostname(){ echo ${HOSTNAME} | grep '/.' &>/dev/null [ X"$?" != X"0" ] && / ECHO_ERROR "Please configure a fully qualified domain name (FQDN) in /etc/hosts before we go further./n/nExample:/n/n127.0.0.1 mail.iredmail.org mail localhost/n" && / exit 255}
程序的意思是利用${HOSTNAME}獲得主機名,如果主機名里沒. 就判斷主機名為不合理的,接著給出了例子,接著退出程序。
shell的函數都是比較簡單的,接著分析check_runtime_dir,代碼如下:
check_runtime_dir() { [ -d ${RUNTIME_DIR} ] || mkdir -p ${RUNTIME_DIR} # Cleanup rm -f ${RUNTIME_DIR}/.pkg_install_failed &>/dev/null}
-d 判斷目錄是否存在
mkdir –p 建立目錄
此處應該是先做判斷,目錄存在清理失敗文件,不存在建立目錄
三個分別檢查用戶,主機名,狀態目錄是否存在的函數執行完畢后,說明安裝條件適合,接下來的代碼定義了下載安裝包的地址:
export IREDMAIL_MIRROR="${IREDMAIL_MIRROR:=http://iredmail.org}"export PKG_DIR="${_ROOTDIR}/pkgs"export PKG_MISC_DIR="${_ROOTDIR}/misc"
http://iredmail.org 這是作者的官方主頁,也是下載安裝包的地址 。因為該腳本對多個主流的LINUX都做支持,所以下面這段代碼是針對多系統進行分析判斷的,看代碼
if [ X"${DISTRO}" == X"RHEL" ]; then # Special package. # command: which. export BIN_WHICH='which' export PKG_WHICH='which' # command: wget. export BIN_WGET='wget' export PKG_WGET='wget'elif [ X"${DISTRO}" == X"DEBIAN" -o X"${DISTRO}" == X"Ubuntu" ]; then if [ X"${OS_ARCH}" == X"x86_64" ]; then export pkg_arch='amd64' else export pkg_arch="${OS_ARCH}" fi # Special package. # command: which. export BIN_WHICH='which' export PKG_WHICH="debianutils" # command: wget. export BIN_WGET='wget' export PKG_WGET="wget" # command: dpkg-scanpackages. export BIN_CREATEREPO="dpkg-scanpackages" export PKG_CREATEREPO="dpkg-dev"fi
典型的條件判斷語句,不過這個里面最重要的參數就是${DISTRO}了,作者沒有把判斷他的值寫成一個函數,而是直接在CONF/global中直接通過程序判斷release來判斷為什么操作系統,
具體的位置如下截圖:
代碼比較長,同時又涉及到KERNEL的類型判斷,所以再寫一個日志,具體分析見下面鏈接:
DISTRO值的來源及操作系統的判斷
通過閱讀上面這篇文章,我們就可以知道DISTRO值的來源依據了,接著讀上面那段程序,可以得到當系統為RHEL時,
export BIN_WHICH='which' export PKG_WHICH='which' # command: wget. export BIN_WGET='wget' export PKG_WGET='wget'
得到這4個變量,在下文中下載軟件包時應該會用到。其他的操作系統歸為了一類,
elif [ X"${DISTRO}" == X"DEBIAN" -o X"${DISTRO}" == X"UBUNTU" ]; then
DEBIAN和UBUNTU從底層來說是一類LINUX,所以在這2個底層命令來說,是相同的,不過此處多了一個關于32位和64位的判斷, OS_ARCH的值從conf/global里面已經設置了,如圖
命令uname –m 可以判斷系統為32位或是64位,傳遞到此處的程序里,定義不同的pkg_arch的值。
if [ X"${OS_ARCH}" == X"x86_64" ]; then export pkg_arch='amd64' else export pkg_arch="${OS_ARCH}" fi
接下來又定義了如下變量
export BIN_WHICH='which' export PKG_WHICH="debianutils" # command: wget. export BIN_WGET='wget' export PKG_WGET="wget" # command: dpkg-scanpackages. export BIN_CREATEREPO="dpkg-scanpackages" export PKG_CREATEREPO="dpkg-dev"
這樣LINUX的判斷就結束了,下面又單獨定義了2個變量
# Binary packages.export pkg_total=$(echo ${PKGLIST} | wc -w | awk '{print $1}')export pkg_counter=1
${PKGLIST}此變量查找了半天,沒發現在哪定義的,只能在此留????
PKG_COUNTER PKG_TOTAL 單從單詞的意思來理解就是包的計數器和包的總數
# Misc file (source tarball) list.if [ X"${DISTRO}" == X"FREEBSD" ]; then PKGMISC='SHASUM.freebsd.misc'elif [ X"${DISTRO}" == X'OPENBSD' ]; then PKGMISC='md5.openbsd'else PKGMISC='MD5.misc'fiMISCLIST="$(cat ${_ROOTDIR}/${PKGMISC} | awk -F'misc/' '{print $2}')"
此處是關于FREEBSD的,是判斷MD5值的。MISC是標記的意思,也就是產生的標記文件。目錄在PKGS下面,有如下三個文件
打開其中一個文件,比如MD5.openbsd,如圖:
相應版本的MD5值都在里面,顯然是為了防止被篡改。
接下來是定義很多準備環境的函數,我們按照執行順序來分析具體函數的意義,作者此處如果放在某個公共文件里,可能比較合適些,所以我們再單開一個日志來分析這些函數
prepare_dirs()
fetch_misc()
check_md5()
create_repo_rhel()
check_new_iredmail()
echo_end_msg()
以上函數從其單詞意思可以得知分別為創建準備目錄,獲得標記, 檢測MD5,創建紅帽的REPO源,也就是YUM源,檢查新版本的IREDMAIL,輸出結束語句,下面這個鏈接對這些函數做詳細分析
工具函數分析
這些函數只是定義了,暫時未使用,接下來是分析狀態的判斷
if [ -e ${STATUS_FILE} ]; then . ${STATUS_FILE}else echo '' > ${STATUS_FILE}fi
if 判斷語句 –e 判斷狀態文件是否存在,如果存在就讀取,里面記錄了執行到了哪一步,如果不存在就創建,開始新的安裝。
接下來,檢查新版本,如下代碼
# Check latest version[ X"${CHECK_NEW_IREDMAIL}" != X'NO' ] && / check_status_before_run check_new_iredmail
判斷CHECK_NEW_IREDMAIL的值是否為NO,如果不為NO,則檢測新版本
不過作者在判斷了這個變量以后,又用check_status_before_run前面又進行了一次判斷,判斷檢查新版本的函數時,是否執行過,由于接下來的工具函數,作者都調用此函數進行檢查,所以在此做詳細解釋:
check_status_before_run(){ # If function was successfully executed, this function will write one line # in $STATUS_FILE: # # export status_[function_name]='DONE' # function_name="${1}" function_status_name="status_${function_name}" function_status_value="$(eval echo /$${function_status_name})"
#根據變量值判斷函數是否執行,保險措施
if [ X"${function_status_value}" == X"DONE" ]; then ECHO_SKIP "Function: $1." else $function_name #if [ X"$?" == X'0' ]; then # echo "export ${function_status_name}='DONE'" >> ${STATUS_FILE} #fi fi}
作者定義這個變量,可能是為了防止有些函數重復執行,雖然看著代碼有些重復。判斷是否更新版本后,接著執行
添加需要的目錄,添加完畢后,開始執行添加安裝源,代碼如下:
if [ X"${DISTRO}" == X"RHEL" ]; then # Create yum repository. check_status_before_run create_repo_rhel # Check required commands, install related package if command doesn't exist. check_pkg ${BIN_WHICH} ${PKG_WHICH} check_pkg ${BIN_WGET} ${PKG_WGET}elif [ X"${DISTRO}" == X'DEBIAN' -o X"${DISTRO}" == X'UBUNTU' ]; then # Force update. ECHO_INFO "Resynchronizing the package index files (apt-get update) ..." ${APTGET} updatefi
上述代碼中,有個check_pkg函數,需要解釋下,他的位置在conf/core中,
check_pkg函數解析
安裝需要的軟件,基礎環境完畢。接下來,執行下載標志文件,執行MD5檢測
check_status_before_run fetch_misc && /check_status_before_run check_md5 && /
函數在前面已經介紹過,不再解釋,接著往下走
check_pkg ${BIN_DIALOG} ${PKG_DIALOG} && /
check_pkg函數檢查命令,BIN_DIALOG, 定義在conf/global
dialog LINUX下的圖形編程工具,詳細介紹,如下:
http://www.ttlsa.com/linux-command/linux-dialog-shell/
終于要結束了
echo_end_msg && /echo 'export status_get_all="DONE"' >> ${STATUS_FILE}
新聞熱點
疑難解答