iredmail是一套以postfix为核心的整合邮件系统的安装脚本,可以达到快速部署邮件服务器的目的。为了让自己不遗忘shell的语法,所以闲来无事,学习一下他的代码。
我从官网下载他的最新版,解压后,结构如下:
iRedMail.sh 是开始安装的启动文件,执行bash iRedMail.sh 后就开始安装,我们从这个文件看起。
tmprootdir="$(dirname $0)"
dirname的用法是:
(1) 带全路径的,例:aguo@auto-test:~> dirname /home/aguo/insert.sql
/home/aguo #获得文件所在目录
(2) 仅仅是文件名的,例:aguo@auto-test:~> dirname Environment
. 表示路径就是当前目录
$0 在shell中的意思是当前执行脚本的名称
一个命令涉及了3个知识点,看来学习还是能加深印象的。
第二个命令如下:
echo ${tmprootdir} | grep '^/' >/dev/null >&
| 管道符 ,管道符左边命令的输出就会作为管道符右边命令的输入。连续使用管道意味着第一个命令的输出会作为 第二个命令的输入,第二个命令的输出又会作为第三个命令的输入,依此类推。
grep ‘^/’ 查找以/开头的,grep的正则表达式用法
>/dev/null >&1 :
可以将/dev/null看作"黑洞". 它非常等价于一个只写文件. 所有写入它的内容都会永远丢失. 而尝试从它那儿读取内容则什么也读不到
1 表示stdout标准输出,系统默认值是1,所以">/dev/null"等同于"1>/dev/null"
2 表示stderr标准错误
& 表示等同于的意思,2>&1,表示2的输出重定向等同于1
很多shell脚本里都会用到,实际作用就是不让输出任何内容
这个命令的整体作用就是为了查找变量tmprootdir的值中以/开头的,下面会根据状态码进行判断
if [ X"$?" == X"" ]; then
export ROOTDIR="${tmprootdir}"
else
export ROOTDIR="$(pwd)"
fi
export export命令可以用来设置或显示环境变量。在shell中,当一个变量创建时,只会在当前的shell中有 效,之后创建的子进程不会有该变量。而命令export可以将一个shell变量变成环境变量,在随后创建的子进程中也能访问到该环境变量,但是子进程在 修改了环境变量的值之后,父进程并不能知道,因为子进程将复制父进程的环境变量。因此,当shell退出时,那些在该shell中export出来的环境 变量也会消失。
这样的话,接下来的脚本中,ROOTDIR的值就会保留下来
确定了脚本所在的系统的绝对路径后,后面就可以对一些函数库和全局变量进行调用了,接下来的代码:
cd ${ROOTDIR} export CONF_DIR="${ROOTDIR}/conf"
export FUNCTIONS_DIR="${ROOTDIR}/functions"
export DIALOG_DIR="${ROOTDIR}/dialog"
export PKG_DIR="${ROOTDIR}/pkgs/pkgs"
export PKG_MISC_DIR="${ROOTDIR}/pkgs/misc"
export SAMPLE_DIR="${ROOTDIR}/samples"
export PATCH_DIR="${ROOTDIR}/patches"
export TOOLS_DIR="${ROOTDIR}/tools" . ${CONF_DIR}/global
. ${CONF_DIR}/core
基本包含了系统以后要用到的所有配置文件和具体的变量文件,global是全局的变量文件 core是核心文件,后面会继续分析,从语法看没什么难的,只有. 需要解释一下
. ${CONF_DIR}/global 这样就把里面的变量或者函数都引入到了脚本的运行中,后面遇到具体的变量再做分析。
# Check downloaded packages, pkg repository.
[ -f ${STATUS_FILE} ] && . ${STATUS_FILE}
${STATUS_FILE} 变量定义的文件是conf/global中,具体位置如下:
从图中可以看出,该变量的值来自另一变量,继续拿出来分析:
export RUNTIME_DIR="${ROOTDIR}/runtime"
export STATUS_FILE="${RUNTIME_DIR}/install.status"
${ROOTDIR}上面已经分析过了,这是脚本执行的绝对路径,从${STATUS_FILE}词的意思可以理解为这个变量是安装状态的变量,作者定义它是为了判断安装状态。${RUNTIME_DIR},也就是所有后面软件安装时的状态都会以文件形式存放在这个目录里。
[-f ${STATUS_FILE} ] -f filename 如果 filename为常规文件,则为真 [ -f /usr/bin/grep ]
&& &&左边的命令(命令1)返回真(即返回0,成功被执行)后,&&右边的命令(命令2)才能够被执行
显然这句的意思是判断这个状态文件是否存在,如果存在,把该文件里的一些变量值读入到进程中,我觉的这是为了解决安装过程中断后,再次安装时,可以继续安装而设置的。继续下面的语句
if [ X"${status_get_all}" != X"DONE" ]; then
cd ${ROOTDIR}/pkgs/ && bash get_all.sh
if [ X"$?" == X'' ]; then
cd ${ROOTDIR}
else
exit
fi
fi
程序中的${status_get_all},如果这个变量的值不是DONE,那么程序就会进入到PKGS里,执行get_all.sh,文件的具体分析见下面:
执行完上述脚本,获得了安装包,我们继续分析代码如下:
chmod go+rx /dev/null /dev/*random &>/dev/null
作者在此加了注释,给这些文件读写权限,是为了防止某些系统不支持,而导致安装过程中断。
check_env函数解析