Diskimage-builder 是openstack社区用于制作镜像的工具.为了深入了解dib制作镜像的全过程,对一个简单的例子进行贯通的分析。
从日志分析DIB流程-1
1. 导入方法和Expand element
日志 1:76 行
首先是导入了一下方法,这些方法再后面都会见到,类似于c语言中的include和python中的import,主要是为了方便组织方法的编写.所以这里不详述
作为参数传入的元素只有vm和ubuntu,但是这些元素可能依赖于其他元素,所有要把所有被依赖的元素找出来.
如何找到这些依赖的元素不是本篇的重点,本篇只分析在制作进行的过程中发生了什么.
2. 创建制作进行目录
日志 77:105 行
Building in /tmp/dib_build.Y0FGIa3a
对应这行日志的方法是:
TMP_BUILD_DIR=$(mktemp -t -d --tmpdir=${TMP_DIR:-/tmp} dib_build.XXXXXXXX)
TMP_IMAGE_DIR=$(mktemp -t -d --tmpdir=${TMP_DIR:-/tmp} dib_image.XXXXXXXX)
[ $? -eq 0 ] || die "Failed to create tmp directory"
export TMP_BUILD_DIR
if tmpfs_check ; then
sudo mount -t tmpfs tmpfs $TMP_BUILD_DIR
sudo mount -t tmpfs tmpfs $TMP_IMAGE_DIR
sudo chown $(id -u):$(id -g) $TMP_BUILD_DIR $TMP_IMAGE_DIR
fi
trap trap_cleanup EXIT
echo Building in $TMP_BUILD_DIR
export TMP_IMAGE_PATH=$TMP_IMAGE_DIR/image.raw
export OUT_IMAGE_PATH=$TMP_IMAGE_PATH
export TMP_HOOKS_PATH=$TMP_BUILD_DIR/hooks
mktemp是用于创建临时目录的(可以看做是随机生产文件名)
然后利用mount命令在这里挂载了tmpfs,tmpfs实际上是存在于内存中的(一般情况下)
然后到这里有几个目录:
TMP_BUILD_DIR : 镜像是在这个目录下面制作的
TMP_IMAGE_DIR : 最终镜像存放在这里
OUT_IMAGE_PATH: 和IMAGE_DIR是一样的
TMP_HOOKS_PATH: 所有的钩子(脚本)被拷贝到的地方.
3. root阶段入口方法
root阶段准备好了制作进行所需的"base"
root阶段的顶层方法如下:
mkdir $TMP_BUILD_DIR/mnt
export TMP_MOUNT_PATH=$TMP_BUILD_DIR/mnt
# Copy data in to the root.
TARGET_ROOT=$TMP_MOUNT_PATH run_d root
if [ -z "$(ls $TMP_MOUNT_PATH | grep -v '^lost+found\|tmp$')" ] ; then
# No root element copied in. Note the test above allows
# root.d elements to put things in /tmp
echo "Failed to deploy the root element."
exit 1
fi
# Configure Image
# Setup resolv.conf so we can chroot to install some packages
if [ -L $TMP_MOUNT_PATH/etc/resolv.conf ] || [ -f $TMP_MOUNT_PATH/etc/resolv.conf ] ; then
sudo mv $TMP_MOUNT_PATH/etc/resolv.conf $TMP_MOUNT_PATH/etc/resolv.conf.ORIG
fi
# Recreate resolv.conf
sudo touch $TMP_MOUNT_PATH/etc/resolv.conf
sudo chmod 777 $TMP_MOUNT_PATH/etc/resolv.conf
# use system configured resolv.conf if available to support internal proxy resolving
if [ -e /etc/resolv.conf ]; then
cat /etc/resolv.conf > $TMP_MOUNT_PATH/etc/resolv.conf
else
echo nameserver 8.8.8.8 > $TMP_MOUNT_PATH/etc/resolv.conf
fi
mount_proc_dev_sys
这个方法做的事情如下:
创建mnt目录和配置DNS都顾名思义,这里的mount_proc_dev中执行了如下命令:
注意,如果root阶段没有建立好base不会进行到proc和dev的挂载
sudo mount -t proc none $TMP_MOUNT_PATH/proc
sudo mount --bind /dev $TMP_MOUNT_PATH/dev
sudo mount --bind /dev/pts $TMP_MOUNT_PATH/dev/pts
sudo mount -t sysfs none $TMP_MOUNT_PATH/sys
mount –bind的用法可以查看“bind mount 的用法”
mount -t proc/sysfs 是指以proc/sysfs 的类型进行挂载,这是chroot中较为常用的用法,可以参考arch linux chroot
以上这几步在所有的dib的过程中都是相同的,下面来解析这个例子中root步骤到底干了啥
3.1 run_d root
run_d 是dib中的一个方法,这个方法是所有过程(pharse)的入口
run_d root表示运行root阶段的所有脚本,这里仅仅只是一个参数名
run_d方法的流程图如下:
过程比较简单:提供了两个debug的断点,生产钩子文件夹,运行脚本.
其中generate_hooks流程如下:
generate会一次性将所有需要的脚本按照阶段放到钩子文件夹中.
dib-run-parts流程图如下:
dib-run-parts首先会寻找 environment.d文件夹,将其中的环境变量导入
然后以此运行该阶段的脚本
之前了解了run_d方法的流程,但是实际上完成root阶段工作的还是root阶段的脚本.
在下一篇中会详细分析例子中的root阶段运行的脚本.