经过上面的一系列分析后,进入到获取安装包的步骤,作者在此处单独写了一个脚本,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}
新闻热点
疑难解答